diff --git a/Config/FilterPlugin.ini b/Config/FilterPlugin.ini index 61919c3f..20f78e07 100644 --- a/Config/FilterPlugin.ini +++ b/Config/FilterPlugin.ini @@ -6,10 +6,15 @@ ; /README.txt ; /Extras/... ; /Binaries/ThirdParty/*.dll -/Doc/*.adoc +/Doc/assets/... +/Doc/doc_root/en-us/*.mdx +/Doc/doc_root/en-us/getting-started/*.mdx +/Doc/fonts/... /Doc/img/... -/Doc/documentation.html +/Doc/js/... +/Doc/.nojekyll /Doc/EULA_20220712_FONT-modio.pdf +/Doc/index.html /README.adoc /.gitignore /.clang-format diff --git a/Doc/EULA_20220712_FONT-modio.pdf b/Doc/EULA_20220712_FONT-modio.pdf new file mode 100644 index 00000000..ef2a9e3c Binary files /dev/null and b/Doc/EULA_20220712_FONT-modio.pdf differ diff --git a/Doc/assets/images/ShowPlatformStoreUI-513a06081ba9b2c613e5e4da85b0ca4b.png b/Doc/assets/images/ShowPlatformStoreUI-513a06081ba9b2c613e5e4da85b0ca4b.png new file mode 100644 index 00000000..e1851dde Binary files /dev/null and b/Doc/assets/images/ShowPlatformStoreUI-513a06081ba9b2c613e5e4da85b0ca4b.png differ diff --git a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions-0f1f9aaa5bd0c3c642e84ee61ceff02e.png b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions-0f1f9aaa5bd0c3c642e84ee61ceff02e.png deleted file mode 100644 index 48104bbc..00000000 Binary files a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions-0f1f9aaa5bd0c3c642e84ee61ceff02e.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions-76f5950618826972f9a40bde21e49d4d.png b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions-76f5950618826972f9a40bde21e49d4d.png new file mode 100644 index 00000000..a6a3f56a Binary files /dev/null and b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions-76f5950618826972f9a40bde21e49d4d.png differ diff --git a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetBackgroundThread-e499bbf6ceac1c576056f97dbfce48d3.png b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetBackgroundThread-e499bbf6ceac1c576056f97dbfce48d3.png new file mode 100644 index 00000000..a32f2528 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetBackgroundThread-e499bbf6ceac1c576056f97dbfce48d3.png differ diff --git a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetPortal-c2c8e44fc927853c409eda072d2e604b.png b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetPortal-c2c8e44fc927853c409eda072d2e604b.png deleted file mode 100644 index 92f2537c..00000000 Binary files a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetPortal-c2c8e44fc927853c409eda072d2e604b.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetPortal-f95434662488c3c1596eda6b76b8d3de.png b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetPortal-f95434662488c3c1596eda6b76b8d3de.png new file mode 100644 index 00000000..8462dd45 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetPortal-f95434662488c3c1596eda6b76b8d3de.png differ diff --git a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier-48bd0814f5db0ec0f8a97d2560a396f7.png b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier-48bd0814f5db0ec0f8a97d2560a396f7.png new file mode 100644 index 00000000..4010a428 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier-48bd0814f5db0ec0f8a97d2560a396f7.png differ diff --git a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier-b89970e3d3bb90d64d62b75ce76c02ee.png b/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier-b89970e3d3bb90d64d62b75ce76c02ee.png deleted file mode 100644 index c545e096..00000000 Binary files a/Doc/assets/images/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier-b89970e3d3bb90d64d62b75ce76c02ee.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob-46906f25f1b2ce9866b41ebe2954f9f0.png b/Doc/assets/images/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob-46906f25f1b2ce9866b41ebe2954f9f0.png deleted file mode 100644 index 2481e581..00000000 Binary files a/Doc/assets/images/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob-46906f25f1b2ce9866b41ebe2954f9f0.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob-935a953f1fa434ce4627432800c74b48.png b/Doc/assets/images/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob-935a953f1fa434ce4627432800c74b48.png new file mode 100644 index 00000000..c41da116 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob-935a953f1fa434ce4627432800c74b48.png differ diff --git a/Doc/assets/images/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync-695ef7f0e8b10ac403d0fd21329dfcf0.png b/Doc/assets/images/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync-695ef7f0e8b10ac403d0fd21329dfcf0.png new file mode 100644 index 00000000..75b0e4e7 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync-695ef7f0e8b10ac403d0fd21329dfcf0.png differ diff --git a/Doc/assets/images/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync-ccc2c7187edf30a0c9472a75052ae921.png b/Doc/assets/images/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync-ccc2c7187edf30a0c9472a75052ae921.png deleted file mode 100644 index e32ec8c9..00000000 Binary files a/Doc/assets/images/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync-ccc2c7187edf30a0c9472a75052ae921.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioModProgressInfoLibrary_GetTotalProgress-bf65b6793b2e6f41218751755daf9136.png b/Doc/assets/images/nd_img_ModioModProgressInfoLibrary_GetTotalProgress-bf65b6793b2e6f41218751755daf9136.png new file mode 100644 index 00000000..3ee23c47 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioModProgressInfoLibrary_GetTotalProgress-bf65b6793b2e6f41218751755daf9136.png differ diff --git a/Doc/assets/images/nd_img_ModioModProgressInfoLibrary_GetTotalProgress-c2dfe9ae5100fa60bf076bdd500a7010.png b/Doc/assets/images/nd_img_ModioModProgressInfoLibrary_GetTotalProgress-c2dfe9ae5100fa60bf076bdd500a7010.png deleted file mode 100644 index 6b588d10..00000000 Binary files a/Doc/assets/images/nd_img_ModioModProgressInfoLibrary_GetTotalProgress-c2dfe9ae5100fa60bf076bdd500a7010.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSDKLibrary_GetMonetizationPurchaseCategory-51bd9f941653bd18b5e75f34e6c26c39.png b/Doc/assets/images/nd_img_ModioSDKLibrary_GetMonetizationPurchaseCategory-51bd9f941653bd18b5e75f34e6c26c39.png new file mode 100644 index 00000000..888e1e4c Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSDKLibrary_GetMonetizationPurchaseCategory-51bd9f941653bd18b5e75f34e6c26c39.png differ diff --git a/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory-43d8d48bf01808ca9f889c44c78586e4.png b/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory-43d8d48bf01808ca9f889c44c78586e4.png new file mode 100644 index 00000000..d1dcc3ea Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory-43d8d48bf01808ca9f889c44c78586e4.png differ diff --git a/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory-4f094cc73d91d2c82cbefe34bc1c172f.png b/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory-4f094cc73d91d2c82cbefe34bc1c172f.png deleted file mode 100644 index 91de5136..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory-4f094cc73d91d2c82cbefe34bc1c172f.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync-31680cc49f35d2f21d91b6d2fbb9c98d.png b/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync-31680cc49f35d2f21d91b6d2fbb9c98d.png new file mode 100644 index 00000000..4747bac8 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync-31680cc49f35d2f21d91b6d2fbb9c98d.png differ diff --git a/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync-5d1e90118f4f934bf9d50957b51391bb.png b/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync-5d1e90118f4f934bf9d50957b51391bb.png deleted file mode 100644 index 7a0a604b..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync-5d1e90118f4f934bf9d50957b51391bb.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_CloseTempModSet-91ea6860470e77c26c197cb71b0b9c3e.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_CloseTempModSet-91ea6860470e77c26c197cb71b0b9c3e.png deleted file mode 100644 index 489c0c56..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_CloseTempModSet-91ea6860470e77c26c197cb71b0b9c3e.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_CloseTempModSet-cd8ef129b913668bdc811207a7c16224.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_CloseTempModSet-cd8ef129b913668bdc811207a7c16224.png new file mode 100644 index 00000000..db66c41a Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_CloseTempModSet-cd8ef129b913668bdc811207a7c16224.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync-83a8301ecba540f8a8592822d3184974.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync-83a8301ecba540f8a8592822d3184974.png new file mode 100644 index 00000000..df1e4049 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync-83a8301ecba540f8a8592822d3184974.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync-959793d6bae5805e609d6948f6071617.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync-959793d6bae5805e609d6948f6071617.png deleted file mode 100644 index 6cc30a31..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync-959793d6bae5805e609d6948f6071617.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync-75895154b0ad395e68b9055871a93dab.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync-75895154b0ad395e68b9055871a93dab.png deleted file mode 100644 index 9519f93a..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync-75895154b0ad395e68b9055871a93dab.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync-d188629fa0a5c265b944f648d53b6440.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync-d188629fa0a5c265b944f648d53b6440.png new file mode 100644 index 00000000..8e219a39 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync-d188629fa0a5c265b944f648d53b6440.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync-a0630066f06cad861a8769948f32f175.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync-a0630066f06cad861a8769948f32f175.png deleted file mode 100644 index 8175f93c..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync-a0630066f06cad861a8769948f32f175.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync-be9588fa4dfac1ac91a102ce0ebd6671.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync-be9588fa4dfac1ac91a102ce0ebd6671.png new file mode 100644 index 00000000..b5a91d2d Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync-be9588fa4dfac1ac91a102ce0ebd6671.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync-5fe3ab434879bb8d9a79b0ee50739d4c.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync-5fe3ab434879bb8d9a79b0ee50739d4c.png new file mode 100644 index 00000000..a0ab878c Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync-5fe3ab434879bb8d9a79b0ee50739d4c.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync-f9b903e0e2833d3380b115b75ab00e07.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync-f9b903e0e2833d3380b115b75ab00e07.png deleted file mode 100644 index 678e320f..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync-f9b903e0e2833d3380b115b75ab00e07.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_InitTempModSet-29cf256b95cf42b433c622b1cae7f884.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_InitTempModSet-29cf256b95cf42b433c622b1cae7f884.png deleted file mode 100644 index a9e4b009..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_InitTempModSet-29cf256b95cf42b433c622b1cae7f884.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_InitTempModSet-34fdcb787f07da43afb64dd9d76ea4d1.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_InitTempModSet-34fdcb787f07da43afb64dd9d76ea4d1.png new file mode 100644 index 00000000..70d24793 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_InitTempModSet-34fdcb787f07da43afb64dd9d76ea4d1.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_ListAllModsAsync-1ad7bc96d31f8ae60bfae27078fc2b9f.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_ListAllModsAsync-1ad7bc96d31f8ae60bfae27078fc2b9f.png new file mode 100644 index 00000000..7efcf565 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_ListAllModsAsync-1ad7bc96d31f8ae60bfae27078fc2b9f.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_ListAllModsAsync-316f57d416ba0ae025eb48f00d1c9aa2.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_ListAllModsAsync-316f57d416ba0ae025eb48f00d1c9aa2.png deleted file mode 100644 index a6b764ca..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_ListAllModsAsync-316f57d416ba0ae025eb48f00d1c9aa2.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync-554e56025bd496213a76378b849c929b.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync-554e56025bd496213a76378b849c929b.png new file mode 100644 index 00000000..918f2734 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync-554e56025bd496213a76378b849c929b.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync-79b62f72079482576f25788aa013f789.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync-79b62f72079482576f25788aa013f789.png deleted file mode 100644 index 37f1cc60..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync-79b62f72079482576f25788aa013f789.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_PurchaseModAsync-1fc3c7edad752491c5d369a742e7d265.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_PurchaseModAsync-1fc3c7edad752491c5d369a742e7d265.png deleted file mode 100644 index 7c9b9ce3..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_PurchaseModAsync-1fc3c7edad752491c5d369a742e7d265.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_PurchaseModAsync-d05b70f5ff78fce8ad562c2c6597af48.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_PurchaseModAsync-d05b70f5ff78fce8ad562c2c6597af48.png new file mode 100644 index 00000000..e9e01a82 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_PurchaseModAsync-d05b70f5ff78fce8ad562c2c6597af48.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_QueryTempModSet-0ba976289a53dbf96b322b103c7afc21.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_QueryTempModSet-0ba976289a53dbf96b322b103c7afc21.png deleted file mode 100644 index df294550..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_QueryTempModSet-0ba976289a53dbf96b322b103c7afc21.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_QueryTempModSet-b0ede8242625cb572430cc348883f3c0.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_QueryTempModSet-b0ede8242625cb572430cc348883f3c0.png new file mode 100644 index 00000000..c53cb9a3 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_QueryTempModSet-b0ede8242625cb572430cc348883f3c0.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_RemoveFromTempModSet-51ca3fcbbd419cf6e470b1f3f159aa84.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_RemoveFromTempModSet-51ca3fcbbd419cf6e470b1f3f159aa84.png deleted file mode 100644 index 688c8dd8..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_RemoveFromTempModSet-51ca3fcbbd419cf6e470b1f3f159aa84.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_RemoveFromTempModSet-6a7cd2b22c7cb4bd05c036821655d49a.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_RemoveFromTempModSet-6a7cd2b22c7cb4bd05c036821655d49a.png new file mode 100644 index 00000000..a3c289e6 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_RemoveFromTempModSet-6a7cd2b22c7cb4bd05c036821655d49a.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_ReportContentAsync-dbaa9fa77a041509aecb7a3e17816909.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_ReportContentAsync-dbaa9fa77a041509aecb7a3e17816909.png deleted file mode 100644 index 5696e671..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_ReportContentAsync-dbaa9fa77a041509aecb7a3e17816909.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_ReportContentAsync-eb434d3157fb7884dc999828eb2d6a6e.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_ReportContentAsync-eb434d3157fb7884dc999828eb2d6a6e.png new file mode 100644 index 00000000..df199f0e Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_ReportContentAsync-eb434d3157fb7884dc999828eb2d6a6e.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_SubmitNewModAsync-8728a8ce4666e4a8d6477ae0f4a43b03.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_SubmitNewModAsync-8728a8ce4666e4a8d6477ae0f4a43b03.png deleted file mode 100644 index 34251e69..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_SubmitNewModAsync-8728a8ce4666e4a8d6477ae0f4a43b03.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_SubmitNewModAsync-95991a02d50891bf9f8e62f8aa9c4161.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_SubmitNewModAsync-95991a02d50891bf9f8e62f8aa9c4161.png new file mode 100644 index 00000000..62e6f599 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_SubmitNewModAsync-95991a02d50891bf9f8e62f8aa9c4161.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_SubscribeToModAsync-40e6cf4f6781ca88098119205d7d7a3f.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_SubscribeToModAsync-40e6cf4f6781ca88098119205d7d7a3f.png deleted file mode 100644 index 587b01f6..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_SubscribeToModAsync-40e6cf4f6781ca88098119205d7d7a3f.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_SubscribeToModAsync-93fd3340617ee4d44dea7078fb1199d4.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_SubscribeToModAsync-93fd3340617ee4d44dea7078fb1199d4.png new file mode 100644 index 00000000..389e6852 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_SubscribeToModAsync-93fd3340617ee4d44dea7078fb1199d4.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_UnmuteUserAsync-3d5fb9c4326a9e0083404c7122e2a1c8.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_UnmuteUserAsync-3d5fb9c4326a9e0083404c7122e2a1c8.png new file mode 100644 index 00000000..d4c73efe Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_UnmuteUserAsync-3d5fb9c4326a9e0083404c7122e2a1c8.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_UnmuteUserAsync-5072928585d25488a648dfc7210fb4e0.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_UnmuteUserAsync-5072928585d25488a648dfc7210fb4e0.png deleted file mode 100644 index 1175a92d..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_UnmuteUserAsync-5072928585d25488a648dfc7210fb4e0.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync-c749be66fec80d0e8bfe214ade7abbcf.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync-c749be66fec80d0e8bfe214ade7abbcf.png deleted file mode 100644 index 2c07b9ba..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync-c749be66fec80d0e8bfe214ade7abbcf.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync-ffcaaed44b92c819c915849b1931814a.png b/Doc/assets/images/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync-ffcaaed44b92c819c915849b1931814a.png new file mode 100644 index 00000000..59f5c69b Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync-ffcaaed44b92c819c915849b1931814a.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_PrioritizeTransferForMod-1eb6142c4dd34c75b675bb6246a0d6d8.png b/Doc/assets/images/nd_img_ModioSubsystem_PrioritizeTransferForMod-1eb6142c4dd34c75b675bb6246a0d6d8.png new file mode 100644 index 00000000..850ec592 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_PrioritizeTransferForMod-1eb6142c4dd34c75b675bb6246a0d6d8.png differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_PrioritizeTransferForMod-597a4fd0e4c6a56d4264f3a2df028b25.png b/Doc/assets/images/nd_img_ModioSubsystem_PrioritizeTransferForMod-597a4fd0e4c6a56d4264f3a2df028b25.png deleted file mode 100644 index 71ac0375..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_PrioritizeTransferForMod-597a4fd0e4c6a56d4264f3a2df028b25.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_QueryUserInstallations-42ae147ec3c47d4b1f17ba58476bf09e.png b/Doc/assets/images/nd_img_ModioSubsystem_QueryUserInstallations-42ae147ec3c47d4b1f17ba58476bf09e.png deleted file mode 100644 index f2b7df79..00000000 Binary files a/Doc/assets/images/nd_img_ModioSubsystem_QueryUserInstallations-42ae147ec3c47d4b1f17ba58476bf09e.png and /dev/null differ diff --git a/Doc/assets/images/nd_img_ModioSubsystem_QueryUserInstallations-ad0e0b0aa88244144d9da4210ee3d5d1.png b/Doc/assets/images/nd_img_ModioSubsystem_QueryUserInstallations-ad0e0b0aa88244144d9da4210ee3d5d1.png new file mode 100644 index 00000000..c83e96c2 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioSubsystem_QueryUserInstallations-ad0e0b0aa88244144d9da4210ee3d5d1.png differ diff --git a/Doc/assets/images/nd_img_ModioUnsigned64Library_MakeFromComponents-5bd42089022b89256170fe2c464b932d.png b/Doc/assets/images/nd_img_ModioUnsigned64Library_MakeFromComponents-5bd42089022b89256170fe2c464b932d.png new file mode 100644 index 00000000..fef40741 Binary files /dev/null and b/Doc/assets/images/nd_img_ModioUnsigned64Library_MakeFromComponents-5bd42089022b89256170fe2c464b932d.png differ diff --git a/Doc/assets/images/nd_img_ModioUnsigned64Library_Percentage_Unsigned64-f7f1acacb3174469af6f03459bea230a.png b/Doc/assets/images/nd_img_ModioUnsigned64Library_Percentage_Unsigned64-f7f1acacb3174469af6f03459bea230a.png deleted file mode 100644 index 0a971ab9..00000000 Binary files a/Doc/assets/images/nd_img_ModioUnsigned64Library_Percentage_Unsigned64-f7f1acacb3174469af6f03459bea230a.png and /dev/null differ diff --git a/Doc/assets/js/0c9c4b9e.33716226.js b/Doc/assets/js/0c9c4b9e.33716226.js deleted file mode 100644 index cc504fdd..00000000 --- a/Doc/assets/js/0c9c4b9e.33716226.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[4944],{50494:(e,i,d)=>{d.r(i),d.d(i,{assets:()=>l,contentTitle:()=>n,default:()=>h,frontMatter:()=>o,metadata:()=>r,toc:()=>c});var t=d(74848),s=d(28453);const o={id:"ue-mod-creation-tool",title:"Mod Creation Tool",slug:"/unreal/mod-creation-tool/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/mod-creation-tool-documentation.mdx"},n=void 0,r={id:"ue-mod-creation-tool",title:"Mod Creation Tool",description:"Mod Creation & Upload Tool",source:"@site/public/en-us/mod-creation-tool-documentation.mdx",sourceDirName:".",slug:"/unreal/mod-creation-tool/",permalink:"/unreal/mod-creation-tool/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/mod-creation-tool-documentation.mdx",tags:[],version:"current",frontMatter:{id:"ue-mod-creation-tool",title:"Mod Creation Tool",slug:"/unreal/mod-creation-tool/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/mod-creation-tool-documentation.mdx"},sidebar:"sidebar",previous:{title:"Profiling",permalink:"/unreal/profiling/"},next:{title:"Android Configuration",permalink:"/unreal/android-configuration/"}},l={},c=[{value:"Mod Creation & Upload Tool",id:"mod-creation--upload-tool",level:2},{value:"Editor-specific plugin modules",id:"editor-specific-plugin-modules",level:3},{value:"Opening the tool",id:"opening-the-tool",level:3},{value:"Authenticating",id:"authenticating",level:3},{value:"Verification Code",id:"verification-code",level:4},{value:"Create or Edit Mods",id:"create-or-edit-mods",level:3},{value:"Create Mod",id:"create-mod",level:4},{value:"Edit Mods",id:"edit-mods",level:4},{value:"Create a Mod",id:"create-a-mod",level:3},{value:"Create Mod",id:"create-mod-1",level:4},{value:"Edit existing mods",id:"edit-existing-mods",level:3},{value:"Edit Mods",id:"edit-mods-1",level:4},{value:"Uploading a new modfile",id:"uploading-a-new-modfile",level:3},{value:"Mod file upload",id:"mod-file-upload",level:3}];function a(e){const i={h2:"h2",h3:"h3",h4:"h4",img:"img",p:"p",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",...(0,s.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(i.h2,{id:"mod-creation--upload-tool",children:"Mod Creation & Upload Tool"}),"\n",(0,t.jsx)(i.p,{children:"The mod.io Unreal Engine's plugin includes an editor module which provides an embedded mod.io - Content Creation and Upload Tool."}),"\n",(0,t.jsx)(i.h3,{id:"editor-specific-plugin-modules",children:"Editor-specific plugin modules"}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Module Name"}),(0,t.jsx)(i.th,{children:"Description"}),(0,t.jsx)(i.th,{children:"Module Type"})]})}),(0,t.jsx)(i.tbody,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"ModioEditor"}),(0,t.jsx)(i.td,{children:"Editor details customization and asset factories for UE content creation and upload tool classes"}),(0,t.jsx)(i.td,{children:"Editor"})]})})]}),"\n",(0,t.jsx)(i.h3,{id:"opening-the-tool",children:"Opening the tool"}),"\n",(0,t.jsxs)(i.p,{children:["Click the mod.io icon in Unreal Engine to show a drop down menu, click the ",(0,t.jsx)(i.strong,{children:"Create & Upload"})," menu. Once opened, it may require mod.io authentication as below."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"how_to_open_tool",src:d(92394).A+"",width:"1280",height:"212"})}),"\n",(0,t.jsx)(i.h3,{id:"authenticating",children:"Authenticating"}),"\n",(0,t.jsx)(i.p,{children:"Authentication may be required either being a first time user of the module or not logged in previously. If you are already logged in, you will automatically be redirected to creating or editing mods."}),"\n",(0,t.jsx)(i.p,{children:"The user will be authenticated if not already logged in to mod.io account."}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"Email:"}),"\nEnter your email in the text box shown below and click Login, a verification code will be sent to the submitted email address."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"auth_enter_email",src:d(11200).A+"",width:"1360",height:"967"})}),"\n",(0,t.jsx)(i.h4,{id:"verification-code",children:"Verification Code"}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"Verification Code:"})," Enter the code received in your email and click Authenticate."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"auth_enter_code",src:d(34353).A+"",width:"1362",height:"961"})}),"\n",(0,t.jsx)(i.p,{children:"If everything goes correctly, you will be authenticated successfully."}),"\n",(0,t.jsx)(i.h3,{id:"create-or-edit-mods",children:"Create or Edit Mods"}),"\n",(0,t.jsxs)(i.p,{children:["After successfull authentication, you can either ",(0,t.jsx)(i.strong,{children:"Create Mod"})," or ",(0,t.jsx)(i.strong,{children:"Edit Mods"}),"."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_or_upload_mod",src:d(99577).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsx)(i.h4,{id:"create-mod",children:"Create Mod"}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"Create Mod"})," lets you create a new mod."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_mod",src:d(54441).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsx)(i.h4,{id:"edit-mods",children:"Edit Mods"}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"Edit Mods"})," lets you edit existing mods and change any specific values or mod files on them."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"edit_mods",src:d(44158).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsx)(i.h3,{id:"create-a-mod",children:"Create a Mod"}),"\n",(0,t.jsx)(i.p,{children:"Creating a mod is made easy directly from within Unreal Engine described in the\nprocess below."}),"\n",(0,t.jsx)(i.h4,{id:"create-mod-1",children:"Create Mod"}),"\n",(0,t.jsxs)(i.p,{children:["Click ",(0,t.jsx)(i.strong,{children:"Create Mod"})," and fill the following required fields:"]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Create Mod Properties"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Fields"}),(0,t.jsx)(i.th,{children:"Description"})]})}),(0,t.jsxs)(i.tbody,{children:[(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Path to Logo File"}),(0,t.jsxs)(i.td,{children:["Browse to select your ",(0,t.jsx)(i.strong,{children:".png"})," file for logo"]})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Name"}),(0,t.jsx)(i.td,{children:"Name of the mod"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Summary"}),(0,t.jsx)(i.td,{children:"A brief summary of the mod"})]})]})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_mod_properties",src:d(17021).A+"",width:"1360",height:"963"})}),"\n",(0,t.jsxs)(i.p,{children:["Once filled all the required fields, click ",(0,t.jsx)(i.strong,{children:"Submit"})," to create the mod."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_mod_properties_submit",src:d(18266).A+"",width:"1360",height:"963"})}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"TIP"}),": You may get an error dialog if the game doesn't support your platform or the game is locked."]}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.img,{alt:"error_platform_not_supported",src:d(83061).A+"",width:"754",height:"244"}),"\n",(0,t.jsx)(i.img,{alt:"error_game_locked",src:d(53138).A+"",width:"652",height:"252"})]}),"\n",(0,t.jsx)(i.p,{children:"Once everything goes successful you will be asked if you also want to upload a mod file right away."}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"mod_created_modfile_prompt",src:d(63711).A+"",width:"433",height:"232"})}),"\n",(0,t.jsxs)(i.p,{children:["If ",(0,t.jsx)(i.strong,{children:"Yes"})," is selected you will be redirected to ",(0,t.jsx)(i.strong,{children:"STEP 6"}),"."]}),"\n",(0,t.jsx)(i.h3,{id:"edit-existing-mods",children:"Edit existing mods"}),"\n",(0,t.jsx)(i.p,{children:"Editing existing mods is made easy in the same way as creating a mod directly from within Unreal Engine described in the process below."}),"\n",(0,t.jsx)(i.h4,{id:"edit-mods-1",children:"Edit Mods"}),"\n",(0,t.jsxs)(i.p,{children:["By clicking ",(0,t.jsx)(i.strong,{children:"Edit Mods"}),", a list of mods will be shown to edit. You can select a mod by clicking on it."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Browse Mods"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Name"}),(0,t.jsx)(i.th,{children:"Description"})]})}),(0,t.jsx)(i.tbody,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Your mod name"}),(0,t.jsx)(i.td,{children:"Your mod description"})]})})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"browse_mods",src:d(93948).A+"",width:"1359",height:"961"})}),"\n",(0,t.jsxs)(i.p,{children:["Once a mod is selected, click ",(0,t.jsx)(i.strong,{children:"Edit Mod"})," to edit the mod."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"browse_mods_edit_mods",src:d(31305).A+"",width:"1359",height:"961"})}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Edit Mod Properties"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Field"}),(0,t.jsx)(i.th,{children:"Description"})]})}),(0,t.jsxs)(i.tbody,{children:[(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Name"}),(0,t.jsx)(i.td,{children:"Your mod name"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Summary"}),(0,t.jsx)(i.td,{children:"Your mod summary"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Homepage URL"}),(0,t.jsx)(i.td,{children:"Your mod homepage url"})]})]})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"edit_mod_properties",src:d(43785).A+"",width:"1357",height:"961"})}),"\n",(0,t.jsxs)(i.p,{children:["If you want to change the mod properties, make desirable changes to above fields and click ",(0,t.jsx)(i.strong,{children:"Submit"}),"."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"edit_mod_properties_submit",src:d(52190).A+"",width:"1357",height:"961"})}),"\n",(0,t.jsxs)(i.p,{children:["Then click ",(0,t.jsx)(i.strong,{children:"Edit Files"})," to see list of existing mod files."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"edit_mod_properties_edit_files",src:d(68546).A+"",width:"1357",height:"961"})}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Modfile"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Name"}),(0,t.jsx)(i.th,{children:"Platform"}),(0,t.jsx)(i.th,{children:"Version"}),(0,t.jsx)(i.th,{children:"Status"})]})}),(0,t.jsx)(i.tbody,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Your mod file name"}),(0,t.jsx)(i.td,{children:"Your mod platform"}),(0,t.jsx)(i.td,{children:"Your mod file version"}),(0,t.jsx)(i.td,{children:"Your mod file status"})]})})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"browse_modfile",src:d(18393).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsx)(i.h3,{id:"uploading-a-new-modfile",children:"Uploading a new modfile"}),"\n",(0,t.jsx)(i.p,{children:"To upload a mod file, a workspace directory path is mandatory which is described in the process below."}),"\n",(0,t.jsxs)(i.p,{children:["Click ",(0,t.jsx)(i.strong,{children:"New Modfile"})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"browse_modfile_new_modfile",src:d(56641).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsxs)(i.p,{children:["You may need to select ",(0,t.jsx)(i.strong,{children:"Create mod for PC"})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_mod_for_pc",src:d(14233).A+"",width:"1354",height:"955"})}),"\n",(0,t.jsx)(i.p,{children:"Fill the required fields:"}),"\n",(0,t.jsx)(i.p,{children:"TIP: You may get an error dialog if the game doesn't support your platform or the game is locked."}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Upload Mod File"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Field"}),(0,t.jsx)(i.th,{children:"Description"})]})}),(0,t.jsxs)(i.tbody,{children:[(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Version"}),(0,t.jsx)(i.td,{children:"Your mod file version"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Changelog"}),(0,t.jsx)(i.td,{children:"Your mod file platform"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Set as Active Release"}),(0,t.jsx)(i.td,{children:"Whether to set as Active Release"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Path to Mod Root Directory"}),(0,t.jsx)(i.td,{children:"Path to the workspace directory"})]})]})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"upload_modfile",src:d(67936).A+"",width:"1351",height:"955"})}),"\n",(0,t.jsxs)(i.p,{children:["Once all fields are filled out, click ",(0,t.jsx)(i.strong,{children:"Submit"})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"upload_modfile_submit",src:d(58681).A+"",width:"1351",height:"955"})}),"\n",(0,t.jsx)(i.p,{children:"Once submitted, the progress bar will be displayed about the workspace directory being zipped and uploaded with all other information provided for the new mod file."}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"upload_modfile_submit_status",src:d(37520).A+"",width:"1357",height:"958"})}),"\n",(0,t.jsx)(i.h3,{id:"mod-file-upload",children:"Mod file upload"}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"TIP"}),": You may get an error dialog if the game doesn't support your platform or the game is locked or any error that occurs due to internet or any other circumstances."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"modfile_successfully_uploaded",src:d(60003).A+"",width:"477",height:"238"})})]})}function h(e={}){const{wrapper:i}={...(0,s.R)(),...e.components};return i?(0,t.jsx)(i,{...e,children:(0,t.jsx)(a,{...e})}):a(e)}},34353:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/auth_enter_code-778c20c0c85b3b699f6b8d8701a2a095.jpg"},11200:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/auth_enter_email-de3079d992602c99281e0e899592e582.jpg"},18393:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/browse_modfile-9bc6ae054b047f63a8c9cfd3c9ac169b.jpg"},56641:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/browse_modfile_new_modfile-d2f495f1a091c98d062c27094ec526d9.jpg"},93948:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/browse_mods-ea8c350b3d963e62b0c05d1b28d0359f.jpg"},31305:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/browse_mods_edit_mods-017a3a3ba9955520dff199b962043182.jpg"},54441:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_mod-6d730263f0d163dee63907805aab7340.jpg"},14233:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_mod_for_pc-96ab4fb29fdb229f3e0a8cab75292971.jpg"},17021:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_mod_properties-622b730a8852b6d218da568ef95a6766.jpg"},18266:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_mod_properties_submit-78a0bcc5ddafb377774177ccecb8c544.jpg"},99577:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_or_upload_mod-ba6c9884ea92b45d8aa8fa183356727b.jpg"},43785:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/edit_mod_properties-6d5a61004bd10b50b4d0a1d2ee9b6a98.jpg"},68546:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/edit_mod_properties_edit_files-c6923cd5e3ee481be4fa0380146a4133.jpg"},52190:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/edit_mod_properties_submit-0ff0cacefbf3dba389c45c7add3d462f.jpg"},44158:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/edit_mods-046e4b1fe58ad5636b9a39a737a09a4f.jpg"},53138:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/error_game_locked-048ef273c39027b8bfc43cae4853ec47.jpg"},83061:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/error_platform_not_supported-cb2c479540dab3244baba6291a1495df.jpg"},92394:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/how_to_open_tool-ad0da9797d0eed19d37463af2ba06c45.jpg"},63711:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/mod_created_modfile_prompt-c1dd493d8fcca3b0274207ce6055d5fc.jpg"},60003:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/modfile_successfully_uploaded-abfddcd13899473da8ba107f8dfeeb40.jpg"},67936:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/upload_modfile-8a8ba2d5682ddc9c385970721c8baf21.jpg"},58681:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/upload_modfile_submit-fc996aea4fa01bc4d7836e391c58e3b4.jpg"},37520:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/upload_modfile_submit_status-fd8311ac91d9bc3bf72ecc7fee5bff01.jpg"},28453:(e,i,d)=>{d.d(i,{R:()=>n,x:()=>r});var t=d(96540);const s={},o=t.createContext(s);function n(e){const i=t.useContext(o);return t.useMemo((function(){return"function"==typeof e?e(i):{...i,...e}}),[i,e])}function r(e){let i;return i=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:n(e.components),t.createElement(o.Provider,{value:i},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/0c9c4b9e.48f78565.js b/Doc/assets/js/0c9c4b9e.48f78565.js new file mode 100644 index 00000000..186ecb74 --- /dev/null +++ b/Doc/assets/js/0c9c4b9e.48f78565.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[4944],{50494:(e,i,d)=>{d.r(i),d.d(i,{assets:()=>l,contentTitle:()=>n,default:()=>h,frontMatter:()=>s,metadata:()=>r,toc:()=>c});var t=d(74848),o=d(28453);const s={id:"ue-mod-creation-tool",title:"Mod Creation Tool",slug:"/unreal/mod-creation-tool/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/mod-creation-tool-documentation.mdx"},n=void 0,r={id:"ue-mod-creation-tool",title:"Mod Creation Tool",description:"Mod Creation & Upload Tool",source:"@site/public/en-us/mod-creation-tool-documentation.mdx",sourceDirName:".",slug:"/unreal/mod-creation-tool/",permalink:"/unreal/mod-creation-tool/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/mod-creation-tool-documentation.mdx",tags:[],version:"current",frontMatter:{id:"ue-mod-creation-tool",title:"Mod Creation Tool",slug:"/unreal/mod-creation-tool/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/mod-creation-tool-documentation.mdx"},sidebar:"sidebar",previous:{title:"Profiling",permalink:"/unreal/profiling/"},next:{title:"Android Configuration",permalink:"/unreal/android-configuration/"}},l={},c=[{value:"Mod Creation & Upload Tool",id:"mod-creation--upload-tool",level:2},{value:"Editor-specific plugin modules",id:"editor-specific-plugin-modules",level:3},{value:"Opening the tool",id:"opening-the-tool",level:3},{value:"Authenticating",id:"authenticating",level:3},{value:"Verification Code",id:"verification-code",level:4},{value:"Create or Edit Mods",id:"create-or-edit-mods",level:3},{value:"Create Mod",id:"create-mod",level:4},{value:"Edit Mods",id:"edit-mods",level:4},{value:"Create a Mod",id:"create-a-mod",level:3},{value:"Create Mod",id:"create-mod-1",level:4},{value:"Edit existing mods",id:"edit-existing-mods",level:3},{value:"Edit Mods",id:"edit-mods-1",level:4},{value:"Uploading a new modfile",id:"uploading-a-new-modfile",level:3},{value:"Mod file upload",id:"mod-file-upload",level:3}];function a(e){const i={h2:"h2",h3:"h3",h4:"h4",img:"img",p:"p",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(i.h2,{id:"mod-creation--upload-tool",children:"Mod Creation & Upload Tool"}),"\n",(0,t.jsx)(i.p,{children:"The mod.io Unreal Engine's plugin includes an editor module which provides an embedded mod.io - Content Creation and Upload Tool."}),"\n",(0,t.jsx)(i.h3,{id:"editor-specific-plugin-modules",children:"Editor-specific plugin modules"}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Module Name"}),(0,t.jsx)(i.th,{children:"Description"}),(0,t.jsx)(i.th,{children:"Module Type"})]})}),(0,t.jsx)(i.tbody,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"ModioEditor"}),(0,t.jsx)(i.td,{children:"Editor details customization and asset factories for UE content creation and upload tool classes"}),(0,t.jsx)(i.td,{children:"Editor"})]})})]}),"\n",(0,t.jsx)(i.h3,{id:"opening-the-tool",children:"Opening the tool"}),"\n",(0,t.jsxs)(i.p,{children:["Click the mod.io icon in Unreal Engine to show a drop down menu, click the ",(0,t.jsx)(i.strong,{children:"Create & Upload"})," menu. Once opened, it may require mod.io authentication as below."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"how_to_open_tool",src:d(94035).A+"",width:"1280",height:"212"})}),"\n",(0,t.jsx)(i.h3,{id:"authenticating",children:"Authenticating"}),"\n",(0,t.jsx)(i.p,{children:"Authentication may be required either being a first time user of the module or not logged in previously. If you are already logged in, you will automatically be redirected to creating or editing mods."}),"\n",(0,t.jsx)(i.p,{children:"The user will be authenticated if not already logged in to mod.io account."}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"Email:"}),"\nEnter your email in the text box shown below and click Login, a verification code will be sent to the submitted email address."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"auth_enter_email",src:d(89777).A+"",width:"1360",height:"967"})}),"\n",(0,t.jsx)(i.h4,{id:"verification-code",children:"Verification Code"}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"Verification Code:"})," Enter the code received in your email and click Authenticate."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"auth_enter_code",src:d(39002).A+"",width:"1362",height:"961"})}),"\n",(0,t.jsx)(i.p,{children:"If everything goes correctly, you will be authenticated successfully."}),"\n",(0,t.jsx)(i.h3,{id:"create-or-edit-mods",children:"Create or Edit Mods"}),"\n",(0,t.jsxs)(i.p,{children:["After successfull authentication, you can either ",(0,t.jsx)(i.strong,{children:"Create Mod"})," or ",(0,t.jsx)(i.strong,{children:"Edit Mods"}),"."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_or_upload_mod",src:d(39508).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsx)(i.h4,{id:"create-mod",children:"Create Mod"}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"Create Mod"})," lets you create a new mod."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_mod",src:d(10768).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsx)(i.h4,{id:"edit-mods",children:"Edit Mods"}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"Edit Mods"})," lets you edit existing mods and change any specific values or mod files on them."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"edit_mods",src:d(56061).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsx)(i.h3,{id:"create-a-mod",children:"Create a Mod"}),"\n",(0,t.jsx)(i.p,{children:"Creating a mod is made easy directly from within Unreal Engine described in the\nprocess below."}),"\n",(0,t.jsx)(i.h4,{id:"create-mod-1",children:"Create Mod"}),"\n",(0,t.jsxs)(i.p,{children:["Click ",(0,t.jsx)(i.strong,{children:"Create Mod"})," and fill the following required fields:"]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Create Mod Properties"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Fields"}),(0,t.jsx)(i.th,{children:"Description"})]})}),(0,t.jsxs)(i.tbody,{children:[(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Path to Logo File"}),(0,t.jsxs)(i.td,{children:["Browse to select your ",(0,t.jsx)(i.strong,{children:".png"})," file for logo"]})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Name"}),(0,t.jsx)(i.td,{children:"Name of the mod"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Summary"}),(0,t.jsx)(i.td,{children:"A brief summary of the mod"})]})]})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_mod_properties",src:d(99142).A+"",width:"1360",height:"963"})}),"\n",(0,t.jsxs)(i.p,{children:["Once filled all the required fields, click ",(0,t.jsx)(i.strong,{children:"Submit"})," to create the mod."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_mod_properties_submit",src:d(21447).A+"",width:"1360",height:"963"})}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"TIP"}),": You may get an error dialog if the game doesn't support your platform or the game is locked."]}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.img,{alt:"error_platform_not_supported",src:d(16028).A+"",width:"754",height:"244"}),"\n",(0,t.jsx)(i.img,{alt:"error_game_locked",src:d(32977).A+"",width:"652",height:"252"})]}),"\n",(0,t.jsx)(i.p,{children:"Once everything goes successful you will be asked if you also want to upload a mod file right away."}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"mod_created_modfile_prompt",src:d(13006).A+"",width:"433",height:"232"})}),"\n",(0,t.jsxs)(i.p,{children:["If ",(0,t.jsx)(i.strong,{children:"Yes"})," is selected you will be redirected to ",(0,t.jsx)(i.strong,{children:"STEP 6"}),"."]}),"\n",(0,t.jsx)(i.h3,{id:"edit-existing-mods",children:"Edit existing mods"}),"\n",(0,t.jsx)(i.p,{children:"Editing existing mods is made easy in the same way as creating a mod directly from within Unreal Engine described in the process below."}),"\n",(0,t.jsx)(i.h4,{id:"edit-mods-1",children:"Edit Mods"}),"\n",(0,t.jsxs)(i.p,{children:["By clicking ",(0,t.jsx)(i.strong,{children:"Edit Mods"}),", a list of mods will be shown to edit. You can select a mod by clicking on it."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Browse Mods"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Name"}),(0,t.jsx)(i.th,{children:"Description"})]})}),(0,t.jsx)(i.tbody,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Your mod name"}),(0,t.jsx)(i.td,{children:"Your mod description"})]})})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"browse_mods",src:d(74999).A+"",width:"1359",height:"961"})}),"\n",(0,t.jsxs)(i.p,{children:["Once a mod is selected, click ",(0,t.jsx)(i.strong,{children:"Edit Mod"})," to edit the mod."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"browse_mods_edit_mods",src:d(47086).A+"",width:"1359",height:"961"})}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Edit Mod Properties"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Field"}),(0,t.jsx)(i.th,{children:"Description"})]})}),(0,t.jsxs)(i.tbody,{children:[(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Name"}),(0,t.jsx)(i.td,{children:"Your mod name"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Summary"}),(0,t.jsx)(i.td,{children:"Your mod summary"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Homepage URL"}),(0,t.jsx)(i.td,{children:"Your mod homepage url"})]})]})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"edit_mod_properties",src:d(52990).A+"",width:"1357",height:"961"})}),"\n",(0,t.jsxs)(i.p,{children:["If you want to change the mod properties, make desirable changes to above fields and click ",(0,t.jsx)(i.strong,{children:"Submit"}),"."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"edit_mod_properties_submit",src:d(30687).A+"",width:"1357",height:"961"})}),"\n",(0,t.jsxs)(i.p,{children:["Then click ",(0,t.jsx)(i.strong,{children:"Edit Files"})," to see list of existing mod files."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"edit_mod_properties_edit_files",src:d(77515).A+"",width:"1357",height:"961"})}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Modfile"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Name"}),(0,t.jsx)(i.th,{children:"Platform"}),(0,t.jsx)(i.th,{children:"Version"}),(0,t.jsx)(i.th,{children:"Status"})]})}),(0,t.jsx)(i.tbody,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Your mod file name"}),(0,t.jsx)(i.td,{children:"Your mod platform"}),(0,t.jsx)(i.td,{children:"Your mod file version"}),(0,t.jsx)(i.td,{children:"Your mod file status"})]})})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"browse_modfile",src:d(58388).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsx)(i.h3,{id:"uploading-a-new-modfile",children:"Uploading a new modfile"}),"\n",(0,t.jsx)(i.p,{children:"To upload a mod file, a workspace directory path is mandatory which is described in the process below."}),"\n",(0,t.jsxs)(i.p,{children:["Click ",(0,t.jsx)(i.strong,{children:"New Modfile"})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"browse_modfile_new_modfile",src:d(92060).A+"",width:"1360",height:"961"})}),"\n",(0,t.jsxs)(i.p,{children:["You may need to select ",(0,t.jsx)(i.strong,{children:"Create mod for PC"})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"create_mod_for_pc",src:d(22462).A+"",width:"1354",height:"955"})}),"\n",(0,t.jsx)(i.p,{children:"Fill the required fields:"}),"\n",(0,t.jsx)(i.p,{children:"TIP: You may get an error dialog if the game doesn't support your platform or the game is locked."}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.strong,{children:"Upload Mod File"})}),"\n",(0,t.jsxs)(i.table,{children:[(0,t.jsx)(i.thead,{children:(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.th,{children:"Field"}),(0,t.jsx)(i.th,{children:"Description"})]})}),(0,t.jsxs)(i.tbody,{children:[(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Version"}),(0,t.jsx)(i.td,{children:"Your mod file version"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Changelog"}),(0,t.jsx)(i.td,{children:"Your mod file platform"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Set as Active Release"}),(0,t.jsx)(i.td,{children:"Whether to set as Active Release"})]}),(0,t.jsxs)(i.tr,{children:[(0,t.jsx)(i.td,{children:"Path to Mod Root Directory"}),(0,t.jsx)(i.td,{children:"Path to the workspace directory"})]})]})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"upload_modfile",src:d(89609).A+"",width:"1351",height:"955"})}),"\n",(0,t.jsxs)(i.p,{children:["Once all fields are filled out, click ",(0,t.jsx)(i.strong,{children:"Submit"})]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"upload_modfile_submit",src:d(66910).A+"",width:"1351",height:"955"})}),"\n",(0,t.jsx)(i.p,{children:"Once submitted, the progress bar will be displayed about the workspace directory being zipped and uploaded with all other information provided for the new mod file."}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"upload_modfile_submit_status",src:d(76473).A+"",width:"1357",height:"958"})}),"\n",(0,t.jsx)(i.h3,{id:"mod-file-upload",children:"Mod file upload"}),"\n",(0,t.jsxs)(i.p,{children:[(0,t.jsx)(i.strong,{children:"TIP"}),": You may get an error dialog if the game doesn't support your platform or the game is locked or any error that occurs due to internet or any other circumstances."]}),"\n",(0,t.jsx)(i.p,{children:(0,t.jsx)(i.img,{alt:"modfile_successfully_uploaded",src:d(70040).A+"",width:"477",height:"238"})})]})}function h(e={}){const{wrapper:i}={...(0,o.R)(),...e.components};return i?(0,t.jsx)(i,{...e,children:(0,t.jsx)(a,{...e})}):a(e)}},39002:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/auth_enter_code-778c20c0c85b3b699f6b8d8701a2a095.jpg"},89777:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/auth_enter_email-de3079d992602c99281e0e899592e582.jpg"},58388:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/browse_modfile-9bc6ae054b047f63a8c9cfd3c9ac169b.jpg"},92060:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/browse_modfile_new_modfile-d2f495f1a091c98d062c27094ec526d9.jpg"},74999:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/browse_mods-ea8c350b3d963e62b0c05d1b28d0359f.jpg"},47086:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/browse_mods_edit_mods-017a3a3ba9955520dff199b962043182.jpg"},10768:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_mod-6d730263f0d163dee63907805aab7340.jpg"},22462:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_mod_for_pc-96ab4fb29fdb229f3e0a8cab75292971.jpg"},99142:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_mod_properties-622b730a8852b6d218da568ef95a6766.jpg"},21447:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_mod_properties_submit-78a0bcc5ddafb377774177ccecb8c544.jpg"},39508:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/create_or_upload_mod-ba6c9884ea92b45d8aa8fa183356727b.jpg"},52990:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/edit_mod_properties-6d5a61004bd10b50b4d0a1d2ee9b6a98.jpg"},77515:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/edit_mod_properties_edit_files-c6923cd5e3ee481be4fa0380146a4133.jpg"},30687:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/edit_mod_properties_submit-0ff0cacefbf3dba389c45c7add3d462f.jpg"},56061:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/edit_mods-046e4b1fe58ad5636b9a39a737a09a4f.jpg"},32977:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/error_game_locked-048ef273c39027b8bfc43cae4853ec47.jpg"},16028:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/error_platform_not_supported-cb2c479540dab3244baba6291a1495df.jpg"},94035:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/how_to_open_tool-ad0da9797d0eed19d37463af2ba06c45.jpg"},13006:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/mod_created_modfile_prompt-c1dd493d8fcca3b0274207ce6055d5fc.jpg"},70040:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/modfile_successfully_uploaded-abfddcd13899473da8ba107f8dfeeb40.jpg"},89609:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/upload_modfile-8a8ba2d5682ddc9c385970721c8baf21.jpg"},66910:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/upload_modfile_submit-fc996aea4fa01bc4d7836e391c58e3b4.jpg"},76473:(e,i,d)=>{d.d(i,{A:()=>t});const t=d.p+"assets/images/upload_modfile_submit_status-fd8311ac91d9bc3bf72ecc7fee5bff01.jpg"},28453:(e,i,d)=>{d.d(i,{R:()=>n,x:()=>r});var t=d(96540);const o={},s=t.createContext(o);function n(e){const i=t.useContext(s);return t.useMemo((function(){return"function"==typeof e?e(i):{...i,...e}}),[i,e])}function r(e){let i;return i=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:n(e.components),t.createElement(s.Provider,{value:i},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/1da2407e.5245f7dc.js b/Doc/assets/js/1da2407e.5245f7dc.js new file mode 100644 index 00000000..14d7c517 --- /dev/null +++ b/Doc/assets/js/1da2407e.5245f7dc.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[2381],{84923:(e,t,s)=>{s.r(t),s.d(t,{assets:()=>d,contentTitle:()=>l,default:()=>m,frontMatter:()=>o,metadata:()=>c,toc:()=>u});var n=s(74848),i=s(28453),a=s(27064),r=s(89236);const o={id:"ue-metrics-play-sessions",title:"Metrics Play Sessions",slug:"/unreal/getting-started/metrics-play-sessions",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/metrics-play-sessions.mdx"},l=void 0,c={id:"getting-started/ue-metrics-play-sessions",title:"Metrics Play Sessions",description:"The mod.io Unreal Engine Plugin supports all of the mod.io metrics features, allowing you to start a metrics play sesion, keeping that session alive via a heartbeat (automatically called, or manually handled) and then ending that session. Metric sessions allow you to track which mods your players interact with most frequently. Visit https://docs.mod.io/metrics/ for an overview of the mod.io metrics system.",source:"@site/public/en-us/getting-started/metrics-play-sessions.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/metrics-play-sessions",permalink:"/unreal/getting-started/metrics-play-sessions",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/metrics-play-sessions.mdx",tags:[],version:"current",frontMatter:{id:"ue-metrics-play-sessions",title:"Metrics Play Sessions",slug:"/unreal/getting-started/metrics-play-sessions",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/metrics-play-sessions.mdx"},sidebar:"sidebar",previous:{title:"Muting and Unmuting a User",permalink:"/unreal/getting-started/mute-user"},next:{title:"Marketplace",permalink:"/unreal/marketplace/"}},d={},u=[{value:"Initialization",id:"initialization",level:4},{value:"Starting a Metrics Session",id:"starting-a-metrics-session",level:4},{value:"Metrics heartbeat",id:"metrics-heartbeat",level:4},{value:"Ending a Metrics Session",id:"ending-a-metrics-session",level:4}];function h(e){const t={a:"a",admonition:"admonition",code:"code",h4:"h4",img:"img",p:"p",pre:"pre",...(0,i.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsxs)(t.p,{children:["The mod.io Unreal Engine Plugin supports all of the mod.io metrics features, allowing you to start a metrics play sesion, keeping that session alive via a heartbeat (automatically called, or manually handled) and then ending that session. Metric sessions allow you to track which mods your players interact with most frequently. Visit ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/metrics/",children:"https://docs.mod.io/metrics/"})," for an overview of the mod.io metrics system."]}),"\n",(0,n.jsx)(t.admonition,{type:"note",children:(0,n.jsxs)(t.p,{children:["Running metrics play sessions is a premium feature. If you are interested in mod.io premium features, please contact ",(0,n.jsx)(t.a,{href:"mailto:developers@mod.io",children:"developers@mod.io"}),"."]})}),"\n",(0,n.jsxs)(t.p,{children:["The Metrics session functionality is all accessible through the ",(0,n.jsx)(t.code,{children:"ModioSubsystem"})," class."]}),"\n",(0,n.jsx)(t.p,{children:"Metrics based on the platform and portal, are transparently taken care of with no additional consideration needed when using the SDK."}),"\n",(0,n.jsx)(t.h4,{id:"initialization",children:"Initialization"}),"\n",(0,n.jsxs)(t.p,{children:["The mod.io metrics features are enabled as part of generating a Metrics Secret Key your API settings in your game dashboard, e.g. ",(0,n.jsx)(t.a,{href:"https://mod.io/g/game-name/admin/api-key",children:"https://mod.io/g/game-name/admin/api-key"}),". Once this key has been generated, you need to pass it in as an ExtendedInitializationParameters entry in your InitializeOptions when Initializing the mod.io SDK, e.g.:"]}),"\n",(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:'FModioInitializeOptions initializeOptions;\ninitializeOptions.ExtendedInitializationParameters.Add("MetricsSecretKey", "00000000-1111-2222-3333-444444444444");\n'})}),"\n",(0,n.jsx)(t.p,{children:"Failing to set up the Metrics Secret Key will result in a `SessionNotInitialized`` error being returned when using the metrics functionality."}),"\n",(0,n.jsx)(t.h4,{id:"starting-a-metrics-session",children:"Starting a Metrics Session"}),"\n",(0,n.jsxs)(t.p,{children:["You can call ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#metricssessionstartasync",children:"MetricsSessionStartAsync"})," to start a new session tracking the usage of mods in the context of your game. You'll notice that ",(0,n.jsx)(t.code,{children:"MetricsSessionStartAsync"})," takes a ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#modiometricssessionparams",children:"MetricsSessionParams"})," struct as its parameter. This contains an optional Session Id, as well as a required vector of mods to track."]}),"\n",(0,n.jsx)(t.admonition,{type:"note",children:(0,n.jsx)(t.p,{children:"If a Session Id is not provided, a random one will be created for you."})}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsxs)(r.A,{value:"blueprint",label:"Blueprint",children:[(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_start_1",src:s(82326).A+"",width:"769",height:"513"})}),(0,n.jsx)(t.admonition,{type:"note",children:(0,n.jsxs)(t.p,{children:["Use the ",(0,n.jsx)(t.code,{children:"SetSessionId"})," node if you want to associate your own Id with this session."]})}),(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_start_2",src:s(73005).A+"",width:"1025",height:"513"})})]}),(0,n.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"FModioMetricsSessionParams Params = FModioMetricsSessionParams(ModIds);\n\nvoid UModioManager::MetricsSessionStart(const FModioMetricsSessionParams& Params)\n{\n\tGEngine->GetEngineSubsystem()->MetricsSessionStartAsync(\n\t\tParams, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionStartCallback));\n}\n"})})})]}),"\n",(0,n.jsxs)(t.p,{children:["The Metrics Session Params accepts an optional Session Id in the form of a ",(0,n.jsx)(t.code,{children:"FModioGuid"})," which you may want to use to associate the new session with any supplementary telemetry you are gathering in your game."]}),"\n",(0,n.jsx)(t.h4,{id:"metrics-heartbeat",children:"Metrics heartbeat"}),"\n",(0,n.jsx)(t.p,{children:"To ensure that the session is kept alive, a heartbeat is required to be submitted at most every 5 minutes. We recommend doing this a bit earlier than the threshold to ensure you do not miss the window."}),"\n",(0,n.jsxs)(t.p,{children:["There are two methods provided to control the behaviour of the heart beat, ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#metricssessionsendheartbeatonceasync",children:"MetricsSessionSendHeartbeatOnceAsync"})," which you can call at your desired precision, as well as a single call and forget ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#metricssessionsendheartbeatatintervalasync",children:"MetricsSessionSendHeartbeatAtIntervalAsync"})," with a desired interval."]}),"\n",(0,n.jsxs)(t.p,{children:["Calling ",(0,n.jsx)(t.code,{children:"MetricsSessionSendHeartbeatOnceAsync"})," will submit a single heartbeat, and return an error code if something went wrong. If no error has occured, the heartbeat has been successfully sent."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_heartbeat",src:s(64202).A+"",width:"641",height:"385"})})}),(0,n.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"void UModioManager::MetricsSessionHeartbeat()\n{\n\tGEngine->GetEngineSubsystem()->MetricsSessionSendHeartbeatOnceAsync(\n\t\tFOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionHeartbeatCallback));\n}\n"})})})]}),"\n",(0,n.jsxs)(t.p,{children:["Calling ",(0,n.jsx)(t.code,{children:"MetricsSessionSendHeartbeatAtIntervalAsync"})," requires a parameter with the desired interval frequency in seconds. An error code will be returned if something went wrong, otherwise you will receive a false-y error if the interval loop has been shut down successfully (such as ending a session)."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_heartbeat_interval",src:s(25772).A+"",width:"641",height:"385"})})}),(0,n.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"const FModioUnsigned64 IntervalSeconds(150);\n\nvoid UModioManager::MetricsSessionHeartbeat(const FModioUnsigned64& IntervalSeconds)\n{\n\tGEngine->GetEngineSubsystem()->MetricsSessionSendHeartbeatAtIntervalAsync(\n\t\tIntervalSeconds, \n\t\tFOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionHeartbeatCallback));\n}\n"})})})]}),"\n",(0,n.jsx)(t.h4,{id:"ending-a-metrics-session",children:"Ending a Metrics Session"}),"\n",(0,n.jsxs)(t.p,{children:["To complete a session, for example when finishing a match, or quitting out of your game, you can call ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#metricssessionendasync",children:"MetricsSessionEndAsync"}),".\nAs with the other calls, you will receive an error if anything has gone wrong, otherwise the operation successfully completed."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_end",src:s(93089).A+"",width:"513",height:"385"})})}),(0,n.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"void UModioManager::MetricsSessionHeartbeat()\n{\n\tGEngine->GetEngineSubsystem()->MetricsSessionEndAsync(\n\t\tFOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionHeartbeatCallback));\n}\n"})})})]})]})}function m(e={}){const{wrapper:t}={...(0,i.R)(),...e.components};return t?(0,n.jsx)(t,{...e,children:(0,n.jsx)(h,{...e})}):h(e)}},19365:(e,t,s)=>{s.d(t,{A:()=>r});s(96540);var n=s(18215);const i={tabItem:"tabItem_Ymn6"};var a=s(74848);function r(e){let{children:t,hidden:s,className:r}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,n.A)(i.tabItem,r),hidden:s,children:t})}},11470:(e,t,s)=>{s.d(t,{A:()=>w});var n=s(96540),i=s(18215),a=s(23104),r=s(56347),o=s(205),l=s(57485),c=s(31682),d=s(70679);function u(e){return n.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,n.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:t,children:s}=e;return(0,n.useMemo)((()=>{const e=t??function(e){return u(e).map((e=>{let{props:{value:t,label:s,attributes:n,default:i}}=e;return{value:t,label:s,attributes:n,default:i}}))}(s);return function(e){const t=(0,c.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,s])}function m(e){let{value:t,tabValues:s}=e;return s.some((e=>e.value===t))}function p(e){let{queryString:t=!1,groupId:s}=e;const i=(0,r.W6)(),a=function(e){let{queryString:t=!1,groupId:s}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!s)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return s??null}({queryString:t,groupId:s});return[(0,l.aZ)(a),(0,n.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(i.location.search);t.set(a,e),i.replace({...i.location,search:t.toString()})}),[a,i])]}function g(e){const{defaultValue:t,queryString:s=!1,groupId:i}=e,a=h(e),[r,l]=(0,n.useState)((()=>function(e){let{defaultValue:t,tabValues:s}=e;if(0===s.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:s}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${s.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const n=s.find((e=>e.default))??s[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:t,tabValues:a}))),[c,u]=p({queryString:s,groupId:i}),[g,b]=function(e){let{groupId:t}=e;const s=function(e){return e?`docusaurus.tab.${e}`:null}(t),[i,a]=(0,d.Dv)(s);return[i,(0,n.useCallback)((e=>{s&&a.set(e)}),[s,a])]}({groupId:i}),f=(()=>{const e=c??g;return m({value:e,tabValues:a})?e:null})();(0,o.A)((()=>{f&&l(f)}),[f]);return{selectedValue:r,selectValue:(0,n.useCallback)((e=>{if(!m({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),u(e),b(e)}),[u,b,a]),tabValues:a}}var b=s(92303);const f={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var y=s(74848);function v(e){let{className:t,block:s,selectedValue:n,selectValue:r,tabValues:o}=e;const l=[],{blockElementScrollPositionUntilNextRender:c}=(0,a.a_)(),d=e=>{const t=e.currentTarget,s=l.indexOf(t),i=o[s].value;i!==n&&(c(t),r(i))},u=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const s=l.indexOf(e.currentTarget)+1;t=l[s]??l[0];break}case"ArrowLeft":{const s=l.indexOf(e.currentTarget)-1;t=l[s]??l[l.length-1];break}}t?.focus()};return(0,y.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,i.A)("tabs",{"tabs--block":s},t),children:o.map((e=>{let{value:t,label:s,attributes:a}=e;return(0,y.jsx)("li",{role:"tab",tabIndex:n===t?0:-1,"aria-selected":n===t,ref:e=>l.push(e),onKeyDown:u,onClick:d,...a,className:(0,i.A)("tabs__item",f.tabItem,a?.className,{"tabs__item--active":n===t}),children:s??t},t)}))})}function x(e){let{lazy:t,children:s,selectedValue:i}=e;const a=(Array.isArray(s)?s:[s]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===i));return e?(0,n.cloneElement)(e,{className:"margin-top--md"}):null}return(0,y.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,n.cloneElement)(e,{key:t,hidden:e.props.value!==i})))})}function j(e){const t=g(e);return(0,y.jsxs)("div",{className:(0,i.A)("tabs-container",f.tabList),children:[(0,y.jsx)(v,{...t,...e}),(0,y.jsx)(x,{...t,...e})]})}function w(e){const t=(0,b.A)();return(0,y.jsx)(j,{...e,children:u(e.children)},String(t))}},89236:(e,t,s)=>{s.d(t,{A:()=>a});var n=s(19365),i=(s(96540),s(74848));function a(e){return(0,i.jsx)(i.Fragment,{children:(0,i.jsx)(n.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,s)=>{s.d(t,{A:()=>a});var n=s(11470),i=(s(96540),s(74848));function a(e){return(0,i.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,i.jsx)(n.A,{...e})})}},93089:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_end-942fd0dad67f632b70e0d47a6b69ceda.png"},64202:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_heartbeat-6c6cae38cd1ecca90c1a7737010bdd34.png"},25772:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_heartbeat_interval-efbee004400f2d4006fd7d541abd445b.png"},82326:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_start_1-fd94f8b4fbb49ad0ba7fcf28ecf5e8f0.png"},73005:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_start_2-43157714ec2d0e21034cfd939a6f9879.png"},28453:(e,t,s)=>{s.d(t,{R:()=>r,x:()=>o});var n=s(96540);const i={},a=n.createContext(i);function r(e){const t=n.useContext(a);return n.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function o(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:r(e.components),n.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/1da2407e.9927ed81.js b/Doc/assets/js/1da2407e.9927ed81.js deleted file mode 100644 index 06108449..00000000 --- a/Doc/assets/js/1da2407e.9927ed81.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[2381],{84923:(e,t,s)=>{s.r(t),s.d(t,{assets:()=>d,contentTitle:()=>l,default:()=>m,frontMatter:()=>o,metadata:()=>c,toc:()=>u});var n=s(74848),i=s(28453),a=s(27064),r=s(89236);const o={id:"ue-metrics-play-sessions",title:"Metrics Play Sessions",slug:"/unreal/getting-started/metrics-play-sessions",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/metrics-play-sessions.mdx"},l=void 0,c={id:"getting-started/ue-metrics-play-sessions",title:"Metrics Play Sessions",description:"The mod.io Unreal Engine Plugin supports all of the mod.io metrics features, allowing you to start a metrics play sesion, keeping that session alive via a heartbeat (automatically called, or manually handled) and then ending that session. Metric sessions allow you to track which mods your players interact with most frequently. Visit https://docs.mod.io/metrics/ for an overview of the mod.io metrics system.",source:"@site/public/en-us/getting-started/metrics-play-sessions.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/metrics-play-sessions",permalink:"/unreal/getting-started/metrics-play-sessions",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/metrics-play-sessions.mdx",tags:[],version:"current",frontMatter:{id:"ue-metrics-play-sessions",title:"Metrics Play Sessions",slug:"/unreal/getting-started/metrics-play-sessions",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/metrics-play-sessions.mdx"},sidebar:"sidebar",previous:{title:"Muting and Unmuting a User",permalink:"/unreal/getting-started/mute-user"},next:{title:"Marketplace",permalink:"/unreal/marketplace/"}},d={},u=[{value:"Initialization",id:"initialization",level:4},{value:"Starting a Metrics Session",id:"starting-a-metrics-session",level:4},{value:"Metrics heartbeat",id:"metrics-heartbeat",level:4},{value:"Ending a Metrics Session",id:"ending-a-metrics-session",level:4}];function h(e){const t={a:"a",admonition:"admonition",code:"code",h4:"h4",img:"img",p:"p",pre:"pre",...(0,i.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsxs)(t.p,{children:["The mod.io Unreal Engine Plugin supports all of the mod.io metrics features, allowing you to start a metrics play sesion, keeping that session alive via a heartbeat (automatically called, or manually handled) and then ending that session. Metric sessions allow you to track which mods your players interact with most frequently. Visit ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/metrics/",children:"https://docs.mod.io/metrics/"})," for an overview of the mod.io metrics system."]}),"\n",(0,n.jsx)(t.admonition,{type:"note",children:(0,n.jsxs)(t.p,{children:["Running metrics play sessions is a premium feature. If you are interested in mod.io premium features, please contact ",(0,n.jsx)(t.a,{href:"mailto:developers@mod.io",children:"developers@mod.io"}),"."]})}),"\n",(0,n.jsxs)(t.p,{children:["The Metrics session functionality is all accessible through the ",(0,n.jsx)(t.code,{children:"ModioSubsystem"})," class."]}),"\n",(0,n.jsx)(t.p,{children:"Metrics based on the platform and portal, are transparently taken care of with no additional consideration needed when using the SDK."}),"\n",(0,n.jsx)(t.h4,{id:"initialization",children:"Initialization"}),"\n",(0,n.jsxs)(t.p,{children:["The mod.io metrics features are enabled as part of generating a Metrics Secret Key your API settings in your game dashboard, e.g. ",(0,n.jsx)(t.a,{href:"https://mod.io/g/game-name/admin/api-key",children:"https://mod.io/g/game-name/admin/api-key"}),". Once this key has been generated, you need to pass it in as an ExtendedInitializationParameters entry in your InitializeOptions when Initializing the mod.io SDK, e.g.:"]}),"\n",(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:'FModioInitializeOptions initializeOptions;\ninitializeOptions.ExtendedInitializationParameters.Add("MetricsSecretKey", "00000000-1111-2222-3333-444444444444");\n'})}),"\n",(0,n.jsx)(t.p,{children:"Failing to set up the Metrics Secret Key will result in a `SessionNotInitialized`` error being returned when using the metrics functionality."}),"\n",(0,n.jsx)(t.h4,{id:"starting-a-metrics-session",children:"Starting a Metrics Session"}),"\n",(0,n.jsxs)(t.p,{children:["You can call ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#MetricsSessionStartAsync",children:"MetricsSessionStartAsync"})," to start a new session tracking the usage of mods in the context of your game. You'll notice that ",(0,n.jsx)(t.code,{children:"MetricsSessionStartAsync"})," takes a ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#_metricssessionparams",children:"MetricsSessionParams"})," struct as its parameter. This contains an optional Session Id, as well as a required vector of mods to track."]}),"\n",(0,n.jsx)(t.admonition,{type:"note",children:(0,n.jsx)(t.p,{children:"If a Session Id is not provided, a random one will be created for you."})}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsxs)(r.A,{value:"blueprint",label:"Blueprint",children:[(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_start_1",src:s(68731).A+"",width:"769",height:"513"})}),(0,n.jsx)(t.admonition,{type:"note",children:(0,n.jsxs)(t.p,{children:["Use the ",(0,n.jsx)(t.code,{children:"SetSessionId"})," node if you want to associate your own Id with this session."]})}),(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_start_2",src:s(71168).A+"",width:"1025",height:"513"})})]}),(0,n.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"FModioMetricsSessionParams Params = FModioMetricsSessionParams(ModIds);\n\nvoid UModioManager::MetricsSessionStart(const FModioMetricsSessionParams& Params)\n{\n\tGEngine->GetEngineSubsystem()->MetricsSessionStartAsync(\n\t\tParams, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionStartCallback));\n}\n"})})})]}),"\n",(0,n.jsxs)(t.p,{children:["The Metrics Session Params accepts an optional Session Id in the form of a ",(0,n.jsx)(t.code,{children:"FModioGuid"})," which you may want to use to associate the new session with any supplementary telemetry you are gathering in your game."]}),"\n",(0,n.jsx)(t.h4,{id:"metrics-heartbeat",children:"Metrics heartbeat"}),"\n",(0,n.jsx)(t.p,{children:"To ensure that the session is kept alive, a heartbeat is required to be submitted at most every 5 minutes. We recommend doing this a bit earlier than the threshold to ensure you do not miss the window."}),"\n",(0,n.jsxs)(t.p,{children:["There are two methods provided to control the behaviour of the heart beat, ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#MetricsEessionSendHeartbeatOnceAsync",children:"MetricsSessionSendHeartbeatOnceAsync"})," which you can call at your desired precision, as well as a single call and forget ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#MetricsSessionSendHeartbeatAtIntervalAsync",children:"MetricsSessionSendHeartbeatAtIntervalAsync"})," with a desired interval."]}),"\n",(0,n.jsxs)(t.p,{children:["Calling ",(0,n.jsx)(t.code,{children:"MetricsSessionSendHeartbeatOnceAsync"})," will submit a single heartbeat, and return an error code if something went wrong. If no error has occured, the heartbeat has been successfully sent."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_heartbeat",src:s(21291).A+"",width:"641",height:"385"})})}),(0,n.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"void UModioManager::MetricsSessionHeartbeat()\n{\n\tGEngine->GetEngineSubsystem()->MetricsSessionSendHeartbeatOnceAsync(\n\t\tFOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionHeartbeatCallback));\n}\n"})})})]}),"\n",(0,n.jsxs)(t.p,{children:["Calling ",(0,n.jsx)(t.code,{children:"MetricsSessionSendHeartbeatAtIntervalAsync"})," requires a parameter with the desired interval frequency in seconds. An error code will be returned if something went wrong, otherwise you will receive a false-y error if the interval loop has been shut down successfully (such as ending a session)."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_heartbeat_interval",src:s(6643).A+"",width:"641",height:"385"})})}),(0,n.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"const FModioUnsigned64 IntervalSeconds(150);\n\nvoid UModioManager::MetricsSessionHeartbeat(const FModioUnsigned64& IntervalSeconds)\n{\n\tGEngine->GetEngineSubsystem()->MetricsSessionSendHeartbeatAtIntervalAsync(\n\t\tIntervalSeconds, \n\t\tFOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionHeartbeatCallback));\n}\n"})})})]}),"\n",(0,n.jsx)(t.h4,{id:"ending-a-metrics-session",children:"Ending a Metrics Session"}),"\n",(0,n.jsxs)(t.p,{children:["To complete a session, for example when finishing a match, or quitting out of your game, you can call ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#MetricsSessionEndAsync",children:"MetricsSessionEndAsync"}),".\nAs with the other calls, you will receive an error if anything has gone wrong, otherwise the operation successfully completed."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"metrics_session_end",src:s(46596).A+"",width:"513",height:"385"})})}),(0,n.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"void UModioManager::MetricsSessionHeartbeat()\n{\n\tGEngine->GetEngineSubsystem()->MetricsSessionEndAsync(\n\t\tFOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionHeartbeatCallback));\n}\n"})})})]})]})}function m(e={}){const{wrapper:t}={...(0,i.R)(),...e.components};return t?(0,n.jsx)(t,{...e,children:(0,n.jsx)(h,{...e})}):h(e)}},19365:(e,t,s)=>{s.d(t,{A:()=>r});s(96540);var n=s(18215);const i={tabItem:"tabItem_Ymn6"};var a=s(74848);function r(e){let{children:t,hidden:s,className:r}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,n.A)(i.tabItem,r),hidden:s,children:t})}},11470:(e,t,s)=>{s.d(t,{A:()=>S});var n=s(96540),i=s(18215),a=s(23104),r=s(56347),o=s(205),l=s(57485),c=s(31682),d=s(70679);function u(e){return n.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,n.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:t,children:s}=e;return(0,n.useMemo)((()=>{const e=t??function(e){return u(e).map((e=>{let{props:{value:t,label:s,attributes:n,default:i}}=e;return{value:t,label:s,attributes:n,default:i}}))}(s);return function(e){const t=(0,c.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,s])}function m(e){let{value:t,tabValues:s}=e;return s.some((e=>e.value===t))}function p(e){let{queryString:t=!1,groupId:s}=e;const i=(0,r.W6)(),a=function(e){let{queryString:t=!1,groupId:s}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!s)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return s??null}({queryString:t,groupId:s});return[(0,l.aZ)(a),(0,n.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(i.location.search);t.set(a,e),i.replace({...i.location,search:t.toString()})}),[a,i])]}function g(e){const{defaultValue:t,queryString:s=!1,groupId:i}=e,a=h(e),[r,l]=(0,n.useState)((()=>function(e){let{defaultValue:t,tabValues:s}=e;if(0===s.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:s}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${s.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const n=s.find((e=>e.default))??s[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:t,tabValues:a}))),[c,u]=p({queryString:s,groupId:i}),[g,b]=function(e){let{groupId:t}=e;const s=function(e){return e?`docusaurus.tab.${e}`:null}(t),[i,a]=(0,d.Dv)(s);return[i,(0,n.useCallback)((e=>{s&&a.set(e)}),[s,a])]}({groupId:i}),f=(()=>{const e=c??g;return m({value:e,tabValues:a})?e:null})();(0,o.A)((()=>{f&&l(f)}),[f]);return{selectedValue:r,selectValue:(0,n.useCallback)((e=>{if(!m({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),u(e),b(e)}),[u,b,a]),tabValues:a}}var b=s(92303);const f={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var y=s(74848);function v(e){let{className:t,block:s,selectedValue:n,selectValue:r,tabValues:o}=e;const l=[],{blockElementScrollPositionUntilNextRender:c}=(0,a.a_)(),d=e=>{const t=e.currentTarget,s=l.indexOf(t),i=o[s].value;i!==n&&(c(t),r(i))},u=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const s=l.indexOf(e.currentTarget)+1;t=l[s]??l[0];break}case"ArrowLeft":{const s=l.indexOf(e.currentTarget)-1;t=l[s]??l[l.length-1];break}}t?.focus()};return(0,y.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,i.A)("tabs",{"tabs--block":s},t),children:o.map((e=>{let{value:t,label:s,attributes:a}=e;return(0,y.jsx)("li",{role:"tab",tabIndex:n===t?0:-1,"aria-selected":n===t,ref:e=>l.push(e),onKeyDown:u,onClick:d,...a,className:(0,i.A)("tabs__item",f.tabItem,a?.className,{"tabs__item--active":n===t}),children:s??t},t)}))})}function x(e){let{lazy:t,children:s,selectedValue:i}=e;const a=(Array.isArray(s)?s:[s]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===i));return e?(0,n.cloneElement)(e,{className:"margin-top--md"}):null}return(0,y.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,n.cloneElement)(e,{key:t,hidden:e.props.value!==i})))})}function j(e){const t=g(e);return(0,y.jsxs)("div",{className:(0,i.A)("tabs-container",f.tabList),children:[(0,y.jsx)(v,{...t,...e}),(0,y.jsx)(x,{...t,...e})]})}function S(e){const t=(0,b.A)();return(0,y.jsx)(j,{...e,children:u(e.children)},String(t))}},89236:(e,t,s)=>{s.d(t,{A:()=>a});var n=s(19365),i=(s(96540),s(74848));function a(e){return(0,i.jsx)(i.Fragment,{children:(0,i.jsx)(n.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,s)=>{s.d(t,{A:()=>a});var n=s(11470),i=(s(96540),s(74848));function a(e){return(0,i.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,i.jsx)(n.A,{...e})})}},46596:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_end-942fd0dad67f632b70e0d47a6b69ceda.png"},21291:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_heartbeat-6c6cae38cd1ecca90c1a7737010bdd34.png"},6643:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_heartbeat_interval-efbee004400f2d4006fd7d541abd445b.png"},68731:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_start_1-fd94f8b4fbb49ad0ba7fcf28ecf5e8f0.png"},71168:(e,t,s)=>{s.d(t,{A:()=>n});const n=s.p+"assets/images/metrics_session_start_2-43157714ec2d0e21034cfd939a6f9879.png"},28453:(e,t,s)=>{s.d(t,{R:()=>r,x:()=>o});var n=s(96540);const i={},a=n.createContext(i);function r(e){const t=n.useContext(a);return n.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function o(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:r(e.components),n.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/2969dc70.3fd40665.js b/Doc/assets/js/2969dc70.3fd40665.js deleted file mode 100644 index 7503f400..00000000 --- a/Doc/assets/js/2969dc70.3fd40665.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[5237],{34348:(e,d,s)=>{s.r(d),s.d(d,{assets:()=>c,contentTitle:()=>t,default:()=>a,frontMatter:()=>n,metadata:()=>l,toc:()=>o});var i=s(74848),r=s(28453);const n={id:"ue-refdocs",title:"Unreal Plugin API Reference",slug:"/unreal/refdocs/"},t=void 0,l={id:"ue-refdocs",title:"Unreal Plugin API Reference",description:"Classes",source:"@site/public/en-us/generated-refdocs.mdx",sourceDirName:".",slug:"/unreal/refdocs/",permalink:"/unreal/refdocs/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-docs/tree/main/public/en-us/generated-refdocs.mdx",tags:[],version:"current",frontMatter:{id:"ue-refdocs",title:"Unreal Plugin API Reference",slug:"/unreal/refdocs/"},sidebar:"sidebar",previous:{title:"Android Configuration",permalink:"/unreal/android-configuration/"}},c={},o=[{value:"Classes",id:"classes",level:2},{value:"ModioCommonTypesLibrary",id:"modiocommontypeslibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy",level:5},{value:"ModioCreateModLibrary",id:"modiocreatemodlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-1",level:5},{value:"ModioErrorCodeLibrary",id:"modioerrorcodelibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-2",level:5},{value:"ModioErrorConditionLibrary",id:"modioerrorconditionlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-3",level:5},{value:"ModioExampleLibrary",id:"modioexamplelibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-4",level:5},{value:"ModioImageLibrary",id:"modioimagelibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-5",level:5},{value:"ModioModCollectionLibrary",id:"modiomodcollectionlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-6",level:5},{value:"ModioModProgressInfoLibrary",id:"modiomodprogressinfolibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-7",level:5},{value:"ModioModTagOptionsLibrary",id:"modiomodtagoptionslibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-8",level:5},{value:"ModioPlatformHelpersLibrary",id:"modioplatformhelperslibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-9",level:5},{value:"ModioPresetFilterParamsLibrary",id:"modiopresetfilterparamslibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-10",level:5},{value:"ModioSDKLibrary",id:"modiosdklibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-11",level:5},{value:"ModioSubmissionExtensionLibrary",id:"modiosubmissionextensionlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-12",level:5},{value:"SubmitModChangesAsync",id:"submitmodchangesasync",level:4},{value:"Requirements",id:"requirements",level:5},{value:"Parameters",id:"parameters",level:5},{value:"Error Values",id:"error-values",level:5},{value:"ModioSubsystem",id:"modiosubsystem",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-13",level:5},{value:"Set Log Level",id:"set-log-level",level:4},{value:"Parameters",id:"parameters-1",level:5},{value:"Run Pending Handlers",id:"run-pending-handlers",level:4},{value:"Parameters",id:"parameters-2",level:5},{value:"Query User Subscriptions",id:"query-user-subscriptions",level:4},{value:"Parameters",id:"parameters-3",level:5},{value:"Returns",id:"returns",level:5},{value:"Query User Purchased Mods",id:"query-user-purchased-mods",level:4},{value:"Requirements",id:"requirements-1",level:5},{value:"Parameters",id:"parameters-4",level:5},{value:"Returns",id:"returns-1",level:5},{value:"Query User Installations",id:"query-user-installations",level:4},{value:"Parameters",id:"parameters-5",level:5},{value:"Returns",id:"returns-2",level:5},{value:"Query System Installations",id:"query-system-installations",level:4},{value:"Parameters",id:"parameters-6",level:5},{value:"Returns",id:"returns-3",level:5},{value:"Prioritize Transfer for Mod",id:"prioritize-transfer-for-mod",level:4},{value:"Requirements",id:"requirements-2",level:5},{value:"Parameters",id:"parameters-7",level:5},{value:"Returns",id:"returns-4",level:5},{value:"Error Values",id:"error-values-1",level:5},{value:"Kill Background Thread",id:"kill-background-thread",level:4},{value:"VerifyUserAuthenticationAsync",id:"verifyuserauthenticationasync",level:4},{value:"Requirements",id:"requirements-3",level:5},{value:"Parameters",id:"parameters-8",level:5},{value:"Error Values",id:"error-values-2",level:5},{value:"UnsubscribeFromModAsync",id:"unsubscribefrommodasync",level:4},{value:"Requirements",id:"requirements-4",level:5},{value:"Parameters",id:"parameters-9",level:5},{value:"Error Values",id:"error-values-3",level:5},{value:"UnmuteUserAsync",id:"unmuteuserasync",level:4},{value:"Requirements",id:"requirements-5",level:5},{value:"Parameters",id:"parameters-10",level:5},{value:"Error Values",id:"error-values-4",level:5},{value:"SubscribeToModAsync",id:"subscribetomodasync",level:4},{value:"Requirements",id:"requirements-6",level:5},{value:"Parameters",id:"parameters-11",level:5},{value:"Error Values",id:"error-values-5",level:5},{value:"SubmitNewModFileForMod",id:"submitnewmodfileformod",level:4},{value:"Requirements",id:"requirements-7",level:5},{value:"Parameters",id:"parameters-12",level:5},{value:"Error Values",id:"error-values-6",level:5},{value:"SubmitNewModAsync",id:"submitnewmodasync",level:4},{value:"Requirements",id:"requirements-8",level:5},{value:"Parameters",id:"parameters-13",level:5},{value:"Error Values",id:"error-values-7",level:5},{value:"SubmitModRatingAsync",id:"submitmodratingasync",level:4},{value:"Requirements",id:"requirements-9",level:5},{value:"Parameters",id:"parameters-14",level:5},{value:"Error Values",id:"error-values-8",level:5},{value:"SubmitModChangesAsync",id:"submitmodchangesasync-1",level:4},{value:"Requirements",id:"requirements-10",level:5},{value:"Parameters",id:"parameters-15",level:5},{value:"Error Values",id:"error-values-9",level:5},{value:"ShutdownAsync",id:"shutdownasync",level:4},{value:"Parameters",id:"parameters-16",level:5},{value:"SetLanguage",id:"setlanguage",level:4},{value:"Parameters",id:"parameters-17",level:5},{value:"RequestEmailAuthCodeAsync",id:"requestemailauthcodeasync",level:4},{value:"Requirements",id:"requirements-11",level:5},{value:"Parameters",id:"parameters-18",level:5},{value:"Error Values",id:"error-values-10",level:5},{value:"ReportContentAsync",id:"reportcontentasync",level:4},{value:"Requirements",id:"requirements-12",level:5},{value:"Parameters",id:"parameters-19",level:5},{value:"Error Values",id:"error-values-11",level:5},{value:"RemoveFromTempModSet",id:"removefromtempmodset",level:4},{value:"Requirements",id:"requirements-13",level:5},{value:"Parameters",id:"parameters-20",level:5},{value:"Returns",id:"returns-5",level:5},{value:"Error Values",id:"error-values-12",level:5},{value:"RefreshUserEntitlementsAsync",id:"refreshuserentitlementsasync",level:4},{value:"Requirements",id:"requirements-14",level:5},{value:"Parameters",id:"parameters-21",level:5},{value:"QueryUserProfile",id:"queryuserprofile",level:4},{value:"Parameters",id:"parameters-22",level:5},{value:"Returns",id:"returns-6",level:5},{value:"QueryTempModSet",id:"querytempmodset",level:4},{value:"Parameters",id:"parameters-23",level:5},{value:"Returns",id:"returns-7",level:5},{value:"QueryCurrentModUpdate",id:"querycurrentmodupdate",level:4},{value:"Parameters",id:"parameters-24",level:5},{value:"Returns",id:"returns-8",level:5},{value:"PurchaseModAsync",id:"purchasemodasync",level:4},{value:"Requirements",id:"requirements-15",level:5},{value:"Parameters",id:"parameters-25",level:5},{value:"PreviewExternalUpdatesAsync",id:"previewexternalupdatesasync",level:4},{value:"Parameters",id:"parameters-26",level:5},{value:"MuteUserAsync",id:"muteuserasync",level:4},{value:"Requirements",id:"requirements-16",level:5},{value:"Parameters",id:"parameters-27",level:5},{value:"Error Values",id:"error-values-13",level:5},{value:"MetricsSessionStartAsync",id:"metricssessionstartasync",level:4},{value:"Parameters",id:"parameters-28",level:5},{value:"Error Values",id:"error-values-14",level:5},{value:"MetricsSessionSendHeartbeatOnceAsync",id:"metricssessionsendheartbeatonceasync",level:4},{value:"Parameters",id:"parameters-29",level:5},{value:"Error Values",id:"error-values-15",level:5},{value:"MetricsSessionSendHeartbeatAtIntervalAsync",id:"metricssessionsendheartbeatatintervalasync",level:4},{value:"Parameters",id:"parameters-30",level:5},{value:"Error Values",id:"error-values-16",level:5},{value:"MetricsSessionEndAsync",id:"metricssessionendasync",level:4},{value:"Parameters",id:"parameters-31",level:5},{value:"Error Values",id:"error-values-17",level:5},{value:"ListUserGamesAsync",id:"listusergamesasync",level:4},{value:"Requirements",id:"requirements-17",level:5},{value:"Parameters",id:"parameters-32",level:5},{value:"Error Values",id:"error-values-18",level:5},{value:"ListUserCreatedModsAsync",id:"listusercreatedmodsasync",level:4},{value:"Requirements",id:"requirements-18",level:5},{value:"Parameters",id:"parameters-33",level:5},{value:"Error Values",id:"error-values-19",level:5},{value:"ListAllModsAsync",id:"listallmodsasync",level:4},{value:"Requirements",id:"requirements-19",level:5},{value:"Parameters",id:"parameters-34",level:5},{value:"Error Values",id:"error-values-20",level:5},{value:"InitTempModSet",id:"inittempmodset",level:4},{value:"Requirements",id:"requirements-20",level:5},{value:"Parameters",id:"parameters-35",level:5},{value:"Returns",id:"returns-9",level:5},{value:"Error Values",id:"error-values-21",level:5},{value:"InitializeAsync",id:"initializeasync",level:4},{value:"Parameters",id:"parameters-36",level:5},{value:"Error Values",id:"error-values-22",level:5},{value:"GetUserWalletBalanceAsync",id:"getuserwalletbalanceasync",level:4},{value:"Requirements",id:"requirements-21",level:5},{value:"Parameters",id:"parameters-37",level:5},{value:"GetUserMediaAsync (Avatar)",id:"getusermediaasync-avatar",level:4},{value:"Requirements",id:"requirements-22",level:5},{value:"Parameters",id:"parameters-38",level:5},{value:"Error Values",id:"error-values-23",level:5},{value:"GetUserDelegationTokenAsync",id:"getuserdelegationtokenasync",level:4},{value:"Requirements",id:"requirements-23",level:5},{value:"Parameters",id:"parameters-39",level:5},{value:"GetTermsOfUseAsync",id:"gettermsofuseasync",level:4},{value:"Requirements",id:"requirements-24",level:5},{value:"Parameters",id:"parameters-40",level:5},{value:"Error Values",id:"error-values-24",level:5},{value:"GetMutedUsersAsync",id:"getmutedusersasync",level:4},{value:"Requirements",id:"requirements-25",level:5},{value:"Parameters",id:"parameters-41",level:5},{value:"Error Values",id:"error-values-25",level:5},{value:"GetModTagOptionsAsync",id:"getmodtagoptionsasync",level:4},{value:"Requirements",id:"requirements-26",level:5},{value:"Parameters",id:"parameters-42",level:5},{value:"Error Values",id:"error-values-26",level:5},{value:"GetModMediaAsync (Logo)",id:"getmodmediaasync-logo",level:4},{value:"Requirements",id:"requirements-27",level:5},{value:"Parameters",id:"parameters-43",level:5},{value:"Error Values",id:"error-values-27",level:5},{value:"GetModMediaAsync (Gallery Image)",id:"getmodmediaasync-gallery-image",level:4},{value:"Requirements",id:"requirements-28",level:5},{value:"Parameters",id:"parameters-44",level:5},{value:"Error Values",id:"error-values-28",level:5},{value:"GetModMediaAsync (Avatar)",id:"getmodmediaasync-avatar",level:4},{value:"Requirements",id:"requirements-29",level:5},{value:"Parameters",id:"parameters-45",level:5},{value:"Error Values",id:"error-values-29",level:5},{value:"GetModInfoAsync",id:"getmodinfoasync",level:4},{value:"Requirements",id:"requirements-30",level:5},{value:"Parameters",id:"parameters-46",level:5},{value:"Error Values",id:"error-values-30",level:5},{value:"GetModDependenciesAsync",id:"getmoddependenciesasync",level:4},{value:"Requirements",id:"requirements-31",level:5},{value:"Parameters",id:"parameters-47",level:5},{value:"Error Values",id:"error-values-31",level:5},{value:"GetModCreationHandle",id:"getmodcreationhandle",level:4},{value:"Parameters",id:"parameters-48",level:5},{value:"GetLanguage",id:"getlanguage",level:4},{value:"Parameters",id:"parameters-49",level:5},{value:"Returns",id:"returns-10",level:5},{value:"GetGameInfoAsync",id:"getgameinfoasync",level:4},{value:"Requirements",id:"requirements-32",level:5},{value:"Parameters",id:"parameters-50",level:5},{value:"Error Values",id:"error-values-32",level:5},{value:"ForceUninstallModAsync",id:"forceuninstallmodasync",level:4},{value:"Parameters",id:"parameters-51",level:5},{value:"Error Values",id:"error-values-33",level:5},{value:"FetchUserPurchasesAsync",id:"fetchuserpurchasesasync",level:4},{value:"Requirements",id:"requirements-33",level:5},{value:"Parameters",id:"parameters-52",level:5},{value:"FetchExternalUpdatesAsync",id:"fetchexternalupdatesasync",level:4},{value:"Parameters",id:"parameters-53",level:5},{value:"EnableModManagement",id:"enablemodmanagement",level:4},{value:"Parameters",id:"parameters-54",level:5},{value:"Returns",id:"returns-11",level:5},{value:"Error Values",id:"error-values-34",level:5},{value:"CloseTempModSet",id:"closetempmodset",level:4},{value:"Requirements",id:"requirements-34",level:5},{value:"Parameters",id:"parameters-55",level:5},{value:"Returns",id:"returns-12",level:5},{value:"Error Values",id:"error-values-35",level:5},{value:"ClearUserDataAsync",id:"clearuserdataasync",level:4},{value:"Requirements",id:"requirements-35",level:5},{value:"Parameters",id:"parameters-56",level:5},{value:"Error Values",id:"error-values-36",level:5},{value:"AuthenticateUserExternalAsync",id:"authenticateuserexternalasync",level:4},{value:"Requirements",id:"requirements-36",level:5},{value:"Parameters",id:"parameters-57",level:5},{value:"Error Values",id:"error-values-37",level:5},{value:"AuthenticateUserEmailAsync",id:"authenticateuseremailasync",level:4},{value:"Requirements",id:"requirements-37",level:5},{value:"Parameters",id:"parameters-58",level:5},{value:"Error Values",id:"error-values-38",level:5},{value:"ArchiveModAsync",id:"archivemodasync",level:4},{value:"Requirements",id:"requirements-38",level:5},{value:"Parameters",id:"parameters-59",level:5},{value:"Error Values",id:"error-values-39",level:5},{value:"AddToTempModSet",id:"addtotempmodset",level:4},{value:"Requirements",id:"requirements-39",level:5},{value:"Parameters",id:"parameters-60",level:5},{value:"Returns",id:"returns-13",level:5},{value:"Error Values",id:"error-values-40",level:5},{value:"Is Using Background Thread",id:"is-using-background-thread",level:4},{value:"Is Mod Management Busy",id:"is-mod-management-busy",level:4},{value:"Parameters",id:"parameters-61",level:5},{value:"Returns",id:"returns-14",level:5},{value:"Get Last Validation Error",id:"get-last-validation-error",level:4},{value:"Requirements",id:"requirements-40",level:5},{value:"Parameters",id:"parameters-62",level:5},{value:"Returns",id:"returns-15",level:5},{value:"Disable Mod Management",id:"disable-mod-management",level:4},{value:"Parameters",id:"parameters-63",level:5},{value:"ModioUIEnumLocalizationLibrary",id:"modiouienumlocalizationlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-14",level:5},{value:"ModioUILocalizationLibrary",id:"modiouilocalizationlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-15",level:5},{value:"ModioUnsigned64Library",id:"modiounsigned64library",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-16",level:5},{value:"Structs",id:"structs",level:2},{value:"ModioGameInfoList",id:"modiogameinfolist",level:3},{value:"ModioGameInfo",id:"modiogameinfo",level:3},{value:"Variables",id:"variables",level:5},{value:"ModioModTagInfo",id:"modiomodtaginfo",level:3},{value:"Variables",id:"variables-1",level:5},{value:"ModioLocalizedTagCategory",id:"modiolocalizedtagcategory",level:3},{value:"Variables",id:"variables-2",level:5},{value:"ModioModTagLocalizationData",id:"modiomodtaglocalizationdata",level:3},{value:"Variables",id:"variables-3",level:5},{value:"ModioGamePlatform",id:"modiogameplatform",level:3},{value:"Variables",id:"variables-4",level:5},{value:"ModioOtherUrl",id:"modiootherurl",level:3},{value:"Variables",id:"variables-5",level:5},{value:"ModioGameStats",id:"modiogamestats",level:3},{value:"Variables",id:"variables-6",level:5},{value:"ModioGameID",id:"modiogameid",level:3},{value:"ModioTheme",id:"modiotheme",level:3},{value:"Variables",id:"variables-7",level:5},{value:"ModioHeaderImage",id:"modioheaderimage",level:3},{value:"Variables",id:"variables-8",level:5},{value:"ModioLogo",id:"modiologo",level:3},{value:"Variables",id:"variables-9",level:5},{value:"ModioIcon",id:"modioicon",level:3},{value:"Variables",id:"variables-10",level:5},{value:"ModioPagedResult",id:"modiopagedresult",level:3},{value:"Variables",id:"variables-11",level:5},{value:"ModioModInfoList",id:"modiomodinfolist",level:3},{value:"ModioModInfo",id:"modiomodinfo",level:3},{value:"Variables",id:"variables-12",level:5},{value:"ModioUnsigned64",id:"modiounsigned64",level:3},{value:"ModioModStats",id:"modiomodstats",level:3},{value:"Variables",id:"variables-13",level:5},{value:"ModioSketchfabURLList",id:"modiosketchfaburllist",level:3},{value:"ModioYoutubeURLList",id:"modioyoutubeurllist",level:3},{value:"ModioModTag",id:"modiomodtag",level:3},{value:"Variables",id:"variables-14",level:5},{value:"ModioMetadata",id:"modiometadata",level:3},{value:"Variables",id:"variables-15",level:5},{value:"ModioFileMetadata",id:"modiofilemetadata",level:3},{value:"Variables",id:"variables-16",level:5},{value:"ModioModID",id:"modiomodid",level:3},{value:"ModioFileMetadataID",id:"modiofilemetadataid",level:3},{value:"ModioUser",id:"modiouser",level:3},{value:"Variables",id:"variables-17",level:5},{value:"ModioUserID",id:"modiouserid",level:3},{value:"ModioModTagOptions",id:"modiomodtagoptions",level:3},{value:"ModioErrorCode",id:"modioerrorcode",level:3},{value:"ModioOptionalGameInfo",id:"modiooptionalgameinfo",level:3},{value:"ModioOptionalImage",id:"modiooptionalimage",level:3},{value:"ModioOptionalModDependencyList",id:"modiooptionalmoddependencylist",level:3},{value:"ModioOptionalModInfo",id:"modiooptionalmodinfo",level:3},{value:"ModioOptionalModTagOptions",id:"modiooptionalmodtagoptions",level:3},{value:"ModioOptionalTerms",id:"modiooptionalterms",level:3},{value:"ModioOptionalUInt64",id:"modiooptionaluint64",level:3},{value:"ModioOptionalModInfoList",id:"modiooptionalmodinfolist",level:3},{value:"ModioOptionalGameInfoList",id:"modiooptionalgameinfolist",level:3},{value:"ModioModManagementEvent",id:"modiomodmanagementevent",level:3},{value:"Variables",id:"variables-18",level:5},{value:"ModioOptionalUserList",id:"modiooptionaluserlist",level:3},{value:"ModioOptionalModChangeMap",id:"modiooptionalmodchangemap",level:3},{value:"ModioOptionalTransactionRecord",id:"modiooptionaltransactionrecord",level:3},{value:"ModioOptionalEntitlementConsumptionStatusList",id:"modiooptionalentitlementconsumptionstatuslist",level:3},{value:"ModioOptionalModID",id:"modiooptionalmodid",level:3},{value:"ModioAuthenticationParams",id:"modioauthenticationparams",level:3},{value:"Variables",id:"variables-19",level:5},{value:"ModioApiKey",id:"modioapikey",level:3},{value:"ModioGuid",id:"modioguid",level:3},{value:"ModioOptionalGuid",id:"modiooptionalguid",level:3},{value:"ModioEmailAddress",id:"modioemailaddress",level:3},{value:"ModioEmailAuthCode",id:"modioemailauthcode",level:3},{value:"ModioEntitlementParams",id:"modioentitlementparams",level:3},{value:"ModioMetricsSessionParams",id:"modiometricssessionparams",level:3},{value:"ModioCreateModFileParams",id:"modiocreatemodfileparams",level:3},{value:"Variables",id:"variables-20",level:5},{value:"ModioCreateModParams",id:"modiocreatemodparams",level:3},{value:"Variables",id:"variables-21",level:5},{value:"ModioEditModParams",id:"modioeditmodparams",level:3},{value:"ModioEntitlementWalletBalance",id:"modioentitlementwalletbalance",level:3},{value:"Variables",id:"variables-22",level:5},{value:"ModioOptionalEntitlementWalletBalance",id:"modiooptionalentitlementwalletbalance",level:3},{value:"ModioEntitlementConsumptionVirtualCurrencyDetails",id:"modioentitlementconsumptionvirtualcurrencydetails",level:3},{value:"Variables",id:"variables-23",level:5},{value:"EntitlementConsumptionStatus",id:"entitlementconsumptionstatus",level:3},{value:"Variables",id:"variables-24",level:5},{value:"ModioEntitlementConsumptionStatusList",id:"modioentitlementconsumptionstatuslist",level:3},{value:"Variables",id:"variables-25",level:5},{value:"ModioFilterParams",id:"modiofilterparams",level:3},{value:"ModioPresetFilterParams",id:"modiopresetfilterparams",level:3},{value:"Variables",id:"variables-26",level:5},{value:"ModioImageWrapper",id:"modioimagewrapper",level:3},{value:"Variables",id:"variables-27",level:5},{value:"ModioInitializeOptions",id:"modioinitializeoptions",level:3},{value:"Variables",id:"variables-28",level:5},{value:"ModioModChangeMap",id:"modiomodchangemap",level:3},{value:"Variables",id:"variables-29",level:5},{value:"ModioModCollectionEntry",id:"modiomodcollectionentry",level:3},{value:"ModioModCreationHandle",id:"modiomodcreationhandle",level:3},{value:"ModioModDependency",id:"modiomoddependency",level:3},{value:"Variables",id:"variables-30",level:5},{value:"ModioModDependencyList",id:"modiomoddependencylist",level:3},{value:"Variables",id:"variables-31",level:5},{value:"ModioModProgressInfo",id:"modiomodprogressinfo",level:3},{value:"Variables",id:"variables-32",level:5},{value:"ModioOptionalModProgressInfo",id:"modiooptionalmodprogressinfo",level:3},{value:"ModioReportParams",id:"modioreportparams",level:3},{value:"ModioLink",id:"modiolink",level:3},{value:"Variables",id:"variables-33",level:5},{value:"ModioTerms",id:"modioterms",level:3},{value:"Variables",id:"variables-34",level:5},{value:"ModioTransactionRecord",id:"modiotransactionrecord",level:3},{value:"Variables",id:"variables-35",level:5},{value:"ModioOptionalUser",id:"modiooptionaluser",level:3},{value:"ModioUserList",id:"modiouserlist",level:3},{value:"Variables",id:"variables-36",level:5},{value:"ModioValidationError",id:"modiovalidationerror",level:3},{value:"Variables",id:"variables-37",level:5},{value:"ModioCreateModFileMemoryParams",id:"modiocreatemodfilememoryparams",level:3},{value:"Variables",id:"variables-38",level:5},{value:"Functions",id:"functions",level:2},{value:"Set Session Identifier",id:"set-session-identifier",level:3},{value:"Parameters",id:"parameters-64",level:4},{value:"Returns",id:"returns-16",level:4},{value:"Set Session Id",id:"set-session-id",level:3},{value:"Parameters",id:"parameters-65",level:4},{value:"Set Portal",id:"set-portal",level:3},{value:"Parameters",id:"parameters-66",level:4},{value:"Returns",id:"returns-17",level:4},{value:"Set Game Id",id:"set-game-id",level:3},{value:"Parameters",id:"parameters-67",level:4},{value:"Returns",id:"returns-18",level:4},{value:"Set Game Environment",id:"set-game-environment",level:3},{value:"Parameters",id:"parameters-68",level:4},{value:"Returns",id:"returns-19",level:4},{value:"Set Extended Initialization Parameters",id:"set-extended-initialization-parameters",level:3},{value:"Parameters",id:"parameters-69",level:4},{value:"Returns",id:"returns-20",level:4},{value:"Set API Key",id:"set-api-key",level:3},{value:"Parameters",id:"parameters-70",level:4},{value:"Returns",id:"returns-21",level:4},{value:"ModioModID != ModioModID",id:"modiomodid--modiomodid",level:3},{value:"Parameters",id:"parameters-71",level:4},{value:"Make Metrics Session Params",id:"make-metrics-session-params",level:3},{value:"Parameters",id:"parameters-72",level:4},{value:"Make Initialize Options",id:"make-initialize-options",level:3},{value:"Parameters",id:"parameters-73",level:4},{value:"Make Guid",id:"make-guid",level:3},{value:"Parameters",id:"parameters-74",level:4},{value:"Make Game Id",id:"make-game-id",level:3},{value:"Parameters",id:"parameters-75",level:4},{value:"Make Entitlement Params",id:"make-entitlement-params",level:3},{value:"Parameters",id:"parameters-76",level:4},{value:"Make Auth Params",id:"make-auth-params",level:3},{value:"Parameters",id:"parameters-77",level:4},{value:"Returns",id:"returns-22",level:4},{value:"Make Api Key",id:"make-api-key",level:3},{value:"Parameters",id:"parameters-78",level:4},{value:"Get Raw Value from Mod ID",id:"get-raw-value-from-mod-id",level:3},{value:"Parameters",id:"parameters-79",level:4},{value:"Returns",id:"returns-23",level:4},{value:"ModioModID == ModioModID",id:"modiomodid--modiomodid-1",level:3},{value:"Parameters",id:"parameters-80",level:4},{value:"Set Version String",id:"set-version-string",level:3},{value:"Set Tags",id:"set-tags",level:3},{value:"Set Modfile Platforms",id:"set-modfile-platforms",level:3},{value:"Set Mod File Metadata Blob",id:"set-mod-file-metadata-blob",level:3},{value:"Set Metadata Blob",id:"set-metadata-blob",level:3},{value:"Set Mark as Active Release",id:"set-mark-as-active-release",level:3},{value:"Set Initial Visibility DEPRECATED",id:"set-initial-visibility-deprecated",level:3},{value:"Set Initial Visibility",id:"set-initial-visibility",level:3},{value:"Set Homepage URL",id:"set-homepage-url",level:3},{value:"Set Description",id:"set-description",level:3},{value:"Set Changelog String",id:"set-changelog-string",level:3},{value:"Get Localized Text for Enum by Name",id:"get-localized-text-for-enum-by-name",level:3},{value:"Parameters",id:"parameters-81",level:4},{value:"Returns",id:"returns-24",level:4},{value:"FileSizeToText (Unsigned64)",id:"filesizetotext-unsigned64",level:3},{value:"Parameters",id:"parameters-82",level:4},{value:"Returns",id:"returns-25",level:4},{value:"Get Localized Text from Default Table by Key",id:"get-localized-text-from-default-table-by-key",level:3},{value:"Parameters",id:"parameters-83",level:4},{value:"Returns",id:"returns-26",level:4},{value:"Reconstruct Error",id:"reconstruct-error",level:3},{value:"Parameters",id:"parameters-84",level:4},{value:"IsError",id:"iserror",level:3},{value:"Parameters",id:"parameters-85",level:4},{value:"Returns",id:"returns-27",level:4},{value:"Get Value",id:"get-value",level:3},{value:"Parameters",id:"parameters-86",level:4},{value:"Returns",id:"returns-28",level:4},{value:"Get Message",id:"get-message",level:3},{value:"Parameters",id:"parameters-87",level:4},{value:"Returns",id:"returns-29",level:4},{value:"Error Code Matches",id:"error-code-matches",level:3},{value:"Parameters",id:"parameters-88",level:4},{value:"Returns",id:"returns-30",level:4},{value:"List User Subscription Async",id:"list-user-subscription-async",level:3},{value:"Parameters",id:"parameters-89",level:4},{value:"Get Logo Thumbnail Size",id:"get-logo-thumbnail-size",level:3},{value:"Get Logo Full Size",id:"get-logo-full-size",level:3},{value:"Get Avatar Thumbnail Size",id:"get-avatar-thumbnail-size",level:3},{value:"To Filter Params",id:"to-filter-params",level:3},{value:"Load Async",id:"load-async",level:3},{value:"Get Texture",id:"get-texture",level:3},{value:"Get State",id:"get-state",level:3},{value:"Get Logo Size",id:"get-logo-size",level:3},{value:"Parameters",id:"parameters-90",level:4},{value:"Returns",id:"returns-31",level:4},{value:"Get Gallery Size",id:"get-gallery-size",level:3},{value:"Parameters",id:"parameters-91",level:4},{value:"Returns",id:"returns-32",level:4},{value:"Get Avatar Size",id:"get-avatar-size",level:3},{value:"Parameters",id:"parameters-92",level:4},{value:"Returns",id:"returns-33",level:4},{value:"Get Path",id:"get-path",level:3},{value:"Parameters",id:"parameters-93",level:4},{value:"Returns",id:"returns-34",level:4},{value:"Get Mod State",id:"get-mod-state",level:3},{value:"Parameters",id:"parameters-94",level:4},{value:"Returns",id:"returns-35",level:4},{value:"Get Mod Profile",id:"get-mod-profile",level:3},{value:"Parameters",id:"parameters-95",level:4},{value:"Returns",id:"returns-36",level:4},{value:"Get ID",id:"get-id",level:3},{value:"Parameters",id:"parameters-96",level:4},{value:"Returns",id:"returns-37",level:4},{value:"Get Total Progress",id:"get-total-progress",level:3},{value:"Parameters",id:"parameters-97",level:4},{value:"Returns",id:"returns-38",level:4},{value:"Get Current State",id:"get-current-state",level:3},{value:"Parameters",id:"parameters-98",level:4},{value:"Get Current Progress",id:"get-current-progress",level:3},{value:"Parameters",id:"parameters-99",level:4},{value:"Returns",id:"returns-39",level:4},{value:"Get Tags",id:"get-tags",level:3},{value:"Parameters",id:"parameters-100",level:4},{value:"Returns",id:"returns-40",level:4},{value:"Get Paged Result",id:"get-paged-result",level:3},{value:"Parameters",id:"parameters-101",level:4},{value:"Returns",id:"returns-41",level:4},{value:"Get Default Portal for Current Platform",id:"get-default-portal-for-current-platform",level:3},{value:"Parameters",id:"parameters-102",level:4},{value:"Returns",id:"returns-42",level:4},{value:"Get Default Auth Provider for Current Platform",id:"get-default-auth-provider-for-current-platform",level:3},{value:"Parameters",id:"parameters-103",level:4},{value:"Returns",id:"returns-43",level:4},{value:"Get Current Platform",id:"get-current-platform",level:3},{value:"Parameters",id:"parameters-104",level:4},{value:"Returns",id:"returns-44",level:4},{value:"Round Number String",id:"round-number-string",level:3},{value:"Parameters",id:"parameters-105",level:4},{value:"Get Percent (integer64/integer64)",id:"get-percent-integer64integer64",level:3},{value:"Parameters",id:"parameters-106",level:4},{value:"Returns",id:"returns-45",level:4},{value:"Is Valid Security Code Format",id:"is-valid-security-code-format",level:3},{value:"Parameters",id:"parameters-107",level:4},{value:"Returns",id:"returns-46",level:4},{value:"Is Valid Email Address Format",id:"is-valid-email-address-format",level:3},{value:"Parameters",id:"parameters-108",level:4},{value:"Returns",id:"returns-47",level:4},{value:"Get Time Span as String",id:"get-time-span-as-string",level:3},{value:"Parameters",id:"parameters-109",level:4},{value:"Get Shortened Number as String",id:"get-shortened-number-as-string",level:3},{value:"Parameters",id:"parameters-110",level:4},{value:"Get Project Initialize Options for Session Id",id:"get-project-initialize-options-for-session-id",level:3},{value:"Parameters",id:"parameters-111",level:4},{value:"Get Project Game Id",id:"get-project-game-id",level:3},{value:"Parameters",id:"parameters-112",level:4},{value:"Get Project Environment",id:"get-project-environment",level:3},{value:"Parameters",id:"parameters-113",level:4},{value:"Get Project Api Key",id:"get-project-api-key",level:3},{value:"Parameters",id:"parameters-114",level:4},{value:"Get Language Code String",id:"get-language-code-string",level:3},{value:"Parameters",id:"parameters-115",level:4},{value:"Returns",id:"returns-48",level:4},{value:"Get Language Code from String",id:"get-language-code-from-string",level:3},{value:"Parameters",id:"parameters-116",level:4},{value:"Returns",id:"returns-49",level:4},{value:"Get Desired File Size Unit",id:"get-desired-file-size-unit",level:3},{value:"Parameters",id:"parameters-117",level:4},{value:"Returns",id:"returns-50",level:4},{value:"Get Default Session Id Windows",id:"get-default-session-id-windows",level:3},{value:"Parameters",id:"parameters-118",level:4},{value:"Returns",id:"returns-51",level:4},{value:"ToString (Filesize)",id:"tostring-filesize",level:3},{value:"Parameters",id:"parameters-119",level:4},{value:"Returns",id:"returns-52",level:4},{value:"GetDefaultModInstallationDirectory",id:"getdefaultmodinstallationdirectory",level:3},{value:"Parameters",id:"parameters-120",level:4},{value:"Returns",id:"returns-53",level:4},{value:"ModioUnsigned64 - ModioUnsigned64",id:"modiounsigned64---modiounsigned64",level:3},{value:"Percentage Unsigned 64",id:"percentage-unsigned-64",level:3},{value:"ModioUnsigned64 != ModioUnsigned64",id:"modiounsigned64--modiounsigned64",level:3},{value:"Make from Components",id:"make-from-components",level:3},{value:"ModioUnsigned64 < ModioUnsigned64",id:"modiounsigned64--modiounsigned64-1",level:3},{value:"ModioUnsigned64 > 0",id:"modiounsigned64--0",level:3},{value:"ModioUnsigned64 > ModioUnsigned64",id:"modiounsigned64--modiounsigned64-2",level:3},{value:"ModioUnsigned64 == ModioUnsigned64",id:"modiounsigned64--modiounsigned64-3",level:3},{value:"ModioUnsigned64 / ModioUnsigned64",id:"modiounsigned64--modiounsigned64-4",level:3},{value:"ModioUnsigned64 / float",id:"modiounsigned64--float",level:3},{value:"ModioUnsigned64 / ModioUnsigned64 (truncate)",id:"modiounsigned64--modiounsigned64-truncate",level:3},{value:"Break to Components",id:"break-to-components",level:3},{value:"ModioUnsigned64 + ModioUnsigned64",id:"modiounsigned64--modiounsigned64-5",level:3},{value:"SubmitNewModFromMemoryAsync",id:"submitnewmodfrommemoryasync",level:3},{value:"Parameters",id:"parameters-121",level:4},{value:"SubmitNewModFileForModFromMemory",id:"submitnewmodfileformodfrommemory",level:3},{value:"Requirements",id:"requirements-41",level:4},{value:"Parameters",id:"parameters-122",level:4},{value:"LoadModFileToMemory",id:"loadmodfiletomemory",level:3},{value:"Requirements",id:"requirements-42",level:4},{value:"Parameters",id:"parameters-123",level:4},{value:"Enums",id:"enums",level:2},{value:"EModioModfilePlatform",id:"EModioModfilePlatform",level:3},{value:"EGameMaturityFlags",id:"EGameMaturityFlags",level:3},{value:"EGameMonetizationFlags",id:"EGameMonetizationFlags",level:3},{value:"EModioModServerSideStatus",id:"EModioModServerSideStatus",level:3},{value:"EModioVirusStatus",id:"EModioVirusStatus",level:3},{value:"EModioVirusScanStatus",id:"EModioVirusScanStatus",level:3},{value:"EModioObjectVisibilityFlags",id:"EModioObjectVisibilityFlags",level:3},{value:"EModioMaturityFlags",id:"EModioMaturityFlags",level:3},{value:"EModioModManagementEventType",id:"EModioModManagementEventType",level:3},{value:"EModioAuthenticationProvider",id:"EModioAuthenticationProvider",level:3},{value:"EModioEnvironment",id:"EModioEnvironment",level:3},{value:"EModioPortal",id:"EModioPortal",level:3},{value:"EModioPlatformName",id:"EModioPlatformName",level:3},{value:"EModioLogoSize",id:"EModioLogoSize",level:3},{value:"EModioAvatarSize",id:"EModioAvatarSize",level:3},{value:"EModioGallerySize",id:"EModioGallerySize",level:3},{value:"EModioLogLevel",id:"EModioLogLevel",level:3},{value:"EModioLanguage",id:"EModioLanguage",level:3},{value:"EModioModChangeType",id:"EModioModChangeType",level:3},{value:"EFileSizeUnit",id:"EFileSizeUnit",level:3},{value:"EModioEntitlementConsumptionState",id:"EModioEntitlementConsumptionState",level:3},{value:"EModioEntitlementType",id:"EModioEntitlementType",level:3},{value:"EModioErrorCondition",id:"EModioErrorCondition",level:3},{value:"EModioSortFieldType",id:"EModioSortFieldType",level:3},{value:"EModioSortDirection",id:"EModioSortDirection",level:3},{value:"EModioRevenueFilterType",id:"EModioRevenueFilterType",level:3},{value:"EModioImageState",id:"EModioImageState",level:3},{value:"EModioModState",id:"EModioModState",level:3},{value:"EModioModProgressState",id:"EModioModProgressState",level:3},{value:"EModioRating",id:"EModioRating",level:3},{value:"EModioReportType",id:"EModioReportType",level:3}];function h(e){const d={a:"a",admonition:"admonition",code:"code",del:"del",em:"em",h2:"h2",h3:"h3",h4:"h4",h5:"h5",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,r.R)(),...e.components},{RefTable:n}=d;return n||function(e,d){throw new Error("Expected "+(d?"component":"object")+" `"+e+"` to be defined: you likely forgot to import, pass, or provide it.")}("RefTable",!0),(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(d.h2,{id:"classes",children:"Classes"}),"\n",(0,i.jsx)(d.h3,{id:"modiocommontypeslibrary",children:"ModioCommonTypesLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiocreatemodlibrary",children:"ModioCreateModLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-1",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioerrorcodelibrary",children:"ModioErrorCodeLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-2",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioerrorconditionlibrary",children:"ModioErrorConditionLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-3",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioexamplelibrary",children:"ModioExampleLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-4",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioimagelibrary",children:"ModioImageLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-5",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodcollectionlibrary",children:"ModioModCollectionLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-6",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodprogressinfolibrary",children:"ModioModProgressInfoLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-7",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodtagoptionslibrary",children:"ModioModTagOptionsLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-8",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioplatformhelperslibrary",children:"ModioPlatformHelpersLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-9",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiopresetfilterparamslibrary",children:"ModioPresetFilterParamsLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-10",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiosdklibrary",children:"ModioSDKLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-11",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiosubmissionextensionlibrary",children:"ModioSubmissionExtensionLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-12",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.h4,{id:"submitmodchangesasync",children:"SubmitModChangesAsync"}),"\n",(0,i.jsx)(d.p,{children:"Edits the parameters of a mod, by updating any fields set in the Params object to match the passed-in values. Fields left empty on the Params object will not be updated. This method also accepts a Png binary file for uploading a new logo."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubmissionExtensionLibrary_K2_SubmitModChangesFromMemoryAsync.png",src:s(98688).A+"",width:"328",height:"244"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitModChangesFromMemoryAsync(FModioModID Mod, FModioEditModParams Params, TArray PngData, FOnGetModInfoDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubmissionExtensionLibrary"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Mod"})}),(0,i.jsx)(d.td,{children:"The ID of the mod you wish to edit"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Params"})}),(0,i.jsx)(d.td,{children:"Descriptor containing the fields that should be altered."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PngData"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"The callback invoked when the changes have been submitted, containing an optional updated ModInfo object if the edits were performed successfully"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:"GenericError::SDKNotInitialized"}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:"UserDataError::InvalidUser"}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:"NetworkError"}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:"InvalidArgsError"}),(0,i.jsx)(d.td,{children:"Some fields in Params did not pass validation"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiosubsystem",children:"ModioSubsystem"}),"\n",(0,i.jsxs)(d.p,{children:[(0,i.jsx)(d.code,{children:"ModioSubsystem"})," is a thin wrapper around the mod.io SDK, wrapping all the functions available in the SDK's public header ",(0,i.jsx)(d.code,{children:"modio/ModioSDK.h"}),". This subsystem also converts mod.io SDK types to unreal-friendly types and caches some expensive operations."]}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-13",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> EngineSubsystem-> DynamicSubsystem-> Subsystem-> Object"}),"\n",(0,i.jsx)(d.h4,{id:"set-log-level",children:"Set Log Level"}),"\n",(0,i.jsx)(d.p,{children:"Sets the global logging level."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_SetLogLevel.png",src:s(23059).A+"",width:"306",height:"182"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetLogLevel(EModioLogLevel UnrealLogLevel)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-1",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UnrealLogLevel"})}),(0,i.jsx)(d.td,{children:"Determines which messages to include in the log output. Messages with a log level below the specified value will not be displayed."})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"run-pending-handlers",children:"Run Pending Handlers"}),"\n",(0,i.jsx)(d.p,{children:"Runs any pending mod.io work on the calling thread and invokes any callbacks passed to asynchronous operations."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_RunPendingHandlers.png",src:s(51672).A+"",width:"227",height:"124"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void RunPendingHandlers()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-2",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"query-user-subscriptions",children:"Query User Subscriptions"}),"\n",(0,i.jsx)(d.p,{children:"Fetches the local view of the user's subscribed mods, including mods that are subscribed but not yet installed"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_QueryUserSubscriptions.png",src:s(49498).A+"",width:"230",height:"94"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"TMap QueryUserSubscriptions()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-3",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:[(0,i.jsx)(d.code,{children:"TMap"})," providing information about the subscribed mods"]}),"\n",(0,i.jsx)(d.h4,{id:"query-user-purchased-mods",children:"Query User Purchased Mods"}),"\n",(0,i.jsxs)(d.p,{children:["Returns the user's purchased mods. ",(0,i.jsx)(d.a,{href:"#fetchuserpurchasesasync",children:(0,i.jsx)(d.code,{children:"FetchUserPurchasesAsync"})})," must be called first to populate the cache."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_QueryUserPurchasedMods.png",src:s(85256).A+"",width:"247",height:"94"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"TMap QueryUserPurchasedMods()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-1",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-4",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-1",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["A ",(0,i.jsx)(d.code,{children:"TMap"})," of all purchases a user has made."]}),"\n",(0,i.jsx)(d.h4,{id:"query-user-installations",children:"Query User Installations"}),"\n",(0,i.jsx)(d.p,{children:"Fetches the subset of the user's subscribed mods that are installed and ready for loading"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_QueryUserInstallations.png",src:s(46391).A+"",width:"333",height:"126"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"TMap QueryUserInstallations(bool bIncludeOutdatedMods)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-5",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bIncludeOutdatedMods"})}),(0,i.jsx)(d.td,{children:"Include subscribed mods that are installed but have an updated version on the server that has not yet been installed"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-2",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:[(0,i.jsx)(d.code,{children:"TMap"})," providing information about the subscribed mods"]}),"\n",(0,i.jsx)(d.h4,{id:"query-system-installations",children:"Query System Installations"}),"\n",(0,i.jsx)(d.p,{children:"Fetches all mods installed on the system, including those installed by other users."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_QuerySystemInstallations.png",src:s(17253).A+"",width:"239",height:"94"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"TMap QuerySystemInstallations()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-6",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-3",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["A ",(0,i.jsx)(d.code,{children:"TMap"})," of all mods installed on the system, including those installed by other users."]}),"\n",(0,i.jsx)(d.h4,{id:"prioritize-transfer-for-mod",children:"Prioritize Transfer for Mod"}),"\n",(0,i.jsx)(d.p,{children:"Cancels or suspends the current mod update, installation, or upload, and begins processing a pending operation for the specified mod ID"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_PrioritizeTransferForMod.png",src:s(39846).A+"",width:"271",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode PrioritizeTransferForMod(FModioModID ModToPrioritize)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-2",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-7",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModToPrioritize"})}),(0,i.jsx)(d.td,{children:"The ID for the mod to prioritize"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-4",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"Error code indicating the status of the prioritization request. Will be empty if the prioritization was successful or if the mod was already being processed"}),"\n",(0,i.jsx)(d.h5,{id:"error-values-1",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid or not present in the list of pending operations"})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"kill-background-thread",children:"Kill Background Thread"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_KillBackgroundThread.png",src:s(25607).A+"",width:"227",height:"124"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void KillBackgroundThread()\n"})}),"\n",(0,i.jsx)(d.h4,{id:"verifyuserauthenticationasync",children:"VerifyUserAuthenticationAsync"}),"\n",(0,i.jsx)(d.p,{children:"Queries the server to verify the state of the currently authenticated user if there is one present"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_VerifyUserAuthenticationAsync.png",src:s(83587).A+"",width:"265",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_VerifyUserAuthenticationAsync(FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-3",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-8",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["Callback invoked with the results of the verification process. An empty ",(0,i.jsx)(d.code,{children:"ModioErrorCode"})," indicates successful verification i.e. the mod.io server was contactable and the user's authentication remains valid."]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-2",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"unsubscribefrommodasync",children:"UnsubscribeFromModAsync"}),"\n",(0,i.jsx)(d.p,{children:"Sends a request to the mod.io server to remove the specified mod from the user's list of subscriptions. If no other local users are subscribed to the specified mod this function will also mark the mod for uninstallation by the SDK."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync.png",src:s(65298).A+"",width:"248",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_UnsubscribeFromModAsync(FModioModID ModToUnsubscribeFrom, FOnErrorOnlyDelegate OnUnsubscribeComplete)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-4",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-9",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModToUnsubscribeFrom"})}),(0,i.jsx)(d.td,{children:"Mod ID of the mod requiring unsubscription."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"On Unsubscribe Complete"})}),(0,i.jsx)(d.td,{children:"Callback invoked when the unsubscription request is completed."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-3",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"unmuteuserasync",children:"UnmuteUserAsync"}),"\n",(0,i.jsx)(d.p,{children:"Unmute a user. This allows mod.io to display mods authored by the now unmuted user when performing searches."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_UnmuteUserAsync.png",src:s(2124).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_UnmuteUserAsync(FModioUserID UserID, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-5",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-10",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserID"})}),(0,i.jsx)(d.td,{children:"ID of the user to unmute"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of unmuting the user."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-4",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied user ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"subscribetomodasync",children:"SubscribeToModAsync"}),"\n",(0,i.jsx)(d.p,{children:"Sends a request to the mod.io server to add the specified mod to the user's list of subscriptions, and marks the mod for local installation by the SDK"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubscribeToModAsync.png",src:s(8274).A+"",width:"251",height:"218"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_SubscribeToModAsync(FModioModID ModToSubscribeTo, bool IncludeDependencies, FOnErrorOnlyDelegate OnSubscribeComplete)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-6",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-11",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModToSubscribeTo"})}),(0,i.jsx)(d.td,{children:"Mod ID of the mod requiring a subscription."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"IncludeDependencies"})}),(0,i.jsx)(d.td,{children:"Subscribe to all dependencies as well."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"On Subscribe Complete"})}),(0,i.jsx)(d.td,{children:"Callback invoked when the subscription request is completed."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-5",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"submitnewmodfileformod",children:"SubmitNewModFileForMod"}),"\n",(0,i.jsxs)(d.p,{children:["Queues the upload of a new modfile release for the specified mod using the submitted parameters. This function takes an ",(0,i.jsx)(d.code,{children:"ModioCreateModFileParams"})," object to specify the path to the root folder of the new modfile. The plugin will compress the folder's contents into a .zip archive and queue the result for upload. When the upload completes, a mod management event will be triggered. Note the plugin is also responsible for decompressing the archive upon its installation at a later point in time."]}),"\n",(0,i.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,i.jsx)(d.p,{children:"This function is part of an experimental feature and is subject to change."})}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubmitNewModFileForMod.png",src:s(95258).A+"",width:"237",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitNewModFileForMod(FModioModID Mod, FModioCreateModFileParams Params)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-7",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-12",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Mod"})}),(0,i.jsx)(d.td,{children:"The ID of the mod you are submitting a file for"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Params"})}),(0,i.jsx)(d.td,{children:"Information about the mod file being created, including the root path of the directory that will be archived"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-6",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"submitnewmodasync",children:"SubmitNewModAsync"}),"\n",(0,i.jsx)(d.p,{children:"Requests the creation of a new mod on the server with the specified parameters"}),"\n",(0,i.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,i.jsx)(d.p,{children:"This function is part of an experimental feature and is subject to change."})}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubmitNewModAsync.png",src:s(8127).A+"",width:"227",height:"214"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitNewModAsync(FModioModCreationHandle Handle, FModioCreateModParams Params, FOnSubmitNewModDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-8",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-13",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Handle"})}),(0,i.jsxs)(d.td,{children:["The ",(0,i.jsx)(d.code,{children:"ModioModCreationHandle"})," for this submission. Once this method invokes your callback indicating success, the ",(0,i.jsx)(d.code,{children:"ModioModCreationHandle"})," is invalid for the rest of the session. You should request a new one for the next submission attempt."]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Params"})}),(0,i.jsx)(d.td,{children:"Information about the new mod to create"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,i.jsx)(d.code,{children:"ModioModID"})," for the newly created mod"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-7",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsxs)(d.td,{children:["Some fields in ",(0,i.jsx)(d.code,{children:"Params"})," did not pass validation"]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"submitmodratingasync",children:"SubmitModRatingAsync"}),"\n",(0,i.jsx)(d.p,{children:"Submits a rating for a mod on behalf of the current user. Submit a neutral rating to effectively clear a rating already submitted by a user. Submitting other values will overwrite any existing rating submitted by this user."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubmitModRatingAsync.png",src:s(69256).A+"",width:"306",height:"242"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitModRatingAsync(FModioModID Mod, EModioRating Rating, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-9",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-14",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Mod"})}),(0,i.jsx)(d.td,{children:"The mod the user is rating"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Rating"})}),(0,i.jsx)(d.td,{children:"The rating the user wishes to submit"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of the rating submission"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-8",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"submitmodchangesasync-1",children:"SubmitModChangesAsync"}),"\n",(0,i.jsxs)(d.p,{children:["Edits the parameters of a mod by updating any fields set in the ",(0,i.jsx)(d.code,{children:"Params"})," object to match the passed-in values. Fields left empty on the ",(0,i.jsx)(d.code,{children:"Params"})," object will not be updated."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubmitModChangesAsync.png",src:s(70604).A+"",width:"233",height:"214"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitModChangesAsync(FModioModID Mod, FModioEditModParams Params, FOnGetModInfoDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-10",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-15",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Mod"})}),(0,i.jsx)(d.td,{children:"The ID of the mod you wish to edit"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Params"})}),(0,i.jsx)(d.td,{children:"Descriptor containing the fields that should be altered."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["The callback invoked when the changes have been submitted containing an optional updated ",(0,i.jsx)(d.code,{children:"ModioModInfo"})," object if the edits were performed successfully"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-9",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsxs)(d.td,{children:["Some fields in ",(0,i.jsx)(d.code,{children:"Params"})," did not pass validation"]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"shutdownasync",children:"ShutdownAsync"}),"\n",(0,i.jsxs)(d.p,{children:["Cancels any running internal operations, frees SDK resources, and invokes any pending callbacks with an ",(0,i.jsx)(d.code,{children:"OperationCanceled"})," error category. This function will NOT block while the deinitialization occurs."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ShutdownAsync.png",src:s(85619).A+"",width:"232",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_ShutdownAsync(FOnErrorOnlyDelegate OnShutdownComplete)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-16",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"On Shutdown Complete"})}),(0,i.jsxs)(d.td,{children:["Callback invoked when the plugin is shut down and calling ",(0,i.jsx)(d.a,{href:"#run-pending-handlers",children:(0,i.jsx)(d.code,{children:"RunPendingHandlers"})})," is no longer required"]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"setlanguage",children:"SetLanguage"}),"\n",(0,i.jsx)(d.p,{children:"Set language to get corresponding data from the server"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SetLanguage.png",src:s(40275).A+"",width:"306",height:"182"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_SetLanguage(EModioLanguage Locale)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-17",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Locale"})}),(0,i.jsx)(d.td,{children:"Language to set"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"requestemailauthcodeasync",children:"RequestEmailAuthCodeAsync"}),"\n",(0,i.jsx)(d.p,{children:"Begins email authentication for the current session by requesting a one-time code be sent to the specified email address."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_RequestEmailAuthCodeAsync.png",src:s(94989).A+"",width:"252",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_RequestEmailAuthCodeAsync(FModioEmailAddress EmailAddress, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-11",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-18",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EmailAddress"})}),(0,i.jsx)(d.td,{children:"The email address to send the code to"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code indicating the outcome of the request"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-10",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserAlreadyAuthenticatedError"})}),(0,i.jsxs)(d.td,{children:["Current user is already authenticated. De-authenticate the current user with ",(0,i.jsx)(d.a,{href:"#clearuserdataasync",children:(0,i.jsx)(d.code,{children:"ClearUserDataAsync"})}),", and re-initialize the SDK by calling ",(0,i.jsx)(d.a,{href:"#shutdownasync",children:(0,i.jsx)(d.code,{children:"ShutdownAsync"})})," then ",(0,i.jsx)(d.a,{href:"#initializeasync",children:(0,i.jsx)(d.code,{children:"InitializeAsync"})}),"."]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"reportcontentasync",children:"ReportContentAsync"}),"\n",(0,i.jsxs)(d.p,{children:["Sends a content report to mod.io. When using this function, please inform your users that if they provide their contact name or details in the ",(0,i.jsx)(d.code,{children:"Report"})," parameter, this data may be shared with the person responsible for the content being reported. For more information on what data in a report will be shared with whom, please see ",(0,i.jsx)(d.a,{href:"https://mod.io/report",children:"our website's report form"}),"."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ReportContentAsync.png",src:s(57110).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_ReportContentAsync(FModioReportParams Report, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-12",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-19",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Report"})}),(0,i.jsx)(d.td,{children:"Information about the content being reported and a description of the report."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code to indicate successful submission of the report."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-11",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"Required information in the report did not pass validation"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsxs)(d.td,{children:["The mod ID, game ID, or user ID supplied to ",(0,i.jsx)(d.code,{children:"Report"})," is invalid"]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"removefromtempmodset",children:"RemoveFromTempModSet"}),"\n",(0,i.jsxs)(d.p,{children:["Remove mods from a temp mod set. Every temp mod specified by ",(0,i.jsx)(d.code,{children:"ModIds"})," will be uninstalled unless the user is already subscribed."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_RemoveFromTempModSet.png",src:s(74399).A+"",width:"235",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_RemoveFromTempModSet(TArray ModIds)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-13",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"mod-management-enabled"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"temp-mod-set-initialized"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-20",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModIds"})}),(0,i.jsxs)(d.td,{children:["TArray of ",(0,i.jsx)(d.code,{children:"ModioModID"}),"s to remove as temp mods"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-5",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["Error code indicating the status of the ",(0,i.jsx)(d.code,{children:"TempModSet"}),". Will be empty if it was successful"]}),"\n",(0,i.jsx)(d.h5,{id:"error-values-12",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,i.jsx)(d.td,{children:"Mod management not enabled"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TempModSetNotInitialized"})}),(0,i.jsxs)(d.td,{children:[(0,i.jsx)(d.code,{children:"TempModSet"})," not initialized"]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"refreshuserentitlementsasync",children:"RefreshUserEntitlementsAsync"}),"\n",(0,i.jsx)(d.p,{children:"Requests mod.io refresh the available entitlements for the current user purchased through the portal and currently authenticated."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_RefreshUserEntitlementsAsync.png",src:s(13821).A+"",width:"265",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_RefreshUserEntitlementsAsync(FModioEntitlementParams Params, FOnRefreshUserEntitlementsDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-14",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-21",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Params"})}),(0,i.jsx)(d.td,{children:"Additional parameters."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the refresh operation."})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"queryuserprofile",children:"QueryUserProfile"}),"\n",(0,i.jsx)(d.p,{children:"Fetches the currently authenticated mod.io user profile if there is one"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_QueryUserProfile.png",src:s(70005).A+"",width:"227",height:"92"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioOptionalUser K2_QueryUserProfile()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-22",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-6",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:[(0,i.jsx)(d.code,{children:"ModioOptionalUser"})," object containing profile information"]}),"\n",(0,i.jsx)(d.h4,{id:"querytempmodset",children:"QueryTempModSet"}),"\n",(0,i.jsxs)(d.p,{children:["Query every system and temp mod in ",(0,i.jsx)(d.code,{children:"TempModSet"})]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_QueryTempModSet.png",src:s(63699).A+"",width:"227",height:"126"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"TMap K2_QueryTempModSet()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-23",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-7",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["TMap using ",(0,i.jsx)(d.code,{children:"ModioModID"})," as keys and ",(0,i.jsx)(d.code,{children:"ModioModCollectionEntry"})," objects providing information about mods in ",(0,i.jsx)(d.code,{children:"TempModSet"})]}),"\n",(0,i.jsx)(d.h4,{id:"querycurrentmodupdate",children:"QueryCurrentModUpdate"}),"\n",(0,i.jsx)(d.p,{children:"Provides progress information for a mod installation or update operation if one is currently in progress."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_QueryCurrentModUpdate.png",src:s(16111).A+"",width:"227",height:"92"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioOptionalModProgressInfo K2_QueryCurrentModUpdate()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-24",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-8",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:[(0,i.jsx)(d.code,{children:"ModioOptionalModProgressInfo"})," object containing information regarding the progress of the installation operation."]}),"\n",(0,i.jsx)(d.h4,{id:"purchasemodasync",children:"PurchaseModAsync"}),"\n",(0,i.jsx)(d.p,{children:"Purchases a mod for the current player"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_PurchaseModAsync.png",src:s(21296).A+"",width:"227",height:"214"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_PurchaseModAsync(FModioModID ModID, FModioUnsigned64 ExpectedPrice, FOnPurchaseModDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-15",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-25",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModID"})}),(0,i.jsx)(d.td,{children:"ID of the mod to purchase"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ExpectedPrice"})}),(0,i.jsxs)(d.td,{children:["The price the user is expected to pay for the mod, generally ",(0,i.jsx)(d.a,{href:"#modiomodinfo",children:(0,i.jsx)(d.code,{children:"ModioModInfo.Price"})}),". This ensures that there is consistency between the displayed price and the price in the backend. If there is a mismatch, the purchase will fail."]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback invoked with purchase information once the purchase is completed."})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"previewexternalupdatesasync",children:"PreviewExternalUpdatesAsync"}),"\n",(0,i.jsxs)(d.p,{children:["Retrieve a list of updates between the users local mod state, and the server-side state. This allows you to identify which mods will be modified by the next call to ",(0,i.jsx)(d.a,{href:"#fetchexternalupdatesasync",children:(0,i.jsx)(d.code,{children:"FetchExternalUpdatesAsync"})})," in order to perform any content management (such as unloading files) that might be required."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_PreviewExternalUpdatesAsync.png",src:s(79630).A+"",width:"261",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_PreviewExternalUpdatesAsync(FOnPreviewExternalUpdatesDelegate OnPreviewDone)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-26",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"On Preview Done"})}),(0,i.jsx)(d.td,{children:"Callback invoked when the external state has been retrieved. It contains a dictionary with ModID as keys and change maps as values. Empty when there are no differences between local and the mod.io API service"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"muteuserasync",children:"MuteUserAsync"}),"\n",(0,i.jsx)(d.p,{children:"Mute a user. This will prevent mod.io from returning mods authored by the muted user. when performing searches."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MuteUserAsync.png",src:s(87007).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_MuteUserAsync(FModioUserID UserID, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-16",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-27",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserID"})}),(0,i.jsx)(d.td,{children:"ID of the User to mute"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of muting the user."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-13",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied user ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"metricssessionstartasync",children:"MetricsSessionStartAsync"}),"\n",(0,i.jsx)(d.p,{children:"Start a metrics play session"}),"\n",(0,i.jsx)(d.admonition,{title:"Premium Feature",type:"info",children:(0,i.jsxs)(d.p,{children:["This function requires the ",(0,i.jsx)(d.strong,{children:"Metrics"})," premium feature. Please contact your mod.io representative for more details."]})}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MetricsSessionStartAsync.png",src:s(35372).A+"",width:"238",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_MetricsSessionStartAsync(FModioMetricsSessionParams Params, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-28",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Params"})}),(0,i.jsxs)(d.td,{children:[(0,i.jsx)(d.code,{children:"ModioMetricsSessionParams"})," struct containing information of what and how to start a metrics session"]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the session start operation"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-14",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RateLimited"})}),(0,i.jsx)(d.td,{children:"Too many frequent calls to the API. Wait some time and try again."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidUser"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionNotInitialized"})}),(0,i.jsx)(d.td,{children:"Metrics session has not yet been initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionIsActive"})}),(0,i.jsx)(d.td,{children:"Metrics session is currently active and running"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"BadParameter"})}),(0,i.jsx)(d.td,{children:"One or more values in the Metric Session Parameters are invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"metricssessionsendheartbeatonceasync",children:"MetricsSessionSendHeartbeatOnceAsync"}),"\n",(0,i.jsx)(d.p,{children:"Sends a single heartbeat to the mod.io server to indicate a session is still active"}),"\n",(0,i.jsx)(d.admonition,{title:"Premium Feature",type:"info",children:(0,i.jsxs)(d.p,{children:["This function requires the ",(0,i.jsx)(d.strong,{children:"Metrics"})," premium feature. Please contact your mod.io representative for more details."]})}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatOnceAsync.png",src:s(91175).A+"",width:"326",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_MetricsSessionSendHeartbeatOnceAsync(FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-29",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the session heartbeat operation"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-15",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidUser"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionNotInitialized"})}),(0,i.jsx)(d.td,{children:"Metrics session has not yet been initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionIsNotActive"})}),(0,i.jsxs)(d.td,{children:["Metrics session is not currently running. Call ",(0,i.jsx)(d.a,{href:"#metricssessionstartasync",children:(0,i.jsx)(d.code,{children:"MetricsSessionStartAsync"})})," before attempting to sending a heartbeat."]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"metricssessionsendheartbeatatintervalasync",children:"MetricsSessionSendHeartbeatAtIntervalAsync"}),"\n",(0,i.jsx)(d.p,{children:"Sends a constant heartbeat at a given interval to the mod.io server to indicate a session is still active"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync.png",src:s(92702).A+"",width:"354",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_MetricsSessionSendHeartbeatAtIntervalAsync(FModioUnsigned64 IntervalSeconds, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-30",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"IntervalSeconds"})}),(0,i.jsx)(d.td,{children:"The frequency in seconds to send a heartbeat to the mod.io server"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the session heartbeat operation"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-16",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidUser"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionNotInitialized"})}),(0,i.jsx)(d.td,{children:"Metrics session has not yet been initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionIsNotActive"})}),(0,i.jsxs)(d.td,{children:["Metrics session is not currently running. Call ",(0,i.jsx)(d.a,{href:"#metricssessionstartasync",children:(0,i.jsx)(d.code,{children:"MetricsSessionStartAsync"})})," before attempting to sending a heartbeat."]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"metricssessionendasync",children:"MetricsSessionEndAsync"}),"\n",(0,i.jsx)(d.p,{children:"Ends a metrics play session"}),"\n",(0,i.jsx)(d.admonition,{title:"Premium Feature",type:"info",children:(0,i.jsxs)(d.p,{children:["This function requires the ",(0,i.jsx)(d.strong,{children:"Metrics"})," premium feature. Please contact your mod.io representative for more details."]})}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MetricsSessionEndAsync.png",src:s(43781).A+"",width:"229",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_MetricsSessionEndAsync(FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-31",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the session end operation"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-17",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RateLimited"})}),(0,i.jsx)(d.td,{children:"Too many frequent calls to the API. Wait some time and try again."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidUser"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionNotInitialized"})}),(0,i.jsx)(d.td,{children:"Metrics session has not yet been initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionIsNotActive"})}),(0,i.jsxs)(d.td,{children:["Metrics session is not currently running. Call ",(0,i.jsx)(d.a,{href:"#metricssessionstartasync",children:(0,i.jsx)(d.code,{children:"MetricsSessionStartAsync"})})," before attempting to end a session."]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"listusergamesasync",children:"ListUserGamesAsync"}),"\n",(0,i.jsx)(d.p,{children:"Provides a list of games for the current user that match the parameters specified in the filter"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ListUserGamesAsync.png",src:s(34935).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_ListUserGamesAsync(FModioFilterParams Filter, FOnListUserGamesDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-17",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-32",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Filter"})}),(0,i.jsxs)(d.td,{children:[(0,i.jsx)(d.a,{href:"#modiofilterparams",children:(0,i.jsx)(d.code,{children:"ModioFilterParams"})})," object containing any filters that should be applied to the query"]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["Callback invoked with a status code and an optional ",(0,i.jsx)(d.code,{children:"ModioGameInfoList"})," providing game profiles"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-18",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"listusercreatedmodsasync",children:"ListUserCreatedModsAsync"}),"\n",(0,i.jsx)(d.p,{children:"Provides a list of mods that the user has submitted or is a team member of for the current game, applying the parameters specified in the filter."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ListUserCreatedModsAsync.png",src:s(25489).A+"",width:"245",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_ListUserCreatedModsAsync(FModioFilterParams Filter, FOnListUserCreatedModsDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-18",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-33",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Filter"})}),(0,i.jsx)(d.td,{children:"Filter to apply when listing the user's created mods."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback invoked when the call succeeds, or when an error occurs."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-19",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RateLimited"})}),(0,i.jsx)(d.td,{children:"Too many frequent calls to the API. Wait some time and try again."})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"listallmodsasync",children:"ListAllModsAsync"}),"\n",(0,i.jsx)(d.p,{children:"Provides a list of mods for the current game that match the parameters specified in the filter"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ListAllModsAsync.png",src:s(1655).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_ListAllModsAsync(FModioFilterParams Filter, FOnListAllModsDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-19",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-34",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Filter"})}),(0,i.jsxs)(d.td,{children:[(0,i.jsx)(d.a,{href:"#modiofilterparams",children:(0,i.jsx)(d.code,{children:"ModioFilterParams"})})," object containing any filters that should be applied to the query"]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["Callback invoked with a status code and an optional ",(0,i.jsx)(d.code,{children:"ModioModInfoList"})," providing mod profiles"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-20",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"inittempmodset",children:"InitTempModSet"}),"\n",(0,i.jsxs)(d.p,{children:["Install every temp mod specified by ",(0,i.jsx)(d.code,{children:"ModIds"})," if not already installed."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_InitTempModSet.png",src:s(59125).A+"",width:"227",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_InitTempModSet(TArray ModIds)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-20",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"mod-management-enabled"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-35",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModIds"})}),(0,i.jsxs)(d.td,{children:["TArray of ",(0,i.jsx)(d.code,{children:"ModioModID"}),"s to install as temp mods"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-9",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["Error code indicating the status of the ",(0,i.jsx)(d.code,{children:"TempModSet"}),". Will be empty if it was successful"]}),"\n",(0,i.jsx)(d.h5,{id:"error-values-21",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,i.jsx)(d.td,{children:"Mod management not enabled"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"initializeasync",children:"InitializeAsync"}),"\n",(0,i.jsx)(d.p,{children:"Initializes the SDK for the given user. Loads the state of mods installed on the system as well as the set of mods the specified user has installed on this device"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_InitializeAsync.png",src:s(1735).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_InitializeAsync(FModioInitializeOptions InitializeOptions, FOnErrorOnlyDelegate OnInitComplete)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-36",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InitializeOptions"})}),(0,i.jsx)(d.td,{children:"Parameters to the function packed as a struct where all members needs to be initialized for the call to succeed"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"On Init Complete"})}),(0,i.jsx)(d.td,{children:"Callback which will be invoked with the result of initialization"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-22",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FilesystemError"})}),(0,i.jsx)(d.td,{children:"Couldn't create the user data or common data folders"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ConfigurationError"})}),(0,i.jsxs)(d.td,{children:["InitializeOptions contains an invalid value - inspect ",(0,i.jsx)(d.code,{children:"ec.value()"})," to determine what was incorrect"]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKAlreadyInitialized"})}),(0,i.jsx)(d.td,{children:"SDK already initialized"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getuserwalletbalanceasync",children:"GetUserWalletBalanceAsync"}),"\n",(0,i.jsxs)(d.p,{children:["Gets the users current wallet balance. This will also create a wallet for a user if one does not exist. You should ensure this is called prior to calling ",(0,i.jsx)(d.a,{href:"#purchasemodasync",children:(0,i.jsx)(d.code,{children:"PurchaseModAsync"})})," purchase will fail."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync.png",src:s(43355).A+"",width:"248",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetUserWalletBalanceAsync(FOnGetUserWalletBalanceDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-21",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-37",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback invoked with the users wallet balance"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getusermediaasync-avatar",children:"GetUserMediaAsync (Avatar)"}),"\n",(0,i.jsx)(d.p,{children:"Downloads the avatar of the currently authenticated user. Will only perform a download if there is no local cache of the avatar or if that cached copy is out-of-date."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetUserMediaAvatarAsync.png",src:s(91533).A+"",width:"306",height:"212"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetUserMediaAvatarAsync(EModioAvatarSize AvatarSize, FOnGetMediaDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-22",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-38",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"AvatarSize"})}),(0,i.jsx)(d.td,{children:"Parameter specifying the size of avatar image to download"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code for the download and an optional path to the downloaded image"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-23",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getuserdelegationtokenasync",children:"GetUserDelegationTokenAsync"}),"\n",(0,i.jsx)(d.p,{children:"Get a user delegation token that can be used for S2S service calls"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetUserDelegationTokenAsync.png",src:s(99183).A+"",width:"263",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetUserDelegationTokenAsync(FOnGetUserDelegationTokenDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-23",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-39",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback invoked with purchase information once the purchase is completed."})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"gettermsofuseasync",children:"GetTermsOfUseAsync"}),"\n",(0,i.jsx)(d.p,{children:"This function retrieves the information required for a game to display the mod.io terms of use to a player who wishes to create a mod.io account"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetTermsOfUseAsync.png",src:s(93492).A+"",width:"227",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetTermsOfUseAsync(FOnGetTermsOfUseDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-24",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-40",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback invoked with the terms of use data once retrieved from the server"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-24",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getmutedusersasync",children:"GetMutedUsersAsync"}),"\n",(0,i.jsx)(d.p,{children:"List all the users that have been muted by the current user."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetMutedUsersAsync.png",src:s(79944).A+"",width:"227",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetMutedUsersAsync(FOnMuteUsersDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-25",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-41",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of the operation, and an optional containing a list of muted users if successful."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-25",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getmodtagoptionsasync",children:"GetModTagOptionsAsync"}),"\n",(0,i.jsxs)(d.p,{children:["Fetches the available tags used on mods for the current game. These tags can them be used in conjunction with the FilterParams passed to ",(0,i.jsx)(d.a,{href:"#listallmodsasync",children:(0,i.jsx)(d.code,{children:"ListAllModsAsync"})})," Will be cached when first received"]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModTagOptionsAsync.png",src:s(57363).A+"",width:"229",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModTagOptionsAsync(FOnGetModTagOptionsDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-26",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-42",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,i.jsx)(d.code,{children:"ModioModTagOptions"})," object containing the available tags"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-26",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getmodmediaasync-logo",children:"GetModMediaAsync (Logo)"}),"\n",(0,i.jsx)(d.p,{children:"Downloads the logo for the specified mod. Will use existing file if it is already present on disk"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModMediaLogoAsync.png",src:s(24398).A+"",width:"306",height:"242"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModMediaLogoAsync(FModioModID ModId, EModioLogoSize LogoSize, FOnGetMediaDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-27",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-43",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModId"})}),(0,i.jsx)(d.td,{children:"Mod ID for use in logo retrieval"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"LogoSize"})}),(0,i.jsx)(d.td,{children:"Parameter indicating the size of logo that's required"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code and an optional path object pointing to the location of the downloaded image"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-27",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"Specified mod media does not exist or was deleted"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InsufficientSpace"})}),(0,i.jsx)(d.td,{children:"Not enough space for the file"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getmodmediaasync-gallery-image",children:"GetModMediaAsync (Gallery Image)"}),"\n",(0,i.jsx)(d.p,{children:"Get a gallery image for the specified mod ID. If it already exists on disk the file will be reused unless it is outdated"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync.png",src:s(80586).A+"",width:"306",height:"277"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModMediaGalleryImageAsync(FModioModID ModId, EModioGallerySize GallerySize, int32 Index, FOnGetMediaDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-28",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-44",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModId"})}),(0,i.jsx)(d.td,{children:"The mod you want to retrieve an image for"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GallerySize"})}),(0,i.jsx)(d.td,{children:"Size of the image you want to retrieve"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Index"})}),(0,i.jsx)(d.td,{children:"The zero-based index of the image you want to retrieve"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback containing a status code and an Optional containing a path to the image file on disk"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-28",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"Specified mod media does not exist or was deleted"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InsufficientSpace"})}),(0,i.jsx)(d.td,{children:"Not enough space for the file"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getmodmediaasync-avatar",children:"GetModMediaAsync (Avatar)"}),"\n",(0,i.jsx)(d.p,{children:"Downloads the creator avatar for a specified mod. Will use existing file if it is already present on disk and not outdated"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModMediaAvatarAsync.png",src:s(59484).A+"",width:"306",height:"242"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModMediaAvatarAsync(FModioModID ModId, EModioAvatarSize AvatarSize, FOnGetMediaDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-29",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-45",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModId"})}),(0,i.jsx)(d.td,{children:"ID of the mod the creator avatar will be retrieved for"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"AvatarSize"})}),(0,i.jsx)(d.td,{children:"Parameter indicating the size of avatar image that's required"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code and an optional path object pointing to the location of the downloaded image"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-29",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"Specified mod media does not exist or was deleted"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InsufficientSpace"})}),(0,i.jsx)(d.td,{children:"Not enough space for the file"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getmodinfoasync",children:"GetModInfoAsync"}),"\n",(0,i.jsx)(d.p,{children:"Fetches detailed information about the specified mod, including description and file metadata for the most recent release"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModInfoAsync.png",src:s(21913).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModInfoAsync(FModioModID ModId, FOnGetModInfoDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-30",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-46",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModId"})}),(0,i.jsx)(d.td,{children:"Mod ID of the mod to fetch data"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,i.jsx)(d.code,{children:"ModioModInfo"})," object with the mod's extended information"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-30",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getmoddependenciesasync",children:"GetModDependenciesAsync"}),"\n",(0,i.jsx)(d.p,{children:"For a given Mod ID, fetches a list of any mods that the creator has marked as dependencies"}),"\n",(0,i.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,i.jsx)(d.p,{children:"This function is part of an experimental feature and is subject to change."})}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModDependenciesAsync.png",src:s(17784).A+"",width:"245",height:"218"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModDependenciesAsync(FModioModID ModID, bool Recursive, FOnGetModDependenciesDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-31",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-47",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModID"})}),(0,i.jsx)(d.td,{children:"The mod to retrieve dependencies for"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Recursive"})}),(0,i.jsx)(d.td,{children:"Fetches dependencies recursively up to a depth of 5"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,i.jsx)(d.code,{children:"ModioModTagOptions"})," object containing the available tags"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-31",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"getmodcreationhandle",children:"GetModCreationHandle"}),"\n",(0,i.jsxs)(d.p,{children:["Gets a new mod handle for use with ",(0,i.jsx)(d.a,{href:"#submitnewmodasync",children:(0,i.jsx)(d.code,{children:"SubmitNewModAsync"})}),"."]}),"\n",(0,i.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,i.jsx)(d.p,{children:"This function is part of an experimental feature and is subject to change."})}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModCreationHandle.png",src:s(68156).A+"",width:"227",height:"124"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioModCreationHandle K2_GetModCreationHandle()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-48",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"getlanguage",children:"GetLanguage"}),"\n",(0,i.jsx)(d.p,{children:"Get the currently applied language"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetLanguage.png",src:s(86039).A+"",width:"227",height:"124"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioLanguage K2_GetLanguage()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-49",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-10",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"Current language"}),"\n",(0,i.jsx)(d.h4,{id:"getgameinfoasync",children:"GetGameInfoAsync"}),"\n",(0,i.jsx)(d.p,{children:"Fetches detailed information about the specified game"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetGameInfoAsync.png",src:s(87367).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_GetGameInfoAsync(FModioGameID GameID, FOnGetGameInfoDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-32",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-50",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameID"})}),(0,i.jsx)(d.td,{children:"Game ID of the game data to fetch"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,i.jsx)(d.code,{children:"ModioGameInfo"})," object with the game's extended information"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-32",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"Specified game does not exist"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied game ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"forceuninstallmodasync",children:"ForceUninstallModAsync"}),"\n",(0,i.jsxs)(d.p,{children:["Forcibly uninstalls a mod from the system. This can be used when the host application requires additional space for other mods. The current user must not be subscribed to the mod to force uninstall. To remove a mod the current user is subscribed to, first use ",(0,i.jsx)(d.a,{href:"#unsubscribefrommodasync",children:(0,i.jsx)(d.code,{children:"UnsubscribeFromModAsync"})}),". If the mod does not uninstall (due to a different user on the same system remaining subscribed), ",(0,i.jsx)(d.code,{children:"ForceUninstallModAsync"})," can be called next."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ForceUninstallModAsync.png",src:s(30612).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_ForceUninstallModAsync(FModioModID ModToRemove, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-51",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModToRemove"})}),(0,i.jsx)(d.td,{children:"The mod ID to force uninstall."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback invoked indicating success or failure of the uninstallation."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-33",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:"ApiErrorRefSuccess"}),(0,i.jsx)(d.td,{children:"User is still subscribed to the specified mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"fetchuserpurchasesasync",children:"FetchUserPurchasesAsync"}),"\n",(0,i.jsxs)(d.p,{children:["Fetches the user's purchases. This populates a runtime cache of purchase information that can be accessed using ",(0,i.jsx)(d.a,{href:"#query-user-purchased-mods",children:(0,i.jsx)(d.code,{children:"QueryUserPurchasedMods"})}),"."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync.png",src:s(10208).A+"",width:"240",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_FetchUserPurchasesAsync(FOnFetchUserPurchasesDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-33",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-52",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback invoked once the call has been completed."})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"fetchexternalupdatesasync",children:"FetchExternalUpdatesAsync"}),"\n",(0,i.jsx)(d.p,{children:"Synchronises the local list of the current user's subscribed mods with the server. Any mods that have been externally subscribed will be automatically marked for installation, and mods that have been externally removed from the user's subscriptions may be uninstalled if no other local users have a current subscription."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync.png",src:s(38798).A+"",width:"247",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_FetchExternalUpdatesAsync(FOnErrorOnlyDelegate OnFetchDone)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-53",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"On Fetch Done"})}),(0,i.jsx)(d.td,{children:"Callback invoked when the external state has been retrieved and merged with the local data"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"enablemodmanagement",children:"EnableModManagement"}),"\n",(0,i.jsx)(d.p,{children:"Enables the automatic management of installed mods on the system based on the user's subscriptions."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_EnableModManagement.png",src:s(58799).A+"",width:"227",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_EnableModManagement(FOnModManagementDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-54",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsxs)(d.td,{children:["This callback handler will be invoked with a ",(0,i.jsx)(d.a,{href:"#modiomodmanagementevent",children:(0,i.jsx)(d.code,{children:"ModioModManagementEvent"})})," for each mod operation performed by the SDK"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-11",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["An error code indicating success or failure of enabling mod management. Note that this is independent of error codes for mod management events. Inspect the ",(0,i.jsx)(d.code,{children:"Callback"})," for information on each mod management event."]}),"\n",(0,i.jsx)(d.h5,{id:"error-values-34",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModManagementAlreadyEnabled"})}),(0,i.jsx)(d.td,{children:"Mod management was already enabled. The mod management callback has not been changed."})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"closetempmodset",children:"CloseTempModSet"}),"\n",(0,i.jsx)(d.p,{children:"Uninstall every temp mod unless the user is subscribed."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_CloseTempModSet.png",src:s(50203).A+"",width:"227",height:"124"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_CloseTempModSet()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-34",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"mod-management-enabled"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"temp-mod-set-initialized"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-55",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-12",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["Error code indicating the status of the ",(0,i.jsx)(d.code,{children:"TempModSet"}),". Will be empty if it was successful"]}),"\n",(0,i.jsx)(d.h5,{id:"error-values-35",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,i.jsx)(d.td,{children:"Mod management not enabled"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TempModSetNotInitialized"})}),(0,i.jsxs)(d.td,{children:[(0,i.jsx)(d.code,{children:"TempModSet"})," not initialized"]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"clearuserdataasync",children:"ClearUserDataAsync"}),"\n",(0,i.jsx)(d.p,{children:"De-authenticates the current mod.io user for the current session, and clears all user-specific data stored on the current device. Any subscribed mods that are installed but do not have other local users subscribed will be uninstalled"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ClearUserDataAsync.png",src:s(73239).A+"",width:"227",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_ClearUserDataAsync(FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-35",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-56",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code indicating the outcome of clearing the user data. Error codes returned by this function are informative only - it will always succeed."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-36",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"authenticateuserexternalasync",children:"AuthenticateUserExternalAsync"}),"\n",(0,i.jsx)(d.p,{children:"Uses platform-specific authentication to associate a mod.io user account with the current platform user"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_AuthenticateUserExternalAsync.png",src:s(69428).A+"",width:"306",height:"242"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_AuthenticateUserExternalAsync(FModioAuthenticationParams User, EModioAuthenticationProvider Provider, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-36",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-57",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"User"})}),(0,i.jsx)(d.td,{children:"Authentication payload data to submit to the provider"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Provider"})}),(0,i.jsx)(d.td,{children:"The provider to use to perform the authentication"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback invoked once the authentication request has been made"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-37",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ConfigurationError"})}),(0,i.jsx)(d.td,{children:"The SDK's configuration is not valid"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The arguments passed to the function have failed validation"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserTermsOfUseError"})}),(0,i.jsx)(d.td,{children:"The user has not yet accepted the mod.io Terms of Use"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserAlreadyAuthenticatedError"})}),(0,i.jsxs)(d.td,{children:["Current user is already authenticated. De-authenticate the current user with ",(0,i.jsx)(d.a,{href:"#clearuserdataasync",children:(0,i.jsx)(d.code,{children:"ClearUserDataAsync"})}),", and re-initialize the SDK by calling ",(0,i.jsx)(d.a,{href:"#shutdownasync",children:(0,i.jsx)(d.code,{children:"ShutdownAsync"})})," followed by ",(0,i.jsx)(d.a,{href:"#initializeasync",children:(0,i.jsx)(d.code,{children:"InitializeAsync"})}),"."]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"authenticateuseremailasync",children:"AuthenticateUserEmailAsync"}),"\n",(0,i.jsx)(d.p,{children:"Completes email authentication for the current session by submitting the one-time code sent to the user's email address"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_AuthenticateUserEmailAsync.png",src:s(36043).A+"",width:"251",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_AuthenticateUserEmailAsync(FModioEmailAuthCode AuthenticationCode, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-37",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-58",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"AuthenticationCode"})}),(0,i.jsx)(d.td,{children:"User's authentication code"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code indicating if authentication was successful or not"})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-38",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserAlreadyAuthenticatedError"})}),(0,i.jsxs)(d.td,{children:["Current user is already authenticated. De-authenticate the current user with ",(0,i.jsx)(d.a,{href:"#clearuserdataasync",children:(0,i.jsx)(d.code,{children:"ClearUserDataAsync"})}),", and re-initialize the SDK by calling ",(0,i.jsx)(d.a,{href:"#shutdownasync",children:(0,i.jsx)(d.code,{children:"ShutdownAsync"})})," then ",(0,i.jsx)(d.a,{href:"#initializeasync",children:(0,i.jsx)(d.code,{children:"InitializeAsync"})}),"."]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"archivemodasync",children:"ArchiveModAsync"}),"\n",(0,i.jsx)(d.p,{children:"Archives a mod. This mod will no longer be able to be viewed or retrieved via the SDK, but it will still exist should you choose to restore it at a later date. Archiving is restricted to team managers and administrators only. Note that restoration and permanent deletion of a mod is possible only via web interface."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ArchiveModAsync.png",src:s(30771).A+"",width:"227",height:"184"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_ArchiveModAsync(FModioModID Mod, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-38",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-59",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Mod"})}),(0,i.jsx)(d.td,{children:"The mod to be archived."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of archiving the mod."})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"error-values-39",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InsufficientPermissions"})}),(0,i.jsx)(d.td,{children:"The authenticated user does not have permission to archive this mod. This action is restricted to team managers and administrators only."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"addtotempmodset",children:"AddToTempModSet"}),"\n",(0,i.jsxs)(d.p,{children:["Add mods to a Temp Mod Set. Every temp mod specified by ",(0,i.jsx)(d.code,{children:"ModIds"})," will be installed if not already installed."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_AddToTempModSet.png",src:s(9981).A+"",width:"227",height:"154"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_AddToTempModSet(TArray ModIds)\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-39",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"mod-management-enabled"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"temp-mod-set-initialized"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-60",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModIds"})}),(0,i.jsxs)(d.td,{children:["TArray of ",(0,i.jsx)(d.code,{children:"ModioModID"}),"s to install as temp mods"]})]})]})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-13",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["Error code indicating the status of the ",(0,i.jsx)(d.code,{children:"TempModSet"}),". Will be empty if it was successful"]}),"\n",(0,i.jsx)(d.h5,{id:"error-values-40",children:"Error Values"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"SDK not initialized"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,i.jsx)(d.td,{children:"Mod management not enabled"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TempModSetNotInitialized"})}),(0,i.jsxs)(d.td,{children:[(0,i.jsx)(d.code,{children:"TempModSet"})," not initialized"]})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"is-using-background-thread",children:"Is Using Background Thread"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_IsUsingBackgroundThread.png",src:s(27331).A+"",width:"243",height:"124"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool IsUsingBackgroundThread()\n"})}),"\n",(0,i.jsx)(d.h4,{id:"is-mod-management-busy",children:"Is Mod Management Busy"}),"\n",(0,i.jsx)(d.p,{children:"Checks if the automatic management process is currently installing or removing mods"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_IsModManagementBusy.png",src:s(40309).A+"",width:"231",height:"124"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool IsModManagementBusy()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-61",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-14",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"True if automatic management is currently performing an operation"}),"\n",(0,i.jsx)(d.h4,{id:"get-last-validation-error",children:"Get Last Validation Error"}),"\n",(0,i.jsx)(d.p,{children:"If the last request to the mod.io servers returned a validation failure, this function returns extended information describing the fields that failed validation."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_GetLastValidationError.png",src:s(33590).A+"",width:"227",height:"124"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"TArray GetLastValidationError()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"requirements-40",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,i.jsx)(d.h5,{id:"parameters-62",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.h5,{id:"returns-15",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["Collection of ",(0,i.jsx)(d.code,{children:"ModioValidationError"})," objects, or empty collection if there were no validation failures"]}),"\n",(0,i.jsx)(d.h4,{id:"disable-mod-management",children:"Disable Mod Management"}),"\n",(0,i.jsx)(d.p,{children:"Disables automatic installation or uninstallation of mods based on the user's subscriptions. Allows currently processing installation to complete. Will cancel any pending operations when called."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_DisableModManagement.png",src:s(63754).A+"",width:"231",height:"124"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void DisableModManagement()\n"})}),"\n",(0,i.jsx)(d.h5,{id:"parameters-63",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiouienumlocalizationlibrary",children:"ModioUIEnumLocalizationLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-14",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiouilocalizationlibrary",children:"ModioUILocalizationLibrary"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-15",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64library",children:"ModioUnsigned64Library"}),"\n",(0,i.jsx)(d.h5,{id:"inheritance-hierarchy-16",children:"Inheritance Hierarchy"}),"\n",(0,i.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h2,{id:"structs",children:"Structs"}),"\n",(0,i.jsx)(d.h3,{id:"modiogameinfolist",children:"ModioGameInfoList"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiogameinfo",children:"ModioGameInfo"}),"\n",(0,i.jsx)(d.h5,{id:"variables",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioGameID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameID"})}),(0,i.jsx)(d.td,{children:"Unique Game ID"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateAdded"})}),(0,i.jsx)(d.td,{children:"Unix timestamp of date the game was registered"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateUpdated"})}),(0,i.jsx)(d.td,{children:"Unix timestamp of date the game was updated"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateLive"})}),(0,i.jsx)(d.td,{children:"Unix timestamp of date the game was set live"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UgcName"})}),(0,i.jsx)(d.td,{children:"Word used to describe user-generated content (mods, items, add-ons etc)"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioIcon"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Icon"})}),(0,i.jsx)(d.td,{children:"Contains media URLs to the icon for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioLogo"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Logo"})}),(0,i.jsx)(d.td,{children:"Contains media URLs to the logo for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioHeaderImage"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"HeaderImage"})}),(0,i.jsx)(d.td,{children:"Contains media URLs to the preview header image for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Name"})}),(0,i.jsx)(d.td,{children:"Name of the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Summary"})}),(0,i.jsx)(d.td,{children:"Summary of the game's mod support"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Instructions"})}),(0,i.jsx)(d.td,{children:"A guide about creating and uploading mods for this game to mod.io"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InstructionsUrl"})}),(0,i.jsx)(d.td,{children:"Link to a mod.io guide, modding wiki, or a page where modders can learn how to make and submit mods to this game's profile"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileUrl"})}),(0,i.jsx)(d.td,{children:"URL to the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioTheme"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Theme"})}),(0,i.jsx)(d.td,{children:"Theme color values for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioGameStats"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Stats"})}),(0,i.jsx)(d.td,{children:"Numerous aggregate stats for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"OtherUrls"})}),(0,i.jsx)(d.td,{children:"Creator defined URLs to share"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Platforms"})}),(0,i.jsx)(d.td,{children:"Platforms that are supported by this title"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bAllowNegativeRatings"})}),(0,i.jsx)(d.td,{children:"Whether or not the game allows negative ratings"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EGameMonetizationFlags"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameMonetizationOptions"})}),(0,i.jsx)(d.td,{children:"Monetization options for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EGameMaturityFlags"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameMaturityOptions"})}),(0,i.jsx)(d.td,{children:"Maturity options for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"VirtualTokenName"})}),(0,i.jsx)(d.td,{children:"Name of the Virtual Tokens for this game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PlatformSupport"})}),(0,i.jsx)(d.td,{children:"Platforms that are supported by this title"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TagOptions"})}),(0,i.jsx)(d.td,{children:"Tags available for this game"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodtaginfo",children:"ModioModTagInfo"}),"\n",(0,i.jsx)(d.h5,{id:"variables-1",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TagGroupName"})}),(0,i.jsx)(d.td,{children:"Raw unlocalized tag group name"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TagGroupValues"})}),(0,i.jsx)(d.td,{children:"Valid raw unlocalized tag values this group contains"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TMap"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TagGroupNameLocalizationData"})}),(0,i.jsx)(d.td,{children:"Culture code -> localized tag category name mapping for all configured languages"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TagGroupValueLocalizationData"})}),(0,i.jsx)(d.td,{children:"Localization data for this tag category's values in all configured languages"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bAllowMultipleSelection"})}),(0,i.jsx)(d.td,{children:"True if multiple tags from the group can be used simultaneously"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bHidden"})}),(0,i.jsx)(d.td,{children:"True if only visible by admins"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bLocked"})}),(0,i.jsx)(d.td,{children:"TrueTrue if only editable by admins"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiolocalizedtagcategory",children:"ModioLocalizedTagCategory"}),"\n",(0,i.jsx)(d.p,{children:"Struct containing pre-localized display strings for a tag group"}),"\n",(0,i.jsx)(d.h5,{id:"variables-2",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GroupName"})}),(0,i.jsx)(d.td,{children:"Localized display string for this tag category's name"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Tags"})}),(0,i.jsx)(d.td,{children:"Localized display strings for all valid values in this tag category"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodtaglocalizationdata",children:"ModioModTagLocalizationData"}),"\n",(0,i.jsx)(d.p,{children:"Localization data for an individual tag value"}),"\n",(0,i.jsx)(d.h5,{id:"variables-3",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Tag"})}),(0,i.jsx)(d.td,{children:"The original raw unlocalized tag value used by the REST API"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TMap"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Translations"})}),(0,i.jsx)(d.td,{children:"Culture code -> Localized tag value string mapping for all configured languages."})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiogameplatform",children:"ModioGamePlatform"}),"\n",(0,i.jsx)(d.h5,{id:"variables-4",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioModfilePlatform"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Platform"})}),(0,i.jsx)(d.td,{children:"A platform supported by a title"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Locked"})}),(0,i.jsx)(d.td,{children:"Whether ot not this platform is locked from having files submitted to it by players"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Moderated"})}),(0,i.jsx)(d.td,{children:"Whether or not this platform's file submissions are moderated or not"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiootherurl",children:"ModioOtherUrl"}),"\n",(0,i.jsx)(d.h5,{id:"variables-5",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Label"})}),(0,i.jsx)(d.td,{children:"Label of the link you are sharing"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Url"})}),(0,i.jsx)(d.td,{children:"The URL to be associated with the label"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiogamestats",children:"ModioGameStats"}),"\n",(0,i.jsx)(d.p,{children:"Numerous aggregate stats for the game"}),"\n",(0,i.jsx)(d.h5,{id:"variables-6",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioGameID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameID"})}),(0,i.jsx)(d.td,{children:"Unique game id"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModCountTotal"})}),(0,i.jsx)(d.td,{children:"Available mod count for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModDownloadsToday"})}),(0,i.jsx)(d.td,{children:"Mods downloaded today for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModDownloadsTotal"})}),(0,i.jsx)(d.td,{children:"Total mods downloaded for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModDownloadsDailyAverage"})}),(0,i.jsx)(d.td,{children:"Average mods downloaded on a daily basis"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModSubscribersTotal"})}),(0,i.jsx)(d.td,{children:"Number of total users who have subscribed to the mods for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateExpires"})}),(0,i.jsx)(d.td,{children:"Unix timestamp until this game's statistics are considered stale"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiogameid",children:"ModioGameID"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiotheme",children:"ModioTheme"}),"\n",(0,i.jsx)(d.h5,{id:"variables-7",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Primary"})}),(0,i.jsx)(d.td,{children:"The primary hex color code"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Dark"})}),(0,i.jsx)(d.td,{children:"The dark hex color code"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Light"})}),(0,i.jsx)(d.td,{children:"The light hex color code"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Success"})}),(0,i.jsx)(d.td,{children:"The success hex color code"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Warning"})}),(0,i.jsx)(d.td,{children:"The warning hex color code"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Danger"})}),(0,i.jsx)(d.td,{children:"The danger hex color code"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioheaderimage",children:"ModioHeaderImage"}),"\n",(0,i.jsx)(d.h5,{id:"variables-8",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Filename"})}),(0,i.jsx)(d.td,{children:"Header image filename including extension"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Original"})}),(0,i.jsx)(d.td,{children:"URL to the full-sized header image"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiologo",children:"ModioLogo"}),"\n",(0,i.jsx)(d.h5,{id:"variables-9",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Filename"})}),(0,i.jsx)(d.td,{children:"Logo filename including extension."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Original"})}),(0,i.jsx)(d.td,{children:"URL to the full - sized logo."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb320x180"})}),(0,i.jsx)(d.td,{children:"URL to the small logo thumbnail."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb640x360"})}),(0,i.jsx)(d.td,{children:"URL to the medium logo thumbnail."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb1280x720"})}),(0,i.jsx)(d.td,{children:"URL to the large logo thumbnail."})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioicon",children:"ModioIcon"}),"\n",(0,i.jsx)(d.h5,{id:"variables-10",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Filename"})}),(0,i.jsx)(d.td,{children:"Icon filename including extension."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Original"})}),(0,i.jsx)(d.td,{children:"URL to the full-sized icon."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb64x64"})}),(0,i.jsx)(d.td,{children:"URL to the small icon thumbnail."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb128x128"})}),(0,i.jsx)(d.td,{children:"URL to the medium icon thumbnail."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb256x256"})}),(0,i.jsx)(d.td,{children:"URL to the large icon thumbnail."})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiopagedresult",children:"ModioPagedResult"}),"\n",(0,i.jsx)(d.h5,{id:"variables-11",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int32"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PageIndex"})}),(0,i.jsx)(d.td,{children:"The index of this page"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int32"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PageSize"})}),(0,i.jsx)(d.td,{children:"The amount of results allowed to be displayed within a page"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int32"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PageCount"})}),(0,i.jsx)(d.td,{children:"The total amount of pages"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int32"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TotalResultCount"})}),(0,i.jsx)(d.td,{children:"The total amount of results"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int32"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ResultCount"})}),(0,i.jsx)(d.td,{children:"The amount of results for this page"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodinfolist",children:"ModioModInfoList"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodinfo",children:"ModioModInfo"}),"\n",(0,i.jsx)(d.h5,{id:"variables-12",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioModID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModId"})}),(0,i.jsx)(d.td,{children:"Unique Mod ID"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileName"})}),(0,i.jsx)(d.td,{children:"Name of the mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileSummary"})}),(0,i.jsx)(d.td,{children:"Summary of the mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileDescription"})}),(0,i.jsx)(d.td,{children:"Detailed description in HTML format"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileDescriptionPlaintext"})}),(0,i.jsx)(d.td,{children:"Detailed description in plaintext"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileURL"})}),(0,i.jsx)(d.td,{children:"URL to the mod profile"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioUser"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileSubmittedBy"})}),(0,i.jsx)(d.td,{children:"Information on the user who submitted the mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileDateAdded"})}),(0,i.jsx)(d.td,{children:"Unix timestamp of the date the mod was registered"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileDateUpdated"})}),(0,i.jsx)(d.td,{children:"Unix timestamp of the date the mod was updated"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileDateLive"})}),(0,i.jsx)(d.td,{children:"Unix timestamp of the date the mod was marked live"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioMaturityFlags"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileMaturityOption"})}),(0,i.jsx)(d.td,{children:"Flags for maturity options. Maturity options are flagged by the mod developer. This is only relevant if the parent game allows mods to be labeled as mature."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.del,{children:(0,i.jsx)(d.code,{children:"bVisible_DEPRECATED"})})}),(0,i.jsxs)(d.td,{children:["Deprecated as of 2023.6 release. Please use ",(0,i.jsx)(d.code,{children:"EModioObjectVisibilityFlags Visibility"})," instead."]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioObjectVisibilityFlags"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Visibility"})}),(0,i.jsx)(d.td,{children:"Enum parameter to signal the backend if the mod to upload would be publicly visible. Default value is Public"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Dependencies"})}),(0,i.jsx)(d.td,{children:"If this mod has any dependencies"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MetadataBlob"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioFileMetadata"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FileInfo"})}),(0,i.jsx)(d.td,{children:"Information about the mod's most recent public release"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MetadataKvp"})}),(0,i.jsx)(d.td,{children:"Arbitrary key-value metadata set for this mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Tags"})}),(0,i.jsx)(d.td,{children:"Tags this mod has set"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int32"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NumGalleryImages"})}),(0,i.jsx)(d.td,{children:"Number of images in the mod's media gallery"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioYoutubeURLList"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"YoutubeURLs"})}),(0,i.jsx)(d.td,{children:"List of youtube links provided by the creator of the mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioSketchfabURLList"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SketchfabURLs"})}),(0,i.jsx)(d.td,{children:"List of sketchfab links provided by the creator of the mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioModStats"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Stats"})}),(0,i.jsx)(d.td,{children:"Stats and rating information for the mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioModServerSideStatus"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModStatus"})}),(0,i.jsx)(d.td,{children:"Status of the mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioUnsigned64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Price"})}),(0,i.jsx)(d.td,{children:"Price of this mod"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64",children:"ModioUnsigned64"}),"\n",(0,i.jsx)(d.p,{children:"Trivial Blueprint-compatible wrapper around an unsigned 64-bit integer"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodstats",children:"ModioModStats"}),"\n",(0,i.jsx)(d.p,{children:"Contains download stats and ratings for a mod"}),"\n",(0,i.jsx)(d.h5,{id:"variables-13",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PopularityRankPosition"})}),(0,i.jsx)(d.td,{children:"Current rank of the mod."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PopularityRankTotalMods"})}),(0,i.jsx)(d.td,{children:"Number of ranking spots the current rank is measured against."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DownloadsTotal"})}),(0,i.jsx)(d.td,{children:"Number of total mod downloads."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SubscribersTotal"})}),(0,i.jsx)(d.td,{children:"Number of total users who have subscribed to the mod."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RatingTotal"})}),(0,i.jsx)(d.td,{children:"Number of times this mod has been rated."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RatingPositive"})}),(0,i.jsx)(d.td,{children:"Number of positive ratings."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RatingNegative"})}),(0,i.jsx)(d.td,{children:"Number of negative ratings."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RatingPercentagePositive"})}),(0,i.jsx)(d.td,{children:"Number of positive ratings, divided by the total ratings to determine it\u2019s percentage score."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"float"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RatingWeightedAggregate"})}),(0,i.jsxs)(d.td,{children:["Overall rating of this item calculated using the ",(0,i.jsx)(d.a,{href:"https://www.evanmiller.org/how-not-to-sort-by-average-Ratinghtml",children:"Wilson score confidence interval"}),". This column is good to sort on, as it will order items based on number of ratings and will place items with many positive ratings above those with a higher score but fewer ratings. We actually get a double back from the server, but it's converted to a float for blueprint support"]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RatingDisplayText"})}),(0,i.jsx)(d.td,{children:"Textual representation of the rating in format: Overwhelmingly Positive -> Very Positive -> Positive -> Mostly Positive -> Mixed -> Negative -> Mostly Negative -> Very Negative -> Overwhelmingly Negative -> Unrated"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiosketchfaburllist",children:"ModioSketchfabURLList"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioyoutubeurllist",children:"ModioYoutubeURLList"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodtag",children:"ModioModTag"}),"\n",(0,i.jsx)(d.h5,{id:"variables-14",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Tag"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FText"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TagLocalized"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiometadata",children:"ModioMetadata"}),"\n",(0,i.jsx)(d.h5,{id:"variables-15",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Key"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Value"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiofilemetadata",children:"ModioFileMetadata"}),"\n",(0,i.jsx)(d.p,{children:"Metadata for a release archive for a mod"}),"\n",(0,i.jsx)(d.h5,{id:"variables-16",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioFileMetadataID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MetadataId"})}),(0,i.jsx)(d.td,{children:"Unique modfile id."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioModID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModId"})}),(0,i.jsx)(d.td,{children:"Unique mod id."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateAdded"})}),(0,i.jsx)(d.td,{children:"Unix timestamp of date file was added."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioVirusScanStatus"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"CurrentVirusScanStatus"})}),(0,i.jsx)(d.td,{children:"Current virus scan status of the file. For newly added files that have yet to be scanned this field will change frequently until a scan is complete"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioVirusStatus"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"CurrentVirusStatus"})}),(0,i.jsx)(d.td,{children:"Was a virus detected?"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Filesize"})}),(0,i.jsx)(d.td,{children:"Size of the file in bytes."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FilesizeUncompressed"})}),(0,i.jsx)(d.td,{children:"Total size of all files in the mod after installation."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Filename"})}),(0,i.jsx)(d.td,{children:"Filename including extension."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Version"})}),(0,i.jsx)(d.td,{children:"Release version this file represents."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Changelog"})}),(0,i.jsx)(d.td,{children:"Changelog for the file."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MetadataBlob"})}),(0,i.jsx)(d.td,{children:"Metadata stored by the game developer for this file."})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodid",children:"ModioModID"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiofilemetadataid",children:"ModioFileMetadataID"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiouser",children:"ModioUser"}),"\n",(0,i.jsx)(d.h5,{id:"variables-17",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioUserID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserId"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Username"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateOnline"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ProfileUrl"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DisplayNamePortal"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiouserid",children:"ModioUserID"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodtagoptions",children:"ModioModTagOptions"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioerrorcode",children:"ModioErrorCode"}),"\n",(0,i.jsxs)(d.p,{children:["Wrapper around ",(0,i.jsx)(d.code,{children:"Modio::ErrorCode"})]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalgameinfo",children:"ModioOptionalGameInfo"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalimage",children:"ModioOptionalImage"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalmoddependencylist",children:"ModioOptionalModDependencyList"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalmodinfo",children:"ModioOptionalModInfo"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalmodtagoptions",children:"ModioOptionalModTagOptions"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalterms",children:"ModioOptionalTerms"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionaluint64",children:"ModioOptionalUInt64"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalmodinfolist",children:"ModioOptionalModInfoList"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalgameinfolist",children:"ModioOptionalGameInfoList"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodmanagementevent",children:"ModioModManagementEvent"}),"\n",(0,i.jsx)(d.p,{children:"Simple struct representing the outcome of a mod management operation"}),"\n",(0,i.jsx)(d.h5,{id:"variables-18",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioModID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ID"})}),(0,i.jsx)(d.td,{children:"ID for the mod that the event occurred on"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioModManagementEventType"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Event"})}),(0,i.jsx)(d.td,{children:"What type of event occurred"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioErrorCode"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Status"})}),(0,i.jsx)(d.td,{children:"Empty if operation completed successfully, truthy/contains error code if operation failed"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionaluserlist",children:"ModioOptionalUserList"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalmodchangemap",children:"ModioOptionalModChangeMap"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionaltransactionrecord",children:"ModioOptionalTransactionRecord"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalentitlementconsumptionstatuslist",children:"ModioOptionalEntitlementConsumptionStatusList"}),"\n",(0,i.jsx)(d.p,{children:"Optional class representing a list of entitlement consumption statuses that may be a page from a larger set of results"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalmodid",children:"ModioOptionalModID"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioauthenticationparams",children:"ModioAuthenticationParams"}),"\n",(0,i.jsx)(d.p,{children:"Simple struct to encapsulate data passed to external authentication systems"}),"\n",(0,i.jsx)(d.h5,{id:"variables-19",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"AuthToken"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserEmail"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bUserHasAcceptedTerms"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TMap"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ExtendedParameters"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioapikey",children:"ModioApiKey"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioguid",children:"ModioGuid"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalguid",children:"ModioOptionalGuid"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioemailaddress",children:"ModioEmailAddress"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioemailauthcode",children:"ModioEmailAuthCode"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioentitlementparams",children:"ModioEntitlementParams"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiometricssessionparams",children:"ModioMetricsSessionParams"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiocreatemodfileparams",children:"ModioCreateModFileParams"}),"\n",(0,i.jsx)(d.h5,{id:"variables-20",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PathToModRootDirectory"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiocreatemodparams",children:"ModioCreateModParams"}),"\n",(0,i.jsx)(d.h5,{id:"variables-21",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PathToLogoFile"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Name"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Summary"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioeditmodparams",children:"ModioEditModParams"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioentitlementwalletbalance",children:"ModioEntitlementWalletBalance"}),"\n",(0,i.jsx)(d.p,{children:"Updated wallet balance from the sync entitlements call"}),"\n",(0,i.jsx)(d.h5,{id:"variables-22",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioUnsigned64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Balance"})}),(0,i.jsx)(d.td,{children:"The updated balance of the wallet"})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalentitlementwalletbalance",children:"ModioOptionalEntitlementWalletBalance"}),"\n",(0,i.jsx)(d.p,{children:"Optional updated wallet balance from the sync entitlements call"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioentitlementconsumptionvirtualcurrencydetails",children:"ModioEntitlementConsumptionVirtualCurrencyDetails"}),"\n",(0,i.jsx)(d.p,{children:"Further details about a Virtual Currency entitlement that was consumed"}),"\n",(0,i.jsx)(d.h5,{id:"variables-23",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int32"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TokensAllocated"})}),(0,i.jsx)(d.td,{children:"Amount of tokens that were issued for this specific entitlement consumption"})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"entitlementconsumptionstatus",children:"EntitlementConsumptionStatus"}),"\n",(0,i.jsx)(d.p,{children:"The result of an entitlement's consumption"}),"\n",(0,i.jsx)(d.h5,{id:"variables-24",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TransactionId"})}),(0,i.jsx)(d.td,{children:"ID of the transaction to redeem this entitlement"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioEntitlementConsumptionState"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TransactionState"})}),(0,i.jsx)(d.td,{children:"State of the transaction"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SkuId"})}),(0,i.jsx)(d.td,{children:"ID of the SKU that we attempted to consume"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntitlementConsumed"})}),(0,i.jsx)(d.td,{children:"Whether this entitlement was consumed or not"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioEntitlementType"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntitlementType"})}),(0,i.jsx)(d.td,{children:"Type of Entitlement that was consumed"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioEntitlementConsumptionVirtualCurrencyDetails"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"VirtualCurrencyDetails"})}),(0,i.jsx)(d.td,{children:"Details about virtual currency entitlement consumption"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioentitlementconsumptionstatuslist",children:"ModioEntitlementConsumptionStatusList"}),"\n",(0,i.jsx)(d.p,{children:"Class representing a list of entitlement consumption statuses that may be a page from a larger set of results"}),"\n",(0,i.jsx)(d.h5,{id:"variables-25",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioPagedResult"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PagedResult"})}),(0,i.jsx)(d.td,{children:"Stored property for a paged result, which provides context on the entitlement consumption status list"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InternalList"})}),(0,i.jsx)(d.td,{children:"Stored property for the dependency list"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioOptionalEntitlementWalletBalance"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"WalletBalance"})}),(0,i.jsx)(d.td,{children:"Updated wallet balance from syncing entitlements"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiofilterparams",children:"ModioFilterParams"}),"\n",(0,i.jsxs)(d.p,{children:["Class storing a set of filter parameters for use in ",(0,i.jsx)(d.a,{href:"#listallmodsasync",children:(0,i.jsx)(d.code,{children:"ListAllModsAsync"})})]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiopresetfilterparams",children:"ModioPresetFilterParams"}),"\n",(0,i.jsx)(d.h5,{id:"variables-26",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FText"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PresetName"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Tags"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ExcludedTags"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioSortDirection"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Direction"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioSortFieldType"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SortField"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Count"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioimagewrapper",children:"ModioImageWrapper"}),"\n",(0,i.jsx)(d.h5,{id:"variables-27",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ImagePath"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioinitializeoptions",children:"ModioInitializeOptions"}),"\n",(0,i.jsx)(d.h5,{id:"variables-28",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioGameID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameId"})}),(0,i.jsx)(d.td,{children:"The mod.io-provided ID for the game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioApiKey"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ApiKey"})}),(0,i.jsx)(d.td,{children:"The mod.io-provided API key for your application or game"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioEnvironment"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameEnvironment"})}),(0,i.jsx)(d.td,{children:"The mod.io environment you want to run the SDK on"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioPortal"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PortalInUse"})}),(0,i.jsx)(d.td,{children:"The portal your title is running through"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TMap"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ExtendedInitializationParameters"})}),(0,i.jsx)(d.td,{children:"Extended platform-specific initialization parameters. Refer to the platform documentation for valid keys and their values. Unrecognized values will be ignored"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bUseBackgroundThread"})}),(0,i.jsx)(d.td,{children:"Set mod.io to run with background thread"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodchangemap",children:"ModioModChangeMap"}),"\n",(0,i.jsx)(d.h5,{id:"variables-29",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TMap"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Changes"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodcollectionentry",children:"ModioModCollectionEntry"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodcreationhandle",children:"ModioModCreationHandle"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomoddependency",children:"ModioModDependency"}),"\n",(0,i.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,i.jsx)(d.p,{children:"This class is part of an experimental feature and is subject to change."})}),"\n",(0,i.jsx)(d.h5,{id:"variables-30",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioModID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModID"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModName"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateAdded"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FDateTime"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateUpdated"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"uint8"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DependencyDepth"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioLogo"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Logo"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioFileMetadata"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FileInfo"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioModServerSideStatus"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Status"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EModioObjectVisibilityFlags"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Visibility"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomoddependencylist",children:"ModioModDependencyList"}),"\n",(0,i.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,i.jsx)(d.p,{children:"This class is part of an experimental feature and is subject to change."})}),"\n",(0,i.jsx)(d.h5,{id:"variables-31",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioPagedResult"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PagedResult"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InternalList"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TotalFilesize"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"int64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TotalFilesizeUncompressed"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodprogressinfo",children:"ModioModProgressInfo"}),"\n",(0,i.jsx)(d.h5,{id:"variables-32",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioModID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ID"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionalmodprogressinfo",children:"ModioOptionalModProgressInfo"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioreportparams",children:"ModioReportParams"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiolink",children:"ModioLink"}),"\n",(0,i.jsx)(d.h5,{id:"variables-33",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Text"})}),(0,i.jsx)(d.td,{children:"The user-facing text for the link"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"URL"})}),(0,i.jsx)(d.td,{children:"The actual URL for the link"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bool"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bRequired"})}),(0,i.jsx)(d.td,{children:"Is displaying this link mandatory?"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modioterms",children:"ModioTerms"}),"\n",(0,i.jsx)(d.h5,{id:"variables-34",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"AgreeButtonText"})}),(0,i.jsx)(d.td,{children:"Text to display on the affirmative/OK button"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DisagreeButtonText"})}),(0,i.jsx)(d.td,{children:"Text to display on the negative/cancel button"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioLink"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"WebsiteLink"})}),(0,i.jsx)(d.td,{children:"Link to the mod.io website"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioLink"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TermsLink"})}),(0,i.jsx)(d.td,{children:"Link to the mod.io terms of use"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioLink"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PrivacyLink"})}),(0,i.jsx)(d.td,{children:"Link to the mod.io Privacy Policy"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioLink"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ManageLink"})}),(0,i.jsx)(d.td,{children:"Link to the mod.io Manage User Account page"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TermsText"})}),(0,i.jsx)(d.td,{children:"The plaintext version of the mod.io terms of use"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiotransactionrecord",children:"ModioTransactionRecord"}),"\n",(0,i.jsx)(d.h5,{id:"variables-35",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioModID"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"AssociatedModID"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioUnsigned64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Price"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioUnsigned64"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UpdatedUserWalletBalance"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiooptionaluser",children:"ModioOptionalUser"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiouserlist",children:"ModioUserList"}),"\n",(0,i.jsx)(d.h5,{id:"variables-36",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FModioPagedResult"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PagedResult"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InternalList"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiovalidationerror",children:"ModioValidationError"}),"\n",(0,i.jsx)(d.p,{children:"Wrapper struct containing information about a field validation error"}),"\n",(0,i.jsx)(d.h5,{id:"variables-37",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FieldName"})}),(0,i.jsx)(d.td,{children:"String description of the field that failed validation"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FString"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ValidationFailureDescription"})}),(0,i.jsx)(d.td,{children:"String description of the validation failure"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiocreatemodfilememoryparams",children:"ModioCreateModFileMemoryParams"}),"\n",(0,i.jsx)(d.h5,{id:"variables-38",children:"Variables"}),"\n",(0,i.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TArray"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModMemory"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h2,{id:"functions",children:"Functions"}),"\n",(0,i.jsx)(d.h3,{id:"set-session-identifier",children:"Set Session Identifier"}),"\n",(0,i.jsxs)(d.p,{children:["Changes the session identifier for the provided set of ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetSessionIdentifier.png",src:s(92464).A+"",width:"297",height:"111"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetSessionIdentifier(FModioInitializeOptions Options, FString SessionIdentifier)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-64",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Options"})}),(0,i.jsxs)(d.td,{children:["The template ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionIdentifier"})}),(0,i.jsx)(d.td,{children:"The new session id to use"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-16",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["New ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the session identifier set to the desired value"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-session-id",children:"Set Session Id"}),"\n",(0,i.jsx)(d.p,{children:"Changes the session identifier for the provided Metrics Session parameters"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetSessionId.png",src:s(81294).A+"",width:"224",height:"106"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioMetricsSessionParams SetSessionId(FModioMetricsSessionParams Params, FModioGuid Id)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-65",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Params"})}),(0,i.jsx)(d.td,{children:"The template Metrics Session parameters"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Id"})}),(0,i.jsx)(d.td,{children:"The intended Guid to store in the Metrics Session parameters"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-portal",children:"Set Portal"}),"\n",(0,i.jsxs)(d.p,{children:["Changes the portal for the provided set of ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetPortal.png",src:s(80807).A+"",width:"379",height:"134"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetPortal(FModioInitializeOptions Options, EModioPortal PortalToUse)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-66",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Options"})}),(0,i.jsxs)(d.td,{children:["The template ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PortalToUse"})}),(0,i.jsx)(d.td,{children:"The new portal to use"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-17",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["New ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the portal set to the desired value"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-game-id",children:"Set Game Id"}),"\n",(0,i.jsxs)(d.p,{children:["Changes the game id for the provided set of ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetGameId.png",src:s(12376).A+"",width:"249",height:"111"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetGameId(FModioInitializeOptions Options, int64 GameId)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-67",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Options"})}),(0,i.jsxs)(d.td,{children:["The template ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameId"})}),(0,i.jsx)(d.td,{children:"The new game id to use"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-18",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["New ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the game id set to the desired value"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-game-environment",children:"Set Game Environment"}),"\n",(0,i.jsxs)(d.p,{children:["Changes the game environment for the provided set of ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetGameEnvironment.png",src:s(39644).A+"",width:"379",height:"134"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetGameEnvironment(FModioInitializeOptions Options, EModioEnvironment GameEnvironment)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-68",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Options"})}),(0,i.jsxs)(d.td,{children:["The template ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameEnvironment"})}),(0,i.jsx)(d.td,{children:"The new environment to use"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-19",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["New ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the game environment set to the desired value"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-extended-initialization-parameters",children:"Set Extended Initialization Parameters"}),"\n",(0,i.jsxs)(d.p,{children:["Sets extended initialization parameters for the provided set of ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetExtendedInitializationParameters.png",src:s(92606).A+"",width:"304",height:"108"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetExtendedInitializationParameters(FModioInitializeOptions Options, TMap ExtendedParameters)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-69",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Options"})}),(0,i.jsxs)(d.td,{children:["The template ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ExtendedParameters"})}),(0,i.jsx)(d.td,{children:"The new extended parameters to use (will overwrite existing values)"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-20",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["New ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the extended parameters set to the desired value"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-api-key",children:"Set API Key"}),"\n",(0,i.jsxs)(d.p,{children:["Changes the API key for the provided set of ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetAPIKey.png",src:s(70504).A+"",width:"244",height:"111"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetAPIKey(FModioInitializeOptions Options, FString APIKey)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-70",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Options"})}),(0,i.jsxs)(d.td,{children:["The template ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"APIKey"})}),(0,i.jsx)(d.td,{children:"The new API key to use"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-21",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["New ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the API key set to the desired value"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodid--modiomodid",children:"ModioModID != ModioModID"}),"\n",(0,i.jsx)(d.p,{children:"Compares two mod IDs to check whether they're not equal"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_NotEqualTo.png",src:s(68979).A+"",width:"159",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool NotEqualTo(FModioModID A, FModioModID B)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-71",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"A"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"B"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"make-metrics-session-params",children:"Make Metrics Session Params"}),"\n",(0,i.jsx)(d.p,{children:"Create Metrics Session parameters"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeMetricsSessionParams.png",src:s(59800).A+"",width:"258",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioMetricsSessionParams MakeMetricsSessionParams(TArray Ids)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-72",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Ids"})}),(0,i.jsx)(d.td,{children:"The list of mods to store within the Metrics Session parameters"})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"make-initialize-options",children:"Make Initialize Options"}),"\n",(0,i.jsxs)(d.p,{children:["Make ",(0,i.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,i.jsx)(d.code,{children:"ModioInitializeOptions"})}),". Should only be used in conjunction with ",(0,i.jsx)(d.a,{href:"#initializeasync",children:(0,i.jsx)(d.code,{children:"InitializeAsync"})}),"."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeInitializeOptions.png",src:s(61327).A+"",width:"379",height:"232"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions MakeInitializeOptions(int64 GameId, FString APIKey, EModioEnvironment GameEnvironment, EModioPortal PortalInUse)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-73",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameId"})}),(0,i.jsxs)(d.td,{children:["A positive integer that maps to your game. This can be found in the admin section of your game's page at ",(0,i.jsx)(d.a,{href:"https://mod.io/",children:"https://mod.io/"})]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"APIKey"})}),(0,i.jsxs)(d.td,{children:["The API key for your game. This can be found in the admin section of your game's page at ",(0,i.jsx)(d.a,{href:"https://mod.io/",children:"https://mod.io/"})]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameEnvironment"})}),(0,i.jsx)(d.td,{children:"The environment your game has been set up on: test or live."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PortalInUse"})}),(0,i.jsxs)(d.td,{children:["The ",(0,i.jsx)(d.a,{href:"#EModioPortal",children:(0,i.jsx)(d.code,{children:"EModioPortal"})})," representing the store or service your game is being distributed through. Defaults to ",(0,i.jsx)(d.code,{children:"EModioPortal::None"}),"."]})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"make-guid",children:"Make Guid"}),"\n",(0,i.jsx)(d.p,{children:"Create a Guid from a string"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeGuid.png",src:s(11520).A+"",width:"230",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioGuid MakeGuid(FString Guid)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-74",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Guid"})}),(0,i.jsx)(d.td,{children:"A string to wrap within a Guid struct"})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"make-game-id",children:"Make Game Id"}),"\n",(0,i.jsxs)(d.p,{children:["Create a game id from a integer. Should only be used in conjunction with ",(0,i.jsx)(d.a,{href:"#initializeasync",children:(0,i.jsx)(d.code,{children:"InitializeAsync"})})]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeGameId.png",src:s(18778).A+"",width:"249",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioGameID MakeGameId(int64 GameId)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-75",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameId"})}),(0,i.jsx)(d.td,{children:"A positive integer that maps to your game"})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"make-entitlement-params",children:"Make Entitlement Params"}),"\n",(0,i.jsx)(d.p,{children:"Create entitlement parameters"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeEntitlementParams.png",src:s(4568).A+"",width:"303",height:"78"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioEntitlementParams MakeEntitlementParams(TMap ExtendedParameters)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-76",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ExtendedParameters"})}),(0,i.jsx)(d.td,{children:"A map to store extended parameters required by some portals"})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"make-auth-params",children:"Make Auth Params"}),"\n",(0,i.jsxs)(d.p,{children:["Creates a ",(0,i.jsx)(d.code,{children:"ModioAuthenticationParams"})," object"]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeAuthParams.png",src:s(11595).A+"",width:"306",height:"150"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioAuthenticationParams MakeAuthParams(FString AuthToken, FString EmailAddress, bool bHasAcceptedTOS)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-77",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"AuthToken"})}),(0,i.jsx)(d.td,{children:"Authentication provider-supplied OAuth token"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EmailAddress"})}),(0,i.jsx)(d.td,{children:"User email address, can be left blank"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bHasAcceptedTOS"})}),(0,i.jsx)(d.td,{children:"Has the user been shown the Terms of Service and accepted the Terms of Service?"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-22",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["The constructed ",(0,i.jsx)(d.code,{children:"ModioAuthenticationParams"})," object for use with ",(0,i.jsx)(d.a,{href:"#authenticateuserexternalasync",children:(0,i.jsx)(d.code,{children:"AuthenticateUserExternalAsync"})})]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"make-api-key",children:"Make Api Key"}),"\n",(0,i.jsxs)(d.p,{children:["Create an ApiKey id from a string. Should only be used in conjunction with ",(0,i.jsx)(d.a,{href:"#initializeasync",children:(0,i.jsx)(d.code,{children:"InitializeAsync"})})]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeApiKey.png",src:s(4650).A+"",width:"246",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioApiKey MakeApiKey(FString ApiKey)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-78",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ApiKey"})}),(0,i.jsx)(d.td,{children:"The api key from your settings panel on mod.io"})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-raw-value-from-mod-id",children:"Get Raw Value from Mod ID"}),"\n",(0,i.jsxs)(d.p,{children:["Retrieves the raw underlying value from an ",(0,i.jsx)(d.code,{children:"ModioModID"}),". ",(0,i.jsx)(d.code,{children:"ModioModID"}),"s are intended as opaque types, so use with care."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_GetRawValueFromModID.png",src:s(89913).A+"",width:"240",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"int64 GetRawValueFromModID(FModioModID In)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-79",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"In"})}),(0,i.jsxs)(d.td,{children:["The ",(0,i.jsx)(d.code,{children:"ModioModID"})," to retrieve the value for"]})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-23",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"The underlying value"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiomodid--modiomodid-1",children:"ModioModID == ModioModID"}),"\n",(0,i.jsx)(d.p,{children:"Compares two mod IDs to check whether they're equal"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_EqualTo.png",src:s(47438).A+"",width:"159",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool EqualTo(FModioModID A, FModioModID B)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-80",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"A"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"B"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-version-string",children:"Set Version String"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetVersionString.png",src:s(99860).A+"",width:"186",height:"143"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetVersionString(FModioCreateModFileParams In, FString Version)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-tags",children:"Set Tags"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetTags.png",src:s(35536).A+"",width:"137",height:"138"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetTags(FModioCreateModParams In, TArray Tags)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-modfile-platforms",children:"Set Modfile Platforms"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetModfilePlatforms.png",src:s(34393).A+"",width:"208",height:"138"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetModfilePlatforms(FModioCreateModFileParams In, TArray Platforms)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-mod-file-metadata-blob",children:"Set Mod File Metadata Blob"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetModFileMetadataBlob.png",src:s(37723).A+"",width:"239",height:"143"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetModFileMetadataBlob(FModioCreateModFileParams In, FString MetadataBlob)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-metadata-blob",children:"Set Metadata Blob"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetMetadataBlob.png",src:s(66981).A+"",width:"212",height:"143"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetMetadataBlob(FModioCreateModParams In, FString MetadataBlob)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-mark-as-active-release",children:"Set Mark as Active Release"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetMarkAsActiveRelease.png",src:s(11563).A+"",width:"258",height:"142"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetMarkAsActiveRelease(FModioCreateModFileParams In, bool bMarkAsActiveRelease)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-initial-visibility-deprecated",children:"Set Initial Visibility DEPRECATED"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetInitialVisibility_DEPRECATED.png",src:s(56359).A+"",width:"269",height:"142"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetInitialVisibility_DEPRECATED(FModioCreateModParams In, bool InitialVisibility)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-initial-visibility",children:"Set Initial Visibility"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetInitialVisibility.png",src:s(7163).A+"",width:"306",height:"166"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetInitialVisibility(FModioCreateModParams In, EModioObjectVisibilityFlags InitialVisibility)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-homepage-url",children:"Set Homepage URL"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetHomepageURL.png",src:s(8796).A+"",width:"216",height:"143"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetHomepageURL(FModioCreateModParams In, FString HomepageURL)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-description",children:"Set Description"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetDescription.png",src:s(81069).A+"",width:"193",height:"143"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetDescription(FModioCreateModParams In, FString Description)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"set-changelog-string",children:"Set Changelog String"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetChangelogString.png",src:s(70690).A+"",width:"202",height:"143"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void SetChangelogString(FModioCreateModFileParams In, FString Changelog)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-localized-text-for-enum-by-name",children:"Get Localized Text for Enum by Name"}),"\n",(0,i.jsxs)(d.p,{children:["Returns the string table ",(0,i.jsx)(d.code,{children:"FText"})," for a given enum value's ",(0,i.jsx)(d.code,{children:"FName"}),". Only works with enums registered via ",(0,i.jsx)(d.code,{children:"ModioUI::RegisterEnumAsLocalizable"}),"."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUIEnumLocalizationLibrary_GetLocalizedTextForEnumByName.png",src:s(49443).A+"",width:"298",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FText GetLocalizedTextForEnumByName(FName EnumName)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-81",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EnumName"})}),(0,i.jsx)(d.td,{children:"The Name from a given enum value"})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-24",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"localized Text for the specified enum value, or dummy FText if not found"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"filesizetotext-unsigned64",children:"FileSizeToText (Unsigned64)"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUIEnumLocalizationLibrary_FileSizeUnsigned64_ToText.png",src:s(54294).A+"",width:"441",height:"197"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FText FileSizeUnsigned64_ToText(FModioUnsigned64 FileSize, int32 MinDecimals, int32 MaxDecimals, TEnumAsByte Unit, bool bIncludeUnitName)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-82",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FileSize"})}),(0,i.jsx)(d.td,{children:"Filesize in bytes"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MinDecimals"})}),(0,i.jsx)(d.td,{children:"Minimum number of decimals to display for the filesize"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MaxDecimals"})}),(0,i.jsx)(d.td,{children:"Maximum number of decimals to display for the filesize"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Unit"})}),(0,i.jsxs)(d.td,{children:["If ",(0,i.jsx)(d.code,{children:"Largest"}),", it tries to display the size in the largest unit that will have a integral part > 0, else it displays the filesize in the specified unit"]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bIncludeUnitName"})}),(0,i.jsx)(d.td,{children:"Whether or not to include the unit name"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-25",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["An ",(0,i.jsx)(d.code,{children:"FText"})," formatted with your specifications"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-localized-text-from-default-table-by-key",children:"Get Localized Text from Default Table by Key"}),"\n",(0,i.jsxs)(d.p,{children:["Returns the string table ",(0,i.jsx)(d.code,{children:"FText"})," for a given string key"]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUILocalizationLibrary_GetLocalizedTextFromDefaultTableByKey.png",src:s(6841).A+"",width:"342",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FText GetLocalizedTextFromDefaultTableByKey(FString StringKey)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-83",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"StringKey"})}),(0,i.jsx)(d.td,{children:"The key to look up in the table"})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-26",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["Localized Text for the specified key, or ",(0,i.jsx)(d.code,{children:"StringKey"})," if not found"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"reconstruct-error",children:"Reconstruct Error"}),"\n",(0,i.jsx)(d.p,{children:"Helper method to reconstruct a mod.io error passed via code that cannot reference mod.io types"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioErrorCodeLibrary_ReconstructError.png",src:s(12246).A+"",width:"254",height:"116"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode ReconstructError(int32 Value, int32 Category)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-84",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Value"})}),(0,i.jsx)(d.td,{children:"The numeric value of the code"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Category"})}),(0,i.jsx)(d.td,{children:"The category ID (populated by native code)"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"iserror",children:"IsError"}),"\n",(0,i.jsx)(d.p,{children:"Checks if an error code contains a error"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioErrorCodeLibrary_IsErrorAsExec.png",src:s(30429).A+"",width:"164",height:"110"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool IsErrorAsExec(FModioErrorCode Error)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-85",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Error"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-27",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"true if the error code is an error"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-value",children:"Get Value"}),"\n",(0,i.jsx)(d.p,{children:"Get underlying error code for an FModioErrorCode."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioErrorCodeLibrary_GetValue.png",src:s(70153).A+"",width:"208",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"int32 GetValue(FModioErrorCode Error)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-86",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Error"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-28",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"The underlying error code. 0 represents no error."}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-message",children:"Get Message"}),"\n",(0,i.jsx)(d.p,{children:"Get the textual representation of the error"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioErrorCodeLibrary_GetMessage.png",src:s(6715).A+"",width:"208",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FString GetMessage(FModioErrorCode Error)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-87",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Error"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-29",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["An ",(0,i.jsx)(d.code,{children:"FString"})," message describing the error"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"error-code-matches",children:"Error Code Matches"}),"\n",(0,i.jsx)(d.p,{children:"Checks if the passed-in ErrorCode matches the specified error condition"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioErrorConditionLibrary_ErrorCodeMatches.png",src:s(20798).A+"",width:"379",height:"166"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool ErrorCodeMatches(FModioErrorCode ErrorCode, EModioErrorCondition Condition)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-88",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ErrorCode"})}),(0,i.jsx)(d.td,{children:"The code to check"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Condition"})}),(0,i.jsx)(d.td,{children:"The error condition to check against"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-30",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"True if the code matches the condition"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"list-user-subscription-async",children:"List User Subscription Async"}),"\n",(0,i.jsx)(d.p,{children:"Runs a filter over the user's subscription list"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioExampleLibrary_ListUserSubscriptionAsync.png",src:s(97963).A+"",width:"249",height:"138"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void ListUserSubscriptionAsync(FModioFilterParams FilterParams, FOnListAllModsDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-89",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FilterParams"})}),(0,i.jsx)(d.td,{children:"The filters to use on the user's subscription list"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Called when mod list has been processed"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-logo-thumbnail-size",children:"Get Logo Thumbnail Size"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioExampleLibrary_GetLogoThumbnailSize.png",src:s(49633).A+"",width:"223",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioLogoSize GetLogoThumbnailSize()\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-logo-full-size",children:"Get Logo Full Size"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioExampleLibrary_GetLogoFullSize.png",src:s(57932).A+"",width:"184",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioLogoSize GetLogoFullSize()\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-avatar-thumbnail-size",children:"Get Avatar Thumbnail Size"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioExampleLibrary_GetAvatarThumbnailSize.png",src:s(79139).A+"",width:"234",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioAvatarSize GetAvatarThumbnailSize()\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"to-filter-params",children:"To Filter Params"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioPresetFilterParamsLibrary_ToFilterParams.png",src:s(5617).A+"",width:"216",height:"108"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioFilterParams ToFilterParams(FModioPresetFilterParams Preset)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"load-async",children:"Load Async"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_LoadAsync.png",src:s(49510).A+"",width:"200",height:"138"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void LoadAsync(FModioImageWrapper Image, FOnLoadImageDelegate OnImageLoaded)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-texture",children:"Get Texture"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetTexture.png",src:s(9549).A+"",width:"215",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"UTexture2DDynamic* GetTexture(FModioImageWrapper Image)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-state",children:"Get State"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetState.png",src:s(33175).A+"",width:"215",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioImageState GetState(FModioImageWrapper Image)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-logo-size",children:"Get Logo Size"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetLogoSize.png",src:s(30484).A+"",width:"379",height:"158"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FVector2D GetLogoSize(UTexture* Logo, EModioLogoSize LogoSize)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-90",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Logo"})}),(0,i.jsxs)(d.td,{children:["If null and ",(0,i.jsx)(d.code,{children:"EModioLogoSize::Original"})," is passed, then (0, 0) is returned"]})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"LogoSize"})}),(0,i.jsx)(d.td,{children:"The size of the logo we want to return"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-31",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"Dimensions of the logo if displayed in a 1:1 pixel ratio"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-gallery-size",children:"Get Gallery Size"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetGallerySize.png",src:s(84145).A+"",width:"379",height:"158"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FVector2D GetGallerySize(UTexture* GalleryImage, EModioGallerySize GallerySize)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-91",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GalleryImage"})}),(0,i.jsx)(d.td,{children:"If null and EModioGallerySize::Original is passed, then (0, 0) is returned"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GallerySize"})}),(0,i.jsx)(d.td,{children:"The size of the gallery image we want to return"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-32",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"Dimensions of the gallery image if displayed in a 1:1 pixel ratio"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-avatar-size",children:"Get Avatar Size"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetAvatarSize.png",src:s(11778).A+"",width:"379",height:"158"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FVector2D GetAvatarSize(UTexture* Avatar, EModioAvatarSize AvatarSize)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-92",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Avatar"})}),(0,i.jsx)(d.td,{children:"If null and EModioAvatarSize::Original is passed, then (0, 0) is returned"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"AvatarSize"})}),(0,i.jsx)(d.td,{children:"The size of the avatar we want to return"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-33",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"Dimensions of the avatar if displayed in a 1:1 pixel ratio"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-path",children:"Get Path"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioModCollectionLibrary_GetPath.png",src:s(41242).A+"",width:"210",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FString GetPath(FModioModCollectionEntry Entry)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-93",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Entry"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-34",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:["Path to the mod's installation folder on disk. If the mod is not yet installed this path may not yet exist. Check ",(0,i.jsx)(d.code,{children:"GetModState"})," before trying to load files in this location"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-mod-state",children:"Get Mod State"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioModCollectionLibrary_GetModState.png",src:s(90164).A+"",width:"210",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioModState GetModState(FModioModCollectionEntry Entry)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-94",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Entry"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-35",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"EModioModState enum representing current state of the mod"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-mod-profile",children:"Get Mod Profile"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioModCollectionLibrary_GetModProfile.png",src:s(28636).A+"",width:"210",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioModInfo GetModProfile(FModioModCollectionEntry Entry)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-95",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Entry"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-36",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"FModioModInfo containing mod profile data"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-id",children:"Get ID"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioModCollectionLibrary_GetID.png",src:s(57720).A+"",width:"210",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioModID GetID(FModioModCollectionEntry Entry)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-96",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Entry"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-37",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"Mod ID"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-total-progress",children:"Get Total Progress"}),"\n",(0,i.jsx)(d.p,{children:"Retrieves the total amount of progress required for the specified state."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioModProgressInfoLibrary_GetTotalProgress.png",src:s(34813).A+"",width:"379",height:"134"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 GetTotalProgress(FModioModProgressInfo Info, EModioModProgressState State)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-97",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Info"})}),(0,i.jsx)(d.td,{children:"the progress struct to query"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"State"})}),(0,i.jsx)(d.td,{children:"which state to query total progress for"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-38",children:"Returns"}),"\n",(0,i.jsxs)(d.p,{children:[(0,i.jsx)(d.code,{children:"Modio::FileSize"})," for total progress in bytes"]}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-current-state",children:"Get Current State"}),"\n",(0,i.jsx)(d.p,{children:"Returns a EModioModProgressState indicating which state the mod operation is in"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioModProgressInfoLibrary_GetCurrentState.png",src:s(54858).A+"",width:"203",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioModProgressState GetCurrentState(FModioModProgressInfo Info)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-98",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Info"})}),(0,i.jsx)(d.td,{children:"The progress struct to query"})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-current-progress",children:"Get Current Progress"}),"\n",(0,i.jsxs)(d.p,{children:["Retrieves the progress value for the specified state. ",(0,i.jsx)(d.code,{children:"CurrentProgress == TotalProgress"})," for states which have completed, for example if a mod is currently Extracting, then passing in Downloading would give you a value equal to the total download size because the download has completed"]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioModProgressInfoLibrary_GetCurrentProgress.png",src:s(62440).A+"",width:"379",height:"134"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 GetCurrentProgress(FModioModProgressInfo Info, EModioModProgressState State)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-99",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Info"})}),(0,i.jsx)(d.td,{children:"the progress struct to query"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"State"})}),(0,i.jsx)(d.td,{children:"which state to query progress information for"})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-39",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"FModioUnsigned64 containing current progress in bytes"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-tags",children:"Get Tags"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioModTagOptionsLibrary_GetTags.png",src:s(90998).A+"",width:"232",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"TArray GetTags(FModioModTagOptions ModTags)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-100",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModTags"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-40",children:"Returns"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-paged-result",children:"Get Paged Result"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioModTagOptionsLibrary_GetPagedResult.png",src:s(91837).A+"",width:"236",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioPagedResult GetPagedResult(FModioModTagOptions ModTags)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-101",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModTags"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-41",children:"Returns"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-default-portal-for-current-platform",children:"Get Default Portal for Current Platform"}),"\n",(0,i.jsx)(d.p,{children:"Get the default portal for the platform the game is running on."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioPlatformHelpersLibrary_GetDefaultPortalForCurrentPlatform.png",src:s(5635).A+"",width:"307",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioPortal GetDefaultPortalForCurrentPlatform()\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-102",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsx)(d.table,{children:(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})})})}),"\n",(0,i.jsx)(d.h4,{id:"returns-42",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"EModioPortal of the portal to use"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-default-auth-provider-for-current-platform",children:"Get Default Auth Provider for Current Platform"}),"\n",(0,i.jsx)(d.p,{children:"Get the default Authentication Provider for the current platform the game is running on"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioPlatformHelpersLibrary_GetDefaultAuthProviderForCurrentPlatform.png",src:s(53632).A+"",width:"351",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioAuthenticationProvider GetDefaultAuthProviderForCurrentPlatform()\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-103",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsx)(d.table,{children:(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})})})}),"\n",(0,i.jsx)(d.h4,{id:"returns-43",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"EModioAuthenticationProvider to use for authentication calls"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-current-platform",children:"Get Current Platform"}),"\n",(0,i.jsx)(d.p,{children:"Gets the current platform that the game is running on"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioPlatformHelpersLibrary_GetCurrentPlatform.png",src:s(55989).A+"",width:"203",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioPlatformName GetCurrentPlatform()\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-104",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsx)(d.table,{children:(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})})})}),"\n",(0,i.jsx)(d.h4,{id:"returns-44",children:"Returns"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"round-number-string",children:"Round Number String"}),"\n",(0,i.jsx)(d.p,{children:"Sets the correct decimals depending on the file size or speed"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_RoundNumberString.png",src:s(10851).A+"",width:"287",height:"84"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FText RoundNumberString(FText inputText)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-105",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"inputText"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-percent-integer64integer64",children:"Get Percent (integer64/integer64)"}),"\n",(0,i.jsxs)(d.p,{children:["Calculates percentage using two ",(0,i.jsx)(d.code,{children:"int64"})," params"]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_Pct_Int64Int64.png",src:s(31773).A+"",width:"267",height:"92"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"float Pct_Int64Int64(int64 Dividend, int64 Divisor)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-106",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Dividend"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Divisor"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-45",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"The floating point result of Dividend/Divisor with no checks"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"is-valid-security-code-format",children:"Is Valid Security Code Format"}),"\n",(0,i.jsx)(d.p,{children:"Checks if the string has the same format as the mod.io security code"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_IsValidSecurityCodeFormat.png",src:s(8421).A+"",width:"252",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool IsValidSecurityCodeFormat(FString String)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-107",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"String"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-46",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"True if the security code has a valid format"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"is-valid-email-address-format",children:"Is Valid Email Address Format"}),"\n",(0,i.jsx)(d.p,{children:"Does a basic validation if the email address supplied has a valid form"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_IsValidEmailAddressFormat.png",src:s(5608).A+"",width:"253",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool IsValidEmailAddressFormat(FString String)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-108",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"String"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-47",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"True if the email address has a valid format"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-time-span-as-string",children:"Get Time Span as String"}),"\n",(0,i.jsx)(d.p,{children:"Gets the time span between present and specified past date FString"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetTimeSpanAsString.png",src:s(52987).A+"",width:"293",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FString GetTimeSpanAsString(FString PastDateString)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-109",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PastDateString"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-shortened-number-as-string",children:"Get Shortened Number as String"}),"\n",(0,i.jsx)(d.p,{children:"Shortens the specified large number"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetShortenedNumberAsString.png",src:s(25047).A+"",width:"269",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FString GetShortenedNumberAsString(int64 Number)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-110",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Number"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-project-initialize-options-for-session-id",children:"Get Project Initialize Options for Session Id"}),"\n",(0,i.jsx)(d.p,{children:"Get the options needed to initialize the mod.io SDK specified in the project settings"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetProjectInitializeOptionsForSessionId.png",src:s(90298).A+"",width:"330",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions GetProjectInitializeOptionsForSessionId(FString SessionId)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-111",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SessionId"})}),(0,i.jsx)(d.td,{children:"The LocalSessionIdentifier option to initialize project with"})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-project-game-id",children:"Get Project Game Id"}),"\n",(0,i.jsx)(d.p,{children:"Get the game id specified in the mod.io project settings"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetProjectGameId.png",src:s(84733).A+"",width:"198",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioGameID GetProjectGameId()\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-112",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsx)(d.table,{children:(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})})})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-project-environment",children:"Get Project Environment"}),"\n",(0,i.jsx)(d.p,{children:"Get the environment specified in the mod.io project settings"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetProjectEnvironment.png",src:s(75517).A+"",width:"223",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioEnvironment GetProjectEnvironment()\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-113",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsx)(d.table,{children:(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})})})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-project-api-key",children:"Get Project Api Key"}),"\n",(0,i.jsx)(d.p,{children:"Get the api key specified in the mod.io project settings"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetProjectApiKey.png",src:s(43465).A+"",width:"193",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioApiKey GetProjectApiKey()\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-114",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsx)(d.table,{children:(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})})})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-language-code-string",children:"Get Language Code String"}),"\n",(0,i.jsx)(d.p,{children:"Get language code string. This can be used for localization purposes"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetLanguageCodeString.png",src:s(11861).A+"",width:"379",height:"104"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FString GetLanguageCodeString(EModioLanguage Language)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-115",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Language"})}),(0,i.jsx)(d.td,{children:"The language code to convert to string"})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-48",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:'The language code as a string (e.g. "en", "fr", "de")'}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-language-code-from-string",children:"Get Language Code from String"}),"\n",(0,i.jsx)(d.p,{children:"Get language code enum from string in ISO 639-1 format. This can be used for localization purposes"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetLanguageCodeFromString.png",src:s(50425).A+"",width:"291",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"EModioLanguage GetLanguageCodeFromString(FString LanguageCode)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-116",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"LanguageCode"})}),(0,i.jsx)(d.td,{children:'The language code as a string (e.g. "en", "fr", "de")'})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-49",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"The language code as an enum"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-desired-file-size-unit",children:"Get Desired File Size Unit"}),"\n",(0,i.jsx)(d.p,{children:"Get desired file size unit based on the size of the file"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetDesiredFileSizeUnit.png",src:s(86318).A+"",width:"249",height:"81"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"TEnumAsByte GetDesiredFileSizeUnit(int64 FileSize)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-117",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FileSize"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-50",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"the desired file size unit"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"get-default-session-id-windows",children:"Get Default Session Id Windows"}),"\n",(0,i.jsx)(d.p,{children:"Get Session Id for Windows for initialization of the SDK"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetDefaultSessionIdWindows.png",src:s(84486).A+"",width:"266",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FString GetDefaultSessionIdWindows()\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-118",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsx)(d.table,{children:(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})})})}),"\n",(0,i.jsx)(d.h4,{id:"returns-51",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"The windows session id, or an empty string if you are not on Windows"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"tostring-filesize",children:"ToString (Filesize)"}),"\n",(0,i.jsx)(d.p,{children:"Converts a filesize to a human readable string with the appropriate unit"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_Filesize_ToString.png",src:s(86011).A+"",width:"441",height:"205"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FText Filesize_ToString(int64 FileSize, int32 MinDecimals, int32 MaxDecimals, TEnumAsByte Unit, bool bIncludeUnitName)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-119",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FileSize"})}),(0,i.jsx)(d.td,{children:"Filesize in bytes"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MinDecimals"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MaxDecimals"})}),(0,i.jsx)(d.td,{children:"Maximum amount of decimals to display of the filesize"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Unit"})}),(0,i.jsx)(d.td,{children:"If Largest, then it tries to display the size in the largest unit that will have a integral part > 0, else it displays the filesize in the specified unit"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"bIncludeUnitName"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-52",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"A text formatted from your specifications"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"getdefaultmodinstallationdirectory",children:"GetDefaultModInstallationDirectory"}),"\n",(0,i.jsx)(d.p,{children:"Returns the default mod installation directory for this game and platform, ignoring overrides and without requiring the SDK to be initialized."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetDefaultModInstallationDirectory.png",src:s(97423).A+"",width:"290",height:"108"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FString K2_GetDefaultModInstallationDirectory(FModioGameID GameID)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-120",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GameID"})}),(0,i.jsxs)(d.td,{children:["The ",(0,i.jsx)(d.code,{children:"ModioGameID"})," of the game we're fetching the default mod installation directory for."]})]})})]})}),"\n",(0,i.jsx)(d.h4,{id:"returns-53",children:"Returns"}),"\n",(0,i.jsx)(d.p,{children:"The default mod installation directory for the specified game on the current platform"}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64---modiounsigned64",children:"ModioUnsigned64 - ModioUnsigned64"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_Subtract.png",src:s(89948).A+"",width:"139",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 Subtract(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"percentage-unsigned-64",children:"Percentage Unsigned 64"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_Percentage_Unsigned64.png",src:s(88754).A+"",width:"222",height:"106"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"float Percentage_Unsigned64(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64",children:"ModioUnsigned64 != ModioUnsigned64"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_NotEqualTo.png",src:s(90214).A+"",width:"159",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool NotEqualTo(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"make-from-components",children:"Make from Components"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_MakeFromComponents.png",src:s(88442).A+"",width:"220",height:"106"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 MakeFromComponents(int32 High, int32 Low)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-1",children:"ModioUnsigned64 < ModioUnsigned64"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_LessThan.png",src:s(76134).A+"",width:"139",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool LessThan(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64--0",children:"ModioUnsigned64 > 0"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_GreaterThanZero.png",src:s(99531).A+"",width:"179",height:"54"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool GreaterThanZero(FModioUnsigned64 In)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-2",children:"ModioUnsigned64 > ModioUnsigned64"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_GreaterThan.png",src:s(42837).A+"",width:"139",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool GreaterThan(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-3",children:"ModioUnsigned64 == ModioUnsigned64"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_EqualTo.png",src:s(68337).A+"",width:"179",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool EqualTo(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-4",children:"ModioUnsigned64 / ModioUnsigned64"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_DivideToFloat.png",src:s(75196).A+"",width:"139",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"float DivideToFloat(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64--float",children:"ModioUnsigned64 / float"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_DivideFloat.png",src:s(27889).A+"",width:"171",height:"84"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"float DivideFloat(FModioUnsigned64 LHS, float RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-truncate",children:"ModioUnsigned64 / ModioUnsigned64 (truncate)"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_Divide.png",src:s(88015).A+"",width:"139",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 Divide(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"break-to-components",children:"Break to Components"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_BreakToComponents.png",src:s(13496).A+"",width:"206",height:"106"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void BreakToComponents(FModioUnsigned64 In, int32 High, int32 Low)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-5",children:"ModioUnsigned64 + ModioUnsigned64"}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_Add.png",src:s(34323).A+"",width:"139",height:"76"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 Add(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"submitnewmodfrommemoryasync",children:"SubmitNewModFromMemoryAsync"}),"\n",(0,i.jsx)(d.p,{children:"Submit a new mod, with its logo data coming from an in-memory buffer rather than a file."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync.png",src:s(64587).A+"",width:"287",height:"198"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitNewModFromMemoryAsync(FModioModCreationHandle Handle, FModioCreateModParams Params, TArray PngData, FOnSubmitNewModDelegate Callback)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"parameters-121",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Handle"})}),(0,i.jsx)(d.td,{children:"Mod creation handle"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Params"})}),(0,i.jsx)(d.td,{children:"Parameters to use when creating the mod"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PngData"})}),(0,i.jsx)(d.td,{children:"In-memory buffer, representative of a PNG file to be used for upload"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Callback"})}),(0,i.jsx)(d.td,{children:"Callback once operation has completed"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"submitnewmodfileformodfrommemory",children:"SubmitNewModFileForModFromMemory"}),"\n",(0,i.jsxs)(d.p,{children:["Queues the upload of a new mod file release for the specified mod, using the submitted parameters. This upload method accepts a block of memory ",(0,i.jsx)(d.code,{children:"TArray"})," rather than a file path. The upload's progress can be tracked in the same way as downloads; when completed, a Mod Management Event will be triggered with the result code for the upload."]}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory.png",src:s(89146).A+"",width:"315",height:"192"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitNewModFileForModFromMemory(UModioSubsystem* Target, FModioModID Mod, FModioCreateModFileMemoryParams Params)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"requirements-41",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h4,{id:"parameters-122",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UModioSubsystem*"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Mod"})}),(0,i.jsx)(d.td,{children:"The ID of the mod you are submitting a file for"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Params"})}),(0,i.jsx)(d.td,{children:"Information about the mod file being created, including the memory that wiull be uploaded as a mod"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"loadmodfiletomemory",children:"LoadModFileToMemory"}),"\n",(0,i.jsx)(d.p,{children:"Loads an installed mod file into memory."}),"\n",(0,i.jsx)(d.p,{children:(0,i.jsx)(d.img,{alt:"nd_img_ModioSubmissionExtensionLibrary_K2_LoadModFileToMemory.png",src:s(33338).A+"",width:"318",height:"162"})}),"\n",(0,i.jsx)(d.pre,{children:(0,i.jsx)(d.code,{className:"language-cpp",children:"bool K2_LoadModFileToMemory(UModioSubsystem* Target, FModioModID ModId, TArray ModData)\n"})}),"\n",(0,i.jsx)(d.h4,{id:"requirements-42",children:"Requirements"}),"\n",(0,i.jsxs)(d.ul,{children:["\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,i.jsx)(d.li,{children:(0,i.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,i.jsx)(d.h4,{id:"parameters-123",children:"Parameters"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Target"})}),(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UModioSubsystem*"})})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModId"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Mod Data"})}),(0,i.jsx)(d.td,{children:"A byte array of the mod that has been loaded"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h2,{id:"enums",children:"Enums"}),"\n",(0,i.jsx)(d.h3,{id:"EModioModfilePlatform",children:"EModioModfilePlatform"}),"\n",(0,i.jsx)(d.p,{children:"Enum representing the platform(s) that a modfile is enabled for"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Windows"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Mac"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Linux"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Android"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"iOS"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"XboxOne"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"XboxSeriesX"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PS4"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PS5"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Switch"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Oculus"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Source"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EGameMaturityFlags",children:"EGameMaturityFlags"}),"\n",(0,i.jsx)(d.p,{children:"Maturity options for a game"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"None"})}),(0,i.jsx)(d.td,{children:"Don't allow mature content in mods (default)"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MatureModsAllowed"})}),(0,i.jsx)(d.td,{children:"This game allows mods containing mature content"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MatureAudiencesOnly"})}),(0,i.jsx)(d.td,{children:"This game is for mature audiences only"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EGameMonetizationFlags",children:"EGameMonetizationFlags"}),"\n",(0,i.jsx)(d.p,{children:"Monetization properties of a game"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"None"})}),(0,i.jsx)(d.td,{children:"None set (default)"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Monetization"})}),(0,i.jsx)(d.td,{children:"Monetization is enabled"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Marketplace"})}),(0,i.jsx)(d.td,{children:"Marketplace is enabled"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PartnerProgram"})}),(0,i.jsx)(d.td,{children:"Partner Program is enabled"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioModServerSideStatus",children:"EModioModServerSideStatus"}),"\n",(0,i.jsx)(d.p,{children:"Enum representing a mod's server side status"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NotAccepted"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Accepted"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Deleted"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioVirusStatus",children:"EModioVirusStatus"}),"\n",(0,i.jsx)(d.p,{children:"If the file has been found to be malicious or not"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NoThreat"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Malicious"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioVirusScanStatus",children:"EModioVirusScanStatus"}),"\n",(0,i.jsx)(d.p,{children:"Current state of the scanned file"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NotScanned"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ScanComplete"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InProgress"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TooLargeToScan"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FileNotFound"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ErrorScanning"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioObjectVisibilityFlags",children:"EModioObjectVisibilityFlags"}),"\n",(0,i.jsx)(d.p,{children:"Enum representing whether or not a mod is visible to users"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Hidden"})}),(0,i.jsx)(d.td,{children:"Mod is concealed from users"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Public"})}),(0,i.jsx)(d.td,{children:"Mod is openly available"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioMaturityFlags",children:"EModioMaturityFlags"}),"\n",(0,i.jsx)(d.p,{children:"Enum representing mature content that a mod may contain"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"None"})}),(0,i.jsx)(d.td,{children:"No maturity"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Alcohol"})}),(0,i.jsx)(d.td,{children:"Content contains alcohol references"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Drugs"})}),(0,i.jsx)(d.td,{children:"Content contains drug references"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Violence"})}),(0,i.jsx)(d.td,{children:"Content contains violence references"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Explicit"})}),(0,i.jsx)(d.td,{children:"Content contains sexual references"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioModManagementEventType",children:"EModioModManagementEventType"}),"\n",(0,i.jsx)(d.p,{children:"What type of event occurred"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Installed"})}),(0,i.jsx)(d.td,{children:"Mod installation to local storage completed"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Uninstalled"})}),(0,i.jsx)(d.td,{children:"Mod uninstallation from local storage completed"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Updated"})}),(0,i.jsx)(d.td,{children:"Mod local installation updated to latest version"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Uploaded"})}),(0,i.jsx)(d.td,{children:"Mod file was uploaded"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"BeginInstall"})}),(0,i.jsx)(d.td,{children:"Mod download and installation has started"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"BeginUninstall"})}),(0,i.jsx)(d.td,{children:"Mod uninstallation has started"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"BeginUpdate"})}),(0,i.jsx)(d.td,{children:"Mod download and update has started"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"BeginUpload"})}),(0,i.jsx)(d.td,{children:"Mod upload has started"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioAuthenticationProvider",children:"EModioAuthenticationProvider"}),"\n",(0,i.jsx)(d.p,{children:"Simple struct to encapsulate data passed to external authentication systems"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"XboxLive"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Steam"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GoG"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Itch"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Switch"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Discord"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PSN"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Epic"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Oculus"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"OpenID"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GoogleIDToken"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GoogleServerSideToken"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioEnvironment",children:"EModioEnvironment"}),"\n",(0,i.jsxs)(d.p,{children:["Enum representing which environment the game is deployed to: ",(0,i.jsx)(d.code,{children:"Test"})," or ",(0,i.jsx)(d.code,{children:"Live"}),"."]}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Test"})}),(0,i.jsx)(d.td,{children:"Test (private) environment"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Live"})}),(0,i.jsx)(d.td,{children:"Live (public) environment"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioPortal",children:"EModioPortal"}),"\n",(0,i.jsx)(d.p,{children:"Enum representing the store or service your game is being distributed through"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"None"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Apple"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EpicGamesStore"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GOG"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Google"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Itchio"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Nintendo"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PSN"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Steam"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"XboxLive"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioPlatformName",children:"EModioPlatformName"}),"\n",(0,i.jsx)(d.p,{children:"Enum representing a named platform that the plugin is running on."}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Windows"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Mac"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Linux"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PS4"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PS5"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"XBoxOne"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"XSX"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Switch"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Unknown"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Android"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"iOS"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioLogoSize",children:"EModioLogoSize"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb320"})}),(0,i.jsx)(d.td,{children:"320x180px"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb640"})}),(0,i.jsx)(d.td,{children:"640x360px"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb1280"})}),(0,i.jsx)(d.td,{children:"1280x720px"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Original"})}),(0,i.jsx)(d.td,{children:"Original Size"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioAvatarSize",children:"EModioAvatarSize"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Original"})}),(0,i.jsx)(d.td,{children:"Original Size"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb50"})}),(0,i.jsx)(d.td,{children:"50x50px Thumbnail"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb100"})}),(0,i.jsx)(d.td,{children:"100x100px Thumbnail"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioGallerySize",children:"EModioGallerySize"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Original"})}),(0,i.jsx)(d.td,{children:"Original Size"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb320"})}),(0,i.jsx)(d.td,{children:"320x180px Thumbnail"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thumb1280"})}),(0,i.jsx)(d.td,{children:"1280x720 Thumbnail"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioLogLevel",children:"EModioLogLevel"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Trace"})}),(0,i.jsx)(d.td,{children:"Detailed low-level debugging output. Not intended for general use"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Detailed"})}),(0,i.jsx)(d.td,{children:"Detailed but not low-level. Generally useful for some mid-level information for debugging."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Info"})}),(0,i.jsx)(d.td,{children:"Informational output containing status messages"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Warning"})}),(0,i.jsx)(d.td,{children:"Warnings about incorrect plugin usage, timeouts"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Error"})}),(0,i.jsx)(d.td,{children:"Only errors"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioLanguage",children:"EModioLanguage"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"English"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Bulgarian"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"French"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"German"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Italian"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Polish"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Portuguese"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Hungarian"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Japanese"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Korean"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Russian"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Spanish"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Thai"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ChineseSimplified"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ChineseTraditional"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Turkish"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Ukrainian"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Arabic"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioModChangeType",children:"EModioModChangeType"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Added"})}),(0,i.jsx)(d.td,{children:"The user's list has a new mod to synchronize"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Removed"})}),(0,i.jsx)(d.td,{children:"The user's list must remove a mod to synchronize"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Updated"})}),(0,i.jsx)(d.td,{children:"The user's list must update a mod to synchronize"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EFileSizeUnit",children:"EFileSizeUnit"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Largest"})}),(0,i.jsx)(d.td,{children:"Will take the largest one that becomes a number larger than 1 (i.e, 1300mb becomes 1.3gb)"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"B"})}),(0,i.jsx)(d.td,{children:"A single byte"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"KB"})}),(0,i.jsx)(d.td,{children:"Kilobytes"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MB"})}),(0,i.jsx)(d.td,{children:"Megabytes"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"GB"})}),(0,i.jsx)(d.td,{children:"Gigabytes"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioEntitlementConsumptionState",children:"EModioEntitlementConsumptionState"}),"\n",(0,i.jsx)(d.p,{children:"State of an entitlement consumption transaction"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Failed"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Pending"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Fulfilled"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ConsumeLimitExceeded"})}),(0,i.jsx)(d.td,{})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioEntitlementType",children:"EModioEntitlementType"}),"\n",(0,i.jsx)(d.p,{children:"Type of entitlement that was consumed"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsx)(d.tbody,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"VirtualCurrency"})}),(0,i.jsx)(d.td,{})]})})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioErrorCondition",children:"EModioErrorCondition"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NoError"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NetworkError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code represents a connection or HTTP error between the client and the mod.io server."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ConfigurationError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates the SDK's configuration is not valid - the game ID or API key are incorrect or the game has been deleted."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InvalidArgsError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates the arguments passed to the function have failed validation or were otherwise invalid."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FilesystemError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates a permission or IO error when accessing local filesystem data."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InternalError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code represents an internal SDK error - please inform mod.io of the error code value."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ApiErrorRefSuccess"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error ref returned by the API indicates an implicit success because the operation has already been performed (ie a no-op is success)."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that a specified game, mod, user, media file or mod file was not found."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserTermsOfUseError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that the user has not yet accepted the mod.io Terms of Use."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SubmitReportError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that a report for the specified content could not be submitted."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that a user is not authenticated."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that the SDK has not been initialized."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UserAlreadyAuthenticatedError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that the user is already authenticated."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SystemError"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that a low-level system error occurred outside of mod.io SDK control."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"OperationCanceled"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that the asynchronous operation was cancelled before it completed."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that Mod Management has not been enabled."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RateLimited"})}),(0,i.jsx)(d.td,{children:"Too many requests made to the mod.io API within the rate-limiting window. Please wait and try again."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModBeingProcessed"})}),(0,i.jsx)(d.td,{children:"The specified mod's files are currently being updated by the SDK. Please try again later."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InsufficientSpace"})}),(0,i.jsx)(d.td,{children:"There is insufficient space to install the mod. Please free up space and try again."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SDKAlreadyInitialized"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that the SDK has already been initialized."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ModManagementAlreadyEnabled"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that Mod Management has already been enabled."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InsufficientPermissions"})}),(0,i.jsx)(d.td,{children:"When this condition is true, the error code indicates that the current user does not have the required permissions for this operation."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EmailLoginCodeInvalid"})}),(0,i.jsx)(d.td,{children:"The email login code is incorrect, has expired, or has already been used."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"AlreadySubscribed"})}),(0,i.jsx)(d.td,{children:"The specified mod is already subscribed to."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InstallOrUpdateCancelled"})}),(0,i.jsx)(d.td,{children:"The current mod installation or update was cancelled."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UploadCancelled"})}),(0,i.jsx)(d.td,{children:"The current modfile upload was cancelled."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"TempModSetNotInitialized"})}),(0,i.jsx)(d.td,{children:"TempModSet need to be initialized first, call InitTempModSet."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MonetizationOperationError"})}),(0,i.jsx)(d.td,{children:"An error occurred while performing a monetization operation."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PaymentTransactionFailed"})}),(0,i.jsx)(d.td,{children:"The transaction requires a payment but it could not be fulfilled. Please retry with funds on the wallet"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"IncorrectPrice"})}),(0,i.jsx)(d.td,{children:"The display price for the mod is out-of-date or incorrect. Please retry with the correct display price."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ItemAlreadyOwned"})}),(0,i.jsx)(d.td,{children:"The authenticated user already has acquired this item"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ParentalControlRestrictions"})}),(0,i.jsx)(d.td,{children:"Parental control restrictions prevent this account from accessing UGC."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MetricsSessionNotInitialized"})}),(0,i.jsx)(d.td,{children:"Metrics session has not yet been initialized. Ensure that you have a metrics secret key set for your project."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MetricsSessionAlreadyInitialized"})}),(0,i.jsx)(d.td,{children:"Metrics session has already been been initialized."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MetricsSessionIsActive"})}),(0,i.jsx)(d.td,{children:"Metrics session has been started."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MetricsSessionIsNotActive"})}),(0,i.jsx)(d.td,{children:"Metrics session has not been started. Please call MetricsSessionStartAsync."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"MetricsSessionHasNoMods"})}),(0,i.jsx)(d.td,{children:"No mods have been added to the session."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"PremiumFeatureNotAvailable"})}),(0,i.jsx)(d.td,{children:"This premium feature is not available for your project."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"EmailExchangeCodeAlreadyRedeemed"})}),(0,i.jsx)(d.td,{children:"The email security code has already been redeemed."})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioSortFieldType",children:"EModioSortFieldType"}),"\n",(0,i.jsx)(d.p,{children:"Enum indicating which field should be used to sort the results"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"ID"})}),(0,i.jsx)(d.td,{children:"Use mod ID (default)"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DownloadsToday"})}),(0,i.jsx)(d.td,{children:'Use number of downloads in last 24 (exposed in REST API as "popular")'})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"SubscriberCount"})}),(0,i.jsx)(d.td,{children:"Use number of subscribers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Rating"})}),(0,i.jsx)(d.td,{children:"Use mod rating"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateMarkedLive"})}),(0,i.jsx)(d.td,{children:"Use date mod was marked live"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DateUpdated"})}),(0,i.jsx)(d.td,{children:"Use date mod was last updated"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DownloadsTotal"})}),(0,i.jsx)(d.td,{children:"Use downloads total"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Alphabetical"})}),(0,i.jsx)(d.td,{children:"Use mod name"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioSortDirection",children:"EModioSortDirection"}),"\n",(0,i.jsx)(d.p,{children:"Enum indicating which direction sorting should be applied"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Ascending"})}),(0,i.jsx)(d.td,{})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Descending"})}),(0,i.jsx)(d.td,{children:"(default)"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioRevenueFilterType",children:"EModioRevenueFilterType"}),"\n",(0,i.jsx)(d.p,{children:"Enum indicating filtering options based off revenue type"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Free"})}),(0,i.jsx)(d.td,{children:"Return only free mods"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Paid"})}),(0,i.jsx)(d.td,{children:"Return only paid mods"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FreeAndPaid"})}),(0,i.jsx)(d.td,{children:"Return both free and paid mods"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioImageState",children:"EModioImageState"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"OnDisc"})}),(0,i.jsx)(d.td,{children:"Image data is located on hard drive"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"LoadingIntoMemory"})}),(0,i.jsx)(d.td,{children:"Image data is transferring to a memory location"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InMemory"})}),(0,i.jsx)(d.td,{children:"Image data is located in memory"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Corrupted"})}),(0,i.jsx)(d.td,{children:"Image data is not readable"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioModState",children:"EModioModState"}),"\n",(0,i.jsx)(d.p,{children:"Enum representing the current state of a mod"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"InstallationPending"})}),(0,i.jsx)(d.td,{children:"The mod is pending installation. This state is not saved."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Installed"})}),(0,i.jsx)(d.td,{children:"The mod is installed."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UpdatePending"})}),(0,i.jsx)(d.td,{children:"The mod is pending an update. This state is saved as installed."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Downloading"})}),(0,i.jsx)(d.td,{children:"The mod is downloading as part of the installation process. This state is not saved."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Extracting"})}),(0,i.jsx)(d.td,{children:"The mod is extracting as part of the installation process. This state is not saved."})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"UninstallPending"})}),(0,i.jsx)(d.td,{children:"The mod is pending uninstallation. This state is saved as installed."})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioModProgressState",children:"EModioModProgressState"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Initializing"})}),(0,i.jsx)(d.td,{children:"Download information is being retrieved from mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Downloading"})}),(0,i.jsx)(d.td,{children:"Mod archive is downloading from mod.io servers"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Extracting"})}),(0,i.jsx)(d.td,{children:"Mod archive is downloaded and now extracting"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Compressing"})}),(0,i.jsx)(d.td,{children:"Mod archive is being compressed from files on disk"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Uploading"})}),(0,i.jsx)(d.td,{children:"Mod archive is uploading to mod.io servers"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioRating",children:"EModioRating"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Neutral"})}),(0,i.jsx)(d.td,{children:"A neutral rating"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Positive"})}),(0,i.jsx)(d.td,{children:"A positive rating"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Negative"})}),(0,i.jsx)(d.td,{children:"A negative rating"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{}),"\n",(0,i.jsx)(d.h3,{id:"EModioReportType",children:"EModioReportType"}),"\n",(0,i.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,i.jsxs)(d.table,{children:[(0,i.jsx)(d.thead,{children:(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.th,{}),(0,i.jsx)(d.th,{})]})}),(0,i.jsxs)(d.tbody,{children:[(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Generic"})}),(0,i.jsx)(d.td,{children:"A generic mod report"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"DMCA"})}),(0,i.jsx)(d.td,{children:"Digital Millennium Copyright Act mod report"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"NotWorking"})}),(0,i.jsx)(d.td,{children:"Not working mod report"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"RudeContent"})}),(0,i.jsx)(d.td,{children:"Rude content mod report"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"IllegalContent"})}),(0,i.jsx)(d.td,{children:"Illegal content mod report"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"StolenContent"})}),(0,i.jsx)(d.td,{children:"Stolen content mod report"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"FalseInformation"})}),(0,i.jsx)(d.td,{children:"False information mod report"})]}),(0,i.jsxs)(d.tr,{children:[(0,i.jsx)(d.td,{children:(0,i.jsx)(d.code,{children:"Other"})}),(0,i.jsx)(d.td,{children:"Other type of mod report"})]})]})]})}),"\n",(0,i.jsx)(d.hr,{})]})}function a(e={}){const{wrapper:d}={...(0,r.R)(),...e.components};return d?(0,i.jsx)(d,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},47438:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},89913:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},4650:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},11595:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCommonTypesLibrary_MakeAuthParams-9889ff972f1927cd2d65350a785da1b8.png"},4568:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCommonTypesLibrary_MakeEntitlementParams-ca9ee94871d4bcb815ae1afeadb102d4.png"},18778:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},11520:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},61327:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions-0f1f9aaa5bd0c3c642e84ee61ceff02e.png"},59800:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},68979:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},70504:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},92606:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetExtendedInitializationParameters-b675994c5f5117d737da5f13331ec33b.png"},39644:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetGameEnvironment-9bf28c658ca9a8175e27f7f1e2e6ce09.png"},12376:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetGameId-733baa40804b516d3926e06d2e5ffed7.png"},80807:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetPortal-c2c8e44fc927853c409eda072d2e604b.png"},81294:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},92464:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier-b89970e3d3bb90d64d62b75ce76c02ee.png"},70690:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},81069:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},8796:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},7163:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCreateModLibrary_SetInitialVisibility-ea630d72471876be3f92ec2a7f62411e.png"},56359:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCreateModLibrary_SetInitialVisibility_DEPRECATED-f8acc13f3be4d819e740b51766dd4ed7.png"},11563:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCreateModLibrary_SetMarkAsActiveRelease-d40c905fb2ac942db96dcf0e0982b680.png"},66981:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},37723:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob-46906f25f1b2ce9866b41ebe2954f9f0.png"},34393:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},35536:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},99860:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},6715:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},70153:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},30429:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},12246:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioErrorCodeLibrary_ReconstructError-fd34e6c485430ac5359952f8ad438ffb.png"},20798:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioErrorConditionLibrary_ErrorCodeMatches-0577e7c35976e2a2d2733042ba1b514d.png"},79139:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},57932:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},49633:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},97963:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync-ccc2c7187edf30a0c9472a75052ae921.png"},11778:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioImageLibrary_GetAvatarSize-082b1eeffcd2b743f94a66210f340b0b.png"},84145:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioImageLibrary_GetGallerySize-9626c66c39dbc11535dffd35e279ef87.png"},30484:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioImageLibrary_GetLogoSize-e5d6147cf8343e10fa23bdb08d0b646a.png"},33175:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},9549:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},49510:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},57720:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},28636:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},90164:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},41242:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},62440:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioModProgressInfoLibrary_GetCurrentProgress-cf02a1331822e1a6d164eb0af3fe972f.png"},54858:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},34813:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioModProgressInfoLibrary_GetTotalProgress-c2dfe9ae5100fa60bf076bdd500a7010.png"},91837:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},90998:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},55989:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},53632:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioPlatformHelpersLibrary_GetDefaultAuthProviderForCurrentPlatform-8684a5c74a56ec16d57182102488818f.png"},5635:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioPlatformHelpersLibrary_GetDefaultPortalForCurrentPlatform-2f5206eeda17f3ae9b21f090893f58e9.png"},5617:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},86011:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},84486:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},86318:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},50425:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSDKLibrary_GetLanguageCodeFromString-07f6fa34ac42129bacc95488775b986f.png"},11861:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSDKLibrary_GetLanguageCodeString-47da87bf1ca6d0b3444e4ea1eaac742e.png"},43465:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},75517:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},84733:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},90298:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSDKLibrary_GetProjectInitializeOptionsForSessionId-794e92929faeee6a2fb661456d91d1d7.png"},25047:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSDKLibrary_GetShortenedNumberAsString-4d9bf0258754df026d7a2082964ffba4.png"},52987:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSDKLibrary_GetTimeSpanAsString-b09ea2de257e9f6134e49e2bae984be6.png"},5608:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},8421:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSDKLibrary_IsValidSecurityCodeFormat-f2b25b06b6647d66e1e55a880bf08f59.png"},31773:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},10851:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},33338:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_LoadModFileToMemory-880f5d8c1ba728bf068807cb3559919b.png"},98688:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitModChangesFromMemoryAsync-4edd860ab7540b93959c8c0f7f2a5137.png"},89146:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory-4f094cc73d91d2c82cbefe34bc1c172f.png"},64587:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync-5d1e90118f4f934bf9d50957b51391bb.png"},63754:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_DisableModManagement-5f0eb0a41951b35be46cb68387079953.png"},33590:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_GetLastValidationError-248302bd685ccf133c8b35edafba9bb3.png"},40309:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_IsModManagementBusy-124248b31972e2a2a47932b93466cdda.png"},27331:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_IsUsingBackgroundThread-09942c052a7a41c9f8586bc302807014.png"},9981:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_AddToTempModSet-bdc1c1978e245ba3122b0de58c45c5d0.png"},30771:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_ArchiveModAsync-301820098783a60cf0bb90984e24ea48.png"},36043:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_AuthenticateUserEmailAsync-d8860759b43511919fe6a4dd23b4d16b.png"},69428:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_AuthenticateUserExternalAsync-20730b90c571ce00ba33752eab60040b.png"},73239:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_ClearUserDataAsync-c2ead944b1bcdee2aa8612351796be8d.png"},50203:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_CloseTempModSet-91ea6860470e77c26c197cb71b0b9c3e.png"},58799:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_EnableModManagement-9fdcb415f8dac2cadba8b3ef4619e8ea.png"},38798:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync-959793d6bae5805e609d6948f6071617.png"},10208:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync-75895154b0ad395e68b9055871a93dab.png"},30612:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_ForceUninstallModAsync-f67b8d699d627e8c5788ca293d0194fd.png"},97423:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetDefaultModInstallationDirectory-358e4440ead73240cbcc7e1351ddd81a.png"},87367:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetGameInfoAsync-c33df4da02b4806e818624cea3158637.png"},86039:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetLanguage-825d1af09bb500dac57886a5a78ca64a.png"},68156:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetModCreationHandle-556d4efd18f7c7c76c23437062cac4d2.png"},17784:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetModDependenciesAsync-bb9a17927f1c969c6787d4dd9d42c76e.png"},21913:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetModInfoAsync-c141dbb87248abd169f33ef254251c9f.png"},59484:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetModMediaAvatarAsync-1fdb3c58a27f52670c442f3aa1bbf016.png"},80586:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync-a0630066f06cad861a8769948f32f175.png"},24398:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetModMediaLogoAsync-0a0a1ed024a69d97fcc0a10bdae9b811.png"},57363:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetModTagOptionsAsync-ed35670d985059ef2f2775e597d94fe9.png"},79944:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetMutedUsersAsync-b889b19125d0360cf4d520ade2ab18c5.png"},93492:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetTermsOfUseAsync-c6e22c2b2868122144640004d2394b3b.png"},99183:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetUserDelegationTokenAsync-80a54ec053e5b99bafc5bf071b5fbc97.png"},91533:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetUserMediaAvatarAsync-2a773ca98cccb4b7e926aaf48bc7ff7e.png"},43355:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync-f9b903e0e2833d3380b115b75ab00e07.png"},59125:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_InitTempModSet-29cf256b95cf42b433c622b1cae7f884.png"},1735:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_InitializeAsync-11220774e42901b7ce5d83720c58c28d.png"},1655:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_ListAllModsAsync-316f57d416ba0ae025eb48f00d1c9aa2.png"},25489:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_ListUserCreatedModsAsync-ff9ef2ec2b8111cc4674281c005b4d0e.png"},34935:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_ListUserGamesAsync-3914aa13ec1f36ad56d3058600385d7d.png"},43781:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_MetricsSessionEndAsync-830d93bd55175fb12d6ada839686ad5d.png"},92702:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync-79b62f72079482576f25788aa013f789.png"},91175:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatOnceAsync-5bfbdb4e8f6cb7d8572b81c5dccc84dc.png"},35372:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_MetricsSessionStartAsync-651730c41eec661e03832ebfea99162f.png"},87007:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_MuteUserAsync-95092e29e2790d8e96bc2cdc90d1ffb1.png"},79630:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_PreviewExternalUpdatesAsync-2855ce40346338726cdf2fc3f6d4c10a.png"},21296:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_PurchaseModAsync-1fc3c7edad752491c5d369a742e7d265.png"},16111:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_QueryCurrentModUpdate-ca721cd3920c490f3d8f27a0e87a9721.png"},63699:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_QueryTempModSet-0ba976289a53dbf96b322b103c7afc21.png"},70005:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_QueryUserProfile-290a47c2579a6de4b0e8b4c843dad5cf.png"},13821:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_RefreshUserEntitlementsAsync-0c70260b80c34c71c10bf74061fb4ea2.png"},74399:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_RemoveFromTempModSet-51ca3fcbbd419cf6e470b1f3f159aa84.png"},57110:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_ReportContentAsync-dbaa9fa77a041509aecb7a3e17816909.png"},94989:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_RequestEmailAuthCodeAsync-72567693be0c3a74f8829f100dd7b9d5.png"},40275:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_SetLanguage-64c43dcff334858421a96bdf06ba16f9.png"},85619:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_ShutdownAsync-ad359e71788c08397a38cf61bcb0be00.png"},70604:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_SubmitModChangesAsync-4ad4759be516c2f3d9b45e350c3eaab7.png"},69256:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_SubmitModRatingAsync-a0e171b1e0c563c3cfdf3ad28389564b.png"},8127:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_SubmitNewModAsync-8728a8ce4666e4a8d6477ae0f4a43b03.png"},95258:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_SubmitNewModFileForMod-24f84c6a68b38837f8b0dfd6bf081b98.png"},8274:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_SubscribeToModAsync-40e6cf4f6781ca88098119205d7d7a3f.png"},2124:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_UnmuteUserAsync-5072928585d25488a648dfc7210fb4e0.png"},65298:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync-c749be66fec80d0e8bfe214ade7abbcf.png"},83587:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_K2_VerifyUserAuthenticationAsync-5a26791679686730e9dcc7880e18c978.png"},25607:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_KillBackgroundThread-7568cfd2a189b287b269ceec5db5d079.png"},39846:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_PrioritizeTransferForMod-597a4fd0e4c6a56d4264f3a2df028b25.png"},17253:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_QuerySystemInstallations-dff87f3a0a9c23416e81db47ca0f25da.png"},46391:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_QueryUserInstallations-42ae147ec3c47d4b1f17ba58476bf09e.png"},85256:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_QueryUserPurchasedMods-066afb38d53390a7926561bafc794593.png"},49498:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_QueryUserSubscriptions-73885d17f3df616346f77111ae8b8ad1.png"},51672:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_RunPendingHandlers-50f98a4a74afd398ef05e160d7717e88.png"},23059:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioSubsystem_SetLogLevel-183d6fd9ceeb266aefcf08587b244729.png"},54294:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioUIEnumLocalizationLibrary_FileSizeUnsigned64_ToText-4b00fdbdb167e45ea0acaffd8e54b333.png"},49443:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioUIEnumLocalizationLibrary_GetLocalizedTextForEnumByName-5849fa04c36e80e669f1faae64312cb1.png"},6841:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioUILocalizationLibrary_GetLocalizedTextFromDefaultTableByKey-f07c4cb6bbe86f7285e78f7564f65111.png"},34323:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},13496:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},88015:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},27889:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},75196:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},68337:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},42837:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},99531:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},76134:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},88442:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},90214:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},88754:(e,d,s)=>{s.d(d,{A:()=>i});const i=s.p+"assets/images/nd_img_ModioUnsigned64Library_Percentage_Unsigned64-f7f1acacb3174469af6f03459bea230a.png"},89948:(e,d,s)=>{s.d(d,{A:()=>i});const i=""},28453:(e,d,s)=>{s.d(d,{R:()=>t,x:()=>l});var i=s(96540);const r={},n=i.createContext(r);function t(e){const d=i.useContext(n);return i.useMemo((function(){return"function"==typeof e?e(d):{...d,...e}}),[d,e])}function l(e){let d;return d=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:t(e.components),i.createElement(n.Provider,{value:d},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/2969dc70.6db03ace.js b/Doc/assets/js/2969dc70.6db03ace.js new file mode 100644 index 00000000..b874ead0 --- /dev/null +++ b/Doc/assets/js/2969dc70.6db03ace.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[5237],{34348:(e,d,i)=>{i.r(d),i.d(d,{assets:()=>o,contentTitle:()=>t,default:()=>a,frontMatter:()=>n,metadata:()=>l,toc:()=>c});var s=i(74848),r=i(28453);const n={id:"ue-refdocs",title:"Unreal Plugin API Reference",slug:"/unreal/refdocs/"},t=void 0,l={id:"ue-refdocs",title:"Unreal Plugin API Reference",description:"Classes",source:"@site/public/en-us/generated-refdocs.mdx",sourceDirName:".",slug:"/unreal/refdocs/",permalink:"/unreal/refdocs/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-docs/tree/main/public/en-us/generated-refdocs.mdx",tags:[],version:"current",frontMatter:{id:"ue-refdocs",title:"Unreal Plugin API Reference",slug:"/unreal/refdocs/"},sidebar:"sidebar",previous:{title:"Android Configuration",permalink:"/unreal/android-configuration/"}},o={},c=[{value:"Classes",id:"classes",level:2},{value:"ModioCommonTypesLibrary",id:"modiocommontypeslibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy",level:5},{value:"ModioCreateModLibrary",id:"modiocreatemodlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-1",level:5},{value:"ModioErrorCodeLibrary",id:"modioerrorcodelibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-2",level:5},{value:"ModioErrorConditionLibrary",id:"modioerrorconditionlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-3",level:5},{value:"ModioExampleLibrary",id:"modioexamplelibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-4",level:5},{value:"ModioImageLibrary",id:"modioimagelibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-5",level:5},{value:"ModioModCollectionLibrary",id:"modiomodcollectionlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-6",level:5},{value:"ModioModProgressInfoLibrary",id:"modiomodprogressinfolibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-7",level:5},{value:"ModioModTagOptionsLibrary",id:"modiomodtagoptionslibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-8",level:5},{value:"ModioPlatformHelpersLibrary",id:"modioplatformhelperslibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-9",level:5},{value:"ModioPresetFilterParamsLibrary",id:"modiopresetfilterparamslibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-10",level:5},{value:"ModioSDKLibrary",id:"modiosdklibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-11",level:5},{value:"ModioSubmissionExtensionLibrary",id:"modiosubmissionextensionlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-12",level:5},{value:"SubmitModChangesAsync",id:"submitmodchangesasync",level:4},{value:"Requirements",id:"requirements",level:5},{value:"Parameters",id:"parameters",level:5},{value:"Error Values",id:"error-values",level:5},{value:"ModioSubsystem",id:"modiosubsystem",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-13",level:5},{value:"Set Log Level",id:"set-log-level",level:4},{value:"Parameters",id:"parameters-1",level:5},{value:"Run Pending Handlers",id:"run-pending-handlers",level:4},{value:"Parameters",id:"parameters-2",level:5},{value:"Query User Subscriptions",id:"query-user-subscriptions",level:4},{value:"Parameters",id:"parameters-3",level:5},{value:"Returns",id:"returns",level:5},{value:"Query User Purchased Mods",id:"query-user-purchased-mods",level:4},{value:"Requirements",id:"requirements-1",level:5},{value:"Parameters",id:"parameters-4",level:5},{value:"Returns",id:"returns-1",level:5},{value:"Query User Installations",id:"query-user-installations",level:4},{value:"Parameters",id:"parameters-5",level:5},{value:"Returns",id:"returns-2",level:5},{value:"Query System Installations",id:"query-system-installations",level:4},{value:"Parameters",id:"parameters-6",level:5},{value:"Returns",id:"returns-3",level:5},{value:"Prioritize Transfer for Mod",id:"prioritize-transfer-for-mod",level:4},{value:"Requirements",id:"requirements-2",level:5},{value:"Parameters",id:"parameters-7",level:5},{value:"Returns",id:"returns-4",level:5},{value:"Error Values",id:"error-values-1",level:5},{value:"Kill Background Thread",id:"kill-background-thread",level:4},{value:"VerifyUserAuthenticationAsync",id:"verifyuserauthenticationasync",level:4},{value:"Requirements",id:"requirements-3",level:5},{value:"Parameters",id:"parameters-8",level:5},{value:"Error Values",id:"error-values-2",level:5},{value:"UnsubscribeFromModAsync",id:"unsubscribefrommodasync",level:4},{value:"Requirements",id:"requirements-4",level:5},{value:"Parameters",id:"parameters-9",level:5},{value:"Error Values",id:"error-values-3",level:5},{value:"UnmuteUserAsync",id:"unmuteuserasync",level:4},{value:"Requirements",id:"requirements-5",level:5},{value:"Parameters",id:"parameters-10",level:5},{value:"Error Values",id:"error-values-4",level:5},{value:"SubscribeToModAsync",id:"subscribetomodasync",level:4},{value:"Requirements",id:"requirements-6",level:5},{value:"Parameters",id:"parameters-11",level:5},{value:"Error Values",id:"error-values-5",level:5},{value:"SubmitNewModFileForMod",id:"submitnewmodfileformod",level:4},{value:"Requirements",id:"requirements-7",level:5},{value:"Parameters",id:"parameters-12",level:5},{value:"Error Values",id:"error-values-6",level:5},{value:"SubmitNewModAsync",id:"submitnewmodasync",level:4},{value:"Requirements",id:"requirements-8",level:5},{value:"Parameters",id:"parameters-13",level:5},{value:"Error Values",id:"error-values-7",level:5},{value:"SubmitModRatingAsync",id:"submitmodratingasync",level:4},{value:"Requirements",id:"requirements-9",level:5},{value:"Parameters",id:"parameters-14",level:5},{value:"Error Values",id:"error-values-8",level:5},{value:"SubmitModChangesAsync",id:"submitmodchangesasync-1",level:4},{value:"Requirements",id:"requirements-10",level:5},{value:"Parameters",id:"parameters-15",level:5},{value:"Error Values",id:"error-values-9",level:5},{value:"ShutdownAsync",id:"shutdownasync",level:4},{value:"Parameters",id:"parameters-16",level:5},{value:"SetLanguage",id:"setlanguage",level:4},{value:"Parameters",id:"parameters-17",level:5},{value:"RequestEmailAuthCodeAsync",id:"requestemailauthcodeasync",level:4},{value:"Requirements",id:"requirements-11",level:5},{value:"Parameters",id:"parameters-18",level:5},{value:"Error Values",id:"error-values-10",level:5},{value:"ReportContentAsync",id:"reportcontentasync",level:4},{value:"Requirements",id:"requirements-12",level:5},{value:"Parameters",id:"parameters-19",level:5},{value:"Error Values",id:"error-values-11",level:5},{value:"RemoveFromTempModSet",id:"removefromtempmodset",level:4},{value:"Requirements",id:"requirements-13",level:5},{value:"Parameters",id:"parameters-20",level:5},{value:"Returns",id:"returns-5",level:5},{value:"Error Values",id:"error-values-12",level:5},{value:"RefreshUserEntitlementsAsync",id:"refreshuserentitlementsasync",level:4},{value:"Requirements",id:"requirements-14",level:5},{value:"Parameters",id:"parameters-21",level:5},{value:"QueryUserProfile",id:"queryuserprofile",level:4},{value:"Parameters",id:"parameters-22",level:5},{value:"Returns",id:"returns-6",level:5},{value:"QueryTempModSet",id:"querytempmodset",level:4},{value:"Parameters",id:"parameters-23",level:5},{value:"Returns",id:"returns-7",level:5},{value:"QueryCurrentModUpdate",id:"querycurrentmodupdate",level:4},{value:"Parameters",id:"parameters-24",level:5},{value:"Returns",id:"returns-8",level:5},{value:"PurchaseModAsync",id:"purchasemodasync",level:4},{value:"Requirements",id:"requirements-15",level:5},{value:"Parameters",id:"parameters-25",level:5},{value:"PreviewExternalUpdatesAsync",id:"previewexternalupdatesasync",level:4},{value:"Parameters",id:"parameters-26",level:5},{value:"MuteUserAsync",id:"muteuserasync",level:4},{value:"Requirements",id:"requirements-16",level:5},{value:"Parameters",id:"parameters-27",level:5},{value:"Error Values",id:"error-values-13",level:5},{value:"MetricsSessionStartAsync",id:"metricssessionstartasync",level:4},{value:"Parameters",id:"parameters-28",level:5},{value:"Error Values",id:"error-values-14",level:5},{value:"MetricsSessionSendHeartbeatOnceAsync",id:"metricssessionsendheartbeatonceasync",level:4},{value:"Parameters",id:"parameters-29",level:5},{value:"Error Values",id:"error-values-15",level:5},{value:"MetricsSessionSendHeartbeatAtIntervalAsync",id:"metricssessionsendheartbeatatintervalasync",level:4},{value:"Parameters",id:"parameters-30",level:5},{value:"Error Values",id:"error-values-16",level:5},{value:"MetricsSessionEndAsync",id:"metricssessionendasync",level:4},{value:"Parameters",id:"parameters-31",level:5},{value:"Error Values",id:"error-values-17",level:5},{value:"ListUserGamesAsync",id:"listusergamesasync",level:4},{value:"Requirements",id:"requirements-17",level:5},{value:"Parameters",id:"parameters-32",level:5},{value:"Error Values",id:"error-values-18",level:5},{value:"ListUserCreatedModsAsync",id:"listusercreatedmodsasync",level:4},{value:"Requirements",id:"requirements-18",level:5},{value:"Parameters",id:"parameters-33",level:5},{value:"Error Values",id:"error-values-19",level:5},{value:"ListAllModsAsync",id:"listallmodsasync",level:4},{value:"Requirements",id:"requirements-19",level:5},{value:"Parameters",id:"parameters-34",level:5},{value:"Error Values",id:"error-values-20",level:5},{value:"InitTempModSet",id:"inittempmodset",level:4},{value:"Requirements",id:"requirements-20",level:5},{value:"Parameters",id:"parameters-35",level:5},{value:"Returns",id:"returns-9",level:5},{value:"Error Values",id:"error-values-21",level:5},{value:"InitializeAsync",id:"initializeasync",level:4},{value:"Parameters",id:"parameters-36",level:5},{value:"Error Values",id:"error-values-22",level:5},{value:"GetUserWalletBalanceAsync",id:"getuserwalletbalanceasync",level:4},{value:"Requirements",id:"requirements-21",level:5},{value:"Parameters",id:"parameters-37",level:5},{value:"GetUserMediaAsync (Avatar)",id:"getusermediaasync-avatar",level:4},{value:"Requirements",id:"requirements-22",level:5},{value:"Parameters",id:"parameters-38",level:5},{value:"Error Values",id:"error-values-23",level:5},{value:"GetUserDelegationTokenAsync",id:"getuserdelegationtokenasync",level:4},{value:"Requirements",id:"requirements-23",level:5},{value:"Parameters",id:"parameters-39",level:5},{value:"GetTermsOfUseAsync",id:"gettermsofuseasync",level:4},{value:"Requirements",id:"requirements-24",level:5},{value:"Parameters",id:"parameters-40",level:5},{value:"Error Values",id:"error-values-24",level:5},{value:"GetMutedUsersAsync",id:"getmutedusersasync",level:4},{value:"Requirements",id:"requirements-25",level:5},{value:"Parameters",id:"parameters-41",level:5},{value:"Error Values",id:"error-values-25",level:5},{value:"GetModTagOptionsAsync",id:"getmodtagoptionsasync",level:4},{value:"Requirements",id:"requirements-26",level:5},{value:"Parameters",id:"parameters-42",level:5},{value:"Error Values",id:"error-values-26",level:5},{value:"GetModMediaAsync (Logo)",id:"getmodmediaasync-logo",level:4},{value:"Requirements",id:"requirements-27",level:5},{value:"Parameters",id:"parameters-43",level:5},{value:"Error Values",id:"error-values-27",level:5},{value:"GetModMediaAsync (Gallery Image)",id:"getmodmediaasync-gallery-image",level:4},{value:"Requirements",id:"requirements-28",level:5},{value:"Parameters",id:"parameters-44",level:5},{value:"Error Values",id:"error-values-28",level:5},{value:"GetModMediaAsync (Avatar)",id:"getmodmediaasync-avatar",level:4},{value:"Requirements",id:"requirements-29",level:5},{value:"Parameters",id:"parameters-45",level:5},{value:"Error Values",id:"error-values-29",level:5},{value:"GetModInfoAsync",id:"getmodinfoasync",level:4},{value:"Requirements",id:"requirements-30",level:5},{value:"Parameters",id:"parameters-46",level:5},{value:"Error Values",id:"error-values-30",level:5},{value:"GetModDependenciesAsync",id:"getmoddependenciesasync",level:4},{value:"Requirements",id:"requirements-31",level:5},{value:"Parameters",id:"parameters-47",level:5},{value:"Error Values",id:"error-values-31",level:5},{value:"GetModCreationHandle",id:"getmodcreationhandle",level:4},{value:"Parameters",id:"parameters-48",level:5},{value:"GetLanguage",id:"getlanguage",level:4},{value:"Parameters",id:"parameters-49",level:5},{value:"Returns",id:"returns-10",level:5},{value:"GetGameInfoAsync",id:"getgameinfoasync",level:4},{value:"Requirements",id:"requirements-32",level:5},{value:"Parameters",id:"parameters-50",level:5},{value:"Error Values",id:"error-values-32",level:5},{value:"ForceUninstallModAsync",id:"forceuninstallmodasync",level:4},{value:"Parameters",id:"parameters-51",level:5},{value:"Error Values",id:"error-values-33",level:5},{value:"FetchUserPurchasesAsync",id:"fetchuserpurchasesasync",level:4},{value:"Requirements",id:"requirements-33",level:5},{value:"Parameters",id:"parameters-52",level:5},{value:"FetchExternalUpdatesAsync",id:"fetchexternalupdatesasync",level:4},{value:"Parameters",id:"parameters-53",level:5},{value:"EnableModManagement",id:"enablemodmanagement",level:4},{value:"Parameters",id:"parameters-54",level:5},{value:"Returns",id:"returns-11",level:5},{value:"Error Values",id:"error-values-34",level:5},{value:"CloseTempModSet",id:"closetempmodset",level:4},{value:"Requirements",id:"requirements-34",level:5},{value:"Parameters",id:"parameters-55",level:5},{value:"Returns",id:"returns-12",level:5},{value:"Error Values",id:"error-values-35",level:5},{value:"ClearUserDataAsync",id:"clearuserdataasync",level:4},{value:"Requirements",id:"requirements-35",level:5},{value:"Parameters",id:"parameters-56",level:5},{value:"Error Values",id:"error-values-36",level:5},{value:"AuthenticateUserExternalAsync",id:"authenticateuserexternalasync",level:4},{value:"Requirements",id:"requirements-36",level:5},{value:"Parameters",id:"parameters-57",level:5},{value:"Error Values",id:"error-values-37",level:5},{value:"AuthenticateUserEmailAsync",id:"authenticateuseremailasync",level:4},{value:"Requirements",id:"requirements-37",level:5},{value:"Parameters",id:"parameters-58",level:5},{value:"Error Values",id:"error-values-38",level:5},{value:"ArchiveModAsync",id:"archivemodasync",level:4},{value:"Requirements",id:"requirements-38",level:5},{value:"Parameters",id:"parameters-59",level:5},{value:"Error Values",id:"error-values-39",level:5},{value:"AddToTempModSet",id:"addtotempmodset",level:4},{value:"Requirements",id:"requirements-39",level:5},{value:"Parameters",id:"parameters-60",level:5},{value:"Returns",id:"returns-13",level:5},{value:"Error Values",id:"error-values-40",level:5},{value:"Is Using Background Thread",id:"is-using-background-thread",level:4},{value:"Is Mod Management Busy",id:"is-mod-management-busy",level:4},{value:"Parameters",id:"parameters-61",level:5},{value:"Returns",id:"returns-14",level:5},{value:"Get Last Validation Error",id:"get-last-validation-error",level:4},{value:"Requirements",id:"requirements-40",level:5},{value:"Parameters",id:"parameters-62",level:5},{value:"Returns",id:"returns-15",level:5},{value:"Disable Mod Management",id:"disable-mod-management",level:4},{value:"Parameters",id:"parameters-63",level:5},{value:"ModioTokenPackLibrary",id:"modiotokenpacklibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-14",level:5},{value:"ModioUIEnumLocalizationLibrary",id:"modiouienumlocalizationlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-15",level:5},{value:"ModioUILocalizationLibrary",id:"modiouilocalizationlibrary",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-16",level:5},{value:"ModioUnsigned64Library",id:"modiounsigned64library",level:3},{value:"Inheritance Hierarchy",id:"inheritance-hierarchy-17",level:5},{value:"Structs",id:"structs",level:2},{value:"ModioGameInfoList",id:"modiogameinfolist",level:3},{value:"ModioGameInfo",id:"modiogameinfo",level:3},{value:"Variables",id:"variables",level:5},{value:"ModioModTagInfo",id:"modiomodtaginfo",level:3},{value:"Variables",id:"variables-1",level:5},{value:"ModioLocalizedTagCategory",id:"modiolocalizedtagcategory",level:3},{value:"Variables",id:"variables-2",level:5},{value:"ModioModTagLocalizationData",id:"modiomodtaglocalizationdata",level:3},{value:"Variables",id:"variables-3",level:5},{value:"ModioGamePlatform",id:"modiogameplatform",level:3},{value:"Variables",id:"variables-4",level:5},{value:"ModioOtherUrl",id:"modiootherurl",level:3},{value:"Variables",id:"variables-5",level:5},{value:"ModioGameStats",id:"modiogamestats",level:3},{value:"Variables",id:"variables-6",level:5},{value:"ModioGameID",id:"modiogameid",level:3},{value:"ModioTheme",id:"modiotheme",level:3},{value:"Variables",id:"variables-7",level:5},{value:"ModioHeaderImage",id:"modioheaderimage",level:3},{value:"Variables",id:"variables-8",level:5},{value:"ModioLogo",id:"modiologo",level:3},{value:"Variables",id:"variables-9",level:5},{value:"ModioIcon",id:"modioicon",level:3},{value:"Variables",id:"variables-10",level:5},{value:"ModioPagedResult",id:"modiopagedresult",level:3},{value:"Variables",id:"variables-11",level:5},{value:"ModioModInfoList",id:"modiomodinfolist",level:3},{value:"ModioModInfo",id:"modiomodinfo",level:3},{value:"Variables",id:"variables-12",level:5},{value:"ModioUnsigned64",id:"modiounsigned64",level:3},{value:"ModioModStats",id:"modiomodstats",level:3},{value:"Variables",id:"variables-13",level:5},{value:"ModioSketchfabURLList",id:"modiosketchfaburllist",level:3},{value:"ModioYoutubeURLList",id:"modioyoutubeurllist",level:3},{value:"ModioModTag",id:"modiomodtag",level:3},{value:"Variables",id:"variables-14",level:5},{value:"ModioMetadata",id:"modiometadata",level:3},{value:"Variables",id:"variables-15",level:5},{value:"ModioFileMetadata",id:"modiofilemetadata",level:3},{value:"Variables",id:"variables-16",level:5},{value:"ModioModID",id:"modiomodid",level:3},{value:"ModioFileMetadataID",id:"modiofilemetadataid",level:3},{value:"ModioUser",id:"modiouser",level:3},{value:"Variables",id:"variables-17",level:5},{value:"ModioUserID",id:"modiouserid",level:3},{value:"ModioModTagOptions",id:"modiomodtagoptions",level:3},{value:"ModioTokenPackList",id:"modiotokenpacklist",level:3},{value:"ModioTokenPack",id:"modiotokenpack",level:3},{value:"ModioErrorCode",id:"modioerrorcode",level:3},{value:"ModioOptionalGameInfo",id:"modiooptionalgameinfo",level:3},{value:"ModioOptionalImage",id:"modiooptionalimage",level:3},{value:"ModioOptionalModDependencyList",id:"modiooptionalmoddependencylist",level:3},{value:"ModioOptionalModInfo",id:"modiooptionalmodinfo",level:3},{value:"ModioOptionalModTagOptions",id:"modiooptionalmodtagoptions",level:3},{value:"ModioOptionalTerms",id:"modiooptionalterms",level:3},{value:"ModioOptionalUInt64",id:"modiooptionaluint64",level:3},{value:"ModioOptionalModInfoList",id:"modiooptionalmodinfolist",level:3},{value:"ModioOptionalGameInfoList",id:"modiooptionalgameinfolist",level:3},{value:"ModioModManagementEvent",id:"modiomodmanagementevent",level:3},{value:"Variables",id:"variables-18",level:5},{value:"ModioOptionalUserList",id:"modiooptionaluserlist",level:3},{value:"ModioOptionalModChangeMap",id:"modiooptionalmodchangemap",level:3},{value:"ModioOptionalTransactionRecord",id:"modiooptionaltransactionrecord",level:3},{value:"ModioOptionalEntitlementConsumptionStatusList",id:"modiooptionalentitlementconsumptionstatuslist",level:3},{value:"ModioOptionalModID",id:"modiooptionalmodid",level:3},{value:"ModioAuthenticationParams",id:"modioauthenticationparams",level:3},{value:"Variables",id:"variables-19",level:5},{value:"ModioApiKey",id:"modioapikey",level:3},{value:"ModioGuid",id:"modioguid",level:3},{value:"ModioOptionalGuid",id:"modiooptionalguid",level:3},{value:"ModioEmailAddress",id:"modioemailaddress",level:3},{value:"ModioEmailAuthCode",id:"modioemailauthcode",level:3},{value:"ModioEntitlementParams",id:"modioentitlementparams",level:3},{value:"ModioMetricsSessionParams",id:"modiometricssessionparams",level:3},{value:"ModioTokenPackID",id:"modiotokenpackid",level:3},{value:"ModioCreateModFileParams",id:"modiocreatemodfileparams",level:3},{value:"Variables",id:"variables-20",level:5},{value:"ModioCreateModParams",id:"modiocreatemodparams",level:3},{value:"Variables",id:"variables-21",level:5},{value:"ModioEditModParams",id:"modioeditmodparams",level:3},{value:"ModioEntitlementWalletBalance",id:"modioentitlementwalletbalance",level:3},{value:"Variables",id:"variables-22",level:5},{value:"ModioOptionalEntitlementWalletBalance",id:"modiooptionalentitlementwalletbalance",level:3},{value:"ModioEntitlementConsumptionVirtualCurrencyDetails",id:"modioentitlementconsumptionvirtualcurrencydetails",level:3},{value:"Variables",id:"variables-23",level:5},{value:"EntitlementConsumptionStatus",id:"entitlementconsumptionstatus",level:3},{value:"Variables",id:"variables-24",level:5},{value:"ModioEntitlementConsumptionStatusList",id:"modioentitlementconsumptionstatuslist",level:3},{value:"Variables",id:"variables-25",level:5},{value:"ModioFilterParams",id:"modiofilterparams",level:3},{value:"ModioPresetFilterParams",id:"modiopresetfilterparams",level:3},{value:"Variables",id:"variables-26",level:5},{value:"ModioImageWrapper",id:"modioimagewrapper",level:3},{value:"Variables",id:"variables-27",level:5},{value:"ModioInitializeOptions",id:"modioinitializeoptions",level:3},{value:"Variables",id:"variables-28",level:5},{value:"ModioModChangeMap",id:"modiomodchangemap",level:3},{value:"Variables",id:"variables-29",level:5},{value:"ModioModCollectionEntry",id:"modiomodcollectionentry",level:3},{value:"ModioModCreationHandle",id:"modiomodcreationhandle",level:3},{value:"ModioModDependency",id:"modiomoddependency",level:3},{value:"Variables",id:"variables-30",level:5},{value:"ModioModDependencyList",id:"modiomoddependencylist",level:3},{value:"Variables",id:"variables-31",level:5},{value:"ModioModProgressInfo",id:"modiomodprogressinfo",level:3},{value:"Variables",id:"variables-32",level:5},{value:"ModioOptionalModProgressInfo",id:"modiooptionalmodprogressinfo",level:3},{value:"ModioReportParams",id:"modioreportparams",level:3},{value:"ModioLink",id:"modiolink",level:3},{value:"Variables",id:"variables-33",level:5},{value:"ModioTerms",id:"modioterms",level:3},{value:"Variables",id:"variables-34",level:5},{value:"ModioOptionalTokenPack",id:"modiooptionaltokenpack",level:3},{value:"ModioOptionalTokenPackList",id:"modiooptionaltokenpacklist",level:3},{value:"ModioTransactionRecord",id:"modiotransactionrecord",level:3},{value:"Variables",id:"variables-35",level:5},{value:"ModioOptionalUser",id:"modiooptionaluser",level:3},{value:"ModioUserList",id:"modiouserlist",level:3},{value:"Variables",id:"variables-36",level:5},{value:"ModioValidationError",id:"modiovalidationerror",level:3},{value:"Variables",id:"variables-37",level:5},{value:"ModioCreateModFileMemoryParams",id:"modiocreatemodfilememoryparams",level:3},{value:"Variables",id:"variables-38",level:5},{value:"Functions",id:"functions",level:2},{value:"Set Session Identifier",id:"set-session-identifier",level:3},{value:"Parameters",id:"parameters-64",level:4},{value:"Returns",id:"returns-16",level:4},{value:"Set Session Id",id:"set-session-id",level:3},{value:"Parameters",id:"parameters-65",level:4},{value:"Set Portal",id:"set-portal",level:3},{value:"Parameters",id:"parameters-66",level:4},{value:"Returns",id:"returns-17",level:4},{value:"Set Game Id",id:"set-game-id",level:3},{value:"Parameters",id:"parameters-67",level:4},{value:"Returns",id:"returns-18",level:4},{value:"Set Game Environment",id:"set-game-environment",level:3},{value:"Parameters",id:"parameters-68",level:4},{value:"Returns",id:"returns-19",level:4},{value:"Set Extended Initialization Parameters",id:"set-extended-initialization-parameters",level:3},{value:"Parameters",id:"parameters-69",level:4},{value:"Returns",id:"returns-20",level:4},{value:"Set Background Thread",id:"set-background-thread",level:3},{value:"Parameters",id:"parameters-70",level:4},{value:"Returns",id:"returns-21",level:4},{value:"Set API Key",id:"set-api-key",level:3},{value:"Parameters",id:"parameters-71",level:4},{value:"Returns",id:"returns-22",level:4},{value:"ModioModID != ModioModID",id:"modiomodid--modiomodid",level:3},{value:"Parameters",id:"parameters-72",level:4},{value:"Make Metrics Session Params",id:"make-metrics-session-params",level:3},{value:"Parameters",id:"parameters-73",level:4},{value:"Make Initialize Options",id:"make-initialize-options",level:3},{value:"Parameters",id:"parameters-74",level:4},{value:"Make Guid",id:"make-guid",level:3},{value:"Parameters",id:"parameters-75",level:4},{value:"Make Game Id",id:"make-game-id",level:3},{value:"Parameters",id:"parameters-76",level:4},{value:"Make Entitlement Params",id:"make-entitlement-params",level:3},{value:"Parameters",id:"parameters-77",level:4},{value:"Make Auth Params",id:"make-auth-params",level:3},{value:"Parameters",id:"parameters-78",level:4},{value:"Returns",id:"returns-23",level:4},{value:"Make Api Key",id:"make-api-key",level:3},{value:"Parameters",id:"parameters-79",level:4},{value:"Get Raw Value from Mod ID",id:"get-raw-value-from-mod-id",level:3},{value:"Parameters",id:"parameters-80",level:4},{value:"Returns",id:"returns-24",level:4},{value:"ModioModID == ModioModID",id:"modiomodid--modiomodid-1",level:3},{value:"Parameters",id:"parameters-81",level:4},{value:"Set Version String",id:"set-version-string",level:3},{value:"Set Tags",id:"set-tags",level:3},{value:"Set Modfile Platforms",id:"set-modfile-platforms",level:3},{value:"Set Mod File Metadata Blob",id:"set-mod-file-metadata-blob",level:3},{value:"Set Metadata Blob",id:"set-metadata-blob",level:3},{value:"Set Mark as Active Release",id:"set-mark-as-active-release",level:3},{value:"Set Initial Visibility DEPRECATED",id:"set-initial-visibility-deprecated",level:3},{value:"Set Initial Visibility",id:"set-initial-visibility",level:3},{value:"Set Homepage URL",id:"set-homepage-url",level:3},{value:"Set Description",id:"set-description",level:3},{value:"Set Changelog String",id:"set-changelog-string",level:3},{value:"Get Localized Text for Enum by Name",id:"get-localized-text-for-enum-by-name",level:3},{value:"Parameters",id:"parameters-82",level:4},{value:"Returns",id:"returns-25",level:4},{value:"FileSizeToText (Unsigned64)",id:"filesizetotext-unsigned64",level:3},{value:"Parameters",id:"parameters-83",level:4},{value:"Returns",id:"returns-26",level:4},{value:"Get Localized Text from Default Table by Key",id:"get-localized-text-from-default-table-by-key",level:3},{value:"Parameters",id:"parameters-84",level:4},{value:"Returns",id:"returns-27",level:4},{value:"Reconstruct Error",id:"reconstruct-error",level:3},{value:"Parameters",id:"parameters-85",level:4},{value:"IsError",id:"iserror",level:3},{value:"Parameters",id:"parameters-86",level:4},{value:"Returns",id:"returns-28",level:4},{value:"Get Value",id:"get-value",level:3},{value:"Parameters",id:"parameters-87",level:4},{value:"Returns",id:"returns-29",level:4},{value:"Get Message",id:"get-message",level:3},{value:"Parameters",id:"parameters-88",level:4},{value:"Returns",id:"returns-30",level:4},{value:"Error Code Matches",id:"error-code-matches",level:3},{value:"Parameters",id:"parameters-89",level:4},{value:"Returns",id:"returns-31",level:4},{value:"List User Subscription Async",id:"list-user-subscription-async",level:3},{value:"Parameters",id:"parameters-90",level:4},{value:"Get Logo Thumbnail Size",id:"get-logo-thumbnail-size",level:3},{value:"Get Logo Full Size",id:"get-logo-full-size",level:3},{value:"Get Avatar Thumbnail Size",id:"get-avatar-thumbnail-size",level:3},{value:"To Filter Params",id:"to-filter-params",level:3},{value:"Parameters",id:"parameters-91",level:4},{value:"Returns",id:"returns-32",level:4},{value:"Load Async",id:"load-async",level:3},{value:"Get Texture",id:"get-texture",level:3},{value:"Get State",id:"get-state",level:3},{value:"Get Logo Size",id:"get-logo-size",level:3},{value:"Parameters",id:"parameters-92",level:4},{value:"Returns",id:"returns-33",level:4},{value:"Get Gallery Size",id:"get-gallery-size",level:3},{value:"Parameters",id:"parameters-93",level:4},{value:"Returns",id:"returns-34",level:4},{value:"Get Avatar Size",id:"get-avatar-size",level:3},{value:"Parameters",id:"parameters-94",level:4},{value:"Returns",id:"returns-35",level:4},{value:"Get Path",id:"get-path",level:3},{value:"Parameters",id:"parameters-95",level:4},{value:"Returns",id:"returns-36",level:4},{value:"Get Mod State",id:"get-mod-state",level:3},{value:"Parameters",id:"parameters-96",level:4},{value:"Returns",id:"returns-37",level:4},{value:"Get Mod Profile",id:"get-mod-profile",level:3},{value:"Parameters",id:"parameters-97",level:4},{value:"Returns",id:"returns-38",level:4},{value:"Get ID",id:"get-id",level:3},{value:"Parameters",id:"parameters-98",level:4},{value:"Returns",id:"returns-39",level:4},{value:"Get Total Progress",id:"get-total-progress",level:3},{value:"Parameters",id:"parameters-99",level:4},{value:"Returns",id:"returns-40",level:4},{value:"Get Current State",id:"get-current-state",level:3},{value:"Parameters",id:"parameters-100",level:4},{value:"Get Current Progress",id:"get-current-progress",level:3},{value:"Parameters",id:"parameters-101",level:4},{value:"Returns",id:"returns-41",level:4},{value:"Get Tags",id:"get-tags",level:3},{value:"Parameters",id:"parameters-102",level:4},{value:"Returns",id:"returns-42",level:4},{value:"Get Paged Result",id:"get-paged-result",level:3},{value:"Parameters",id:"parameters-103",level:4},{value:"Returns",id:"returns-43",level:4},{value:"Get Default Portal for Current Platform",id:"get-default-portal-for-current-platform",level:3},{value:"Parameters",id:"parameters-104",level:4},{value:"Returns",id:"returns-44",level:4},{value:"Get Default Auth Provider for Current Platform",id:"get-default-auth-provider-for-current-platform",level:3},{value:"Parameters",id:"parameters-105",level:4},{value:"Returns",id:"returns-45",level:4},{value:"Get Current Platform",id:"get-current-platform",level:3},{value:"Parameters",id:"parameters-106",level:4},{value:"Returns",id:"returns-46",level:4},{value:"Round Number String",id:"round-number-string",level:3},{value:"Parameters",id:"parameters-107",level:4},{value:"Get Percent (integer64/integer64)",id:"get-percent-integer64integer64",level:3},{value:"Parameters",id:"parameters-108",level:4},{value:"Returns",id:"returns-47",level:4},{value:"Is Valid Security Code Format",id:"is-valid-security-code-format",level:3},{value:"Parameters",id:"parameters-109",level:4},{value:"Returns",id:"returns-48",level:4},{value:"Is Valid Email Address Format",id:"is-valid-email-address-format",level:3},{value:"Parameters",id:"parameters-110",level:4},{value:"Returns",id:"returns-49",level:4},{value:"Get Time Span as String",id:"get-time-span-as-string",level:3},{value:"Parameters",id:"parameters-111",level:4},{value:"Get Shortened Number as String",id:"get-shortened-number-as-string",level:3},{value:"Parameters",id:"parameters-112",level:4},{value:"Get Project Initialize Options for Session Id",id:"get-project-initialize-options-for-session-id",level:3},{value:"Parameters",id:"parameters-113",level:4},{value:"Get Project Game Id",id:"get-project-game-id",level:3},{value:"Parameters",id:"parameters-114",level:4},{value:"Get Project Environment",id:"get-project-environment",level:3},{value:"Parameters",id:"parameters-115",level:4},{value:"Get Project Api Key",id:"get-project-api-key",level:3},{value:"Parameters",id:"parameters-116",level:4},{value:"Get Monetization Purchase Category",id:"get-monetization-purchase-category",level:3},{value:"Parameters",id:"parameters-117",level:4},{value:"Returns",id:"returns-50",level:4},{value:"Get Language Code String",id:"get-language-code-string",level:3},{value:"Parameters",id:"parameters-118",level:4},{value:"Returns",id:"returns-51",level:4},{value:"Get Language Code from String",id:"get-language-code-from-string",level:3},{value:"Parameters",id:"parameters-119",level:4},{value:"Returns",id:"returns-52",level:4},{value:"Get Desired File Size Unit",id:"get-desired-file-size-unit",level:3},{value:"Parameters",id:"parameters-120",level:4},{value:"Returns",id:"returns-53",level:4},{value:"Get Default Session Id Windows",id:"get-default-session-id-windows",level:3},{value:"Parameters",id:"parameters-121",level:4},{value:"Returns",id:"returns-54",level:4},{value:"ToString (Filesize)",id:"tostring-filesize",level:3},{value:"Parameters",id:"parameters-122",level:4},{value:"Returns",id:"returns-55",level:4},{value:"GetDefaultModInstallationDirectory",id:"getdefaultmodinstallationdirectory",level:3},{value:"Parameters",id:"parameters-123",level:4},{value:"Returns",id:"returns-56",level:4},{value:"Is Purchasable",id:"is-purchasable",level:3},{value:"Get Title",id:"get-title",level:3},{value:"Get Regular Price",id:"get-regular-price",level:3},{value:"Get Numeric Price",id:"get-numeric-price",level:3},{value:"Get Modio Id",id:"get-modio-id",level:3},{value:"Get Long Description",id:"get-long-description",level:3},{value:"Get Id",id:"get-id-1",level:3},{value:"Get Fields",id:"get-fields",level:3},{value:"Get Display Price",id:"get-display-price",level:3},{value:"Get Description",id:"get-description",level:3},{value:"ModioUnsigned64 - ModioUnsigned64",id:"modiounsigned64---modiounsigned64",level:3},{value:"Percentage Unsigned 64",id:"percentage-unsigned-64",level:3},{value:"ModioUnsigned64 != ModioUnsigned64",id:"modiounsigned64--modiounsigned64",level:3},{value:"Make from Components",id:"make-from-components",level:3},{value:"ModioUnsigned64 < ModioUnsigned64",id:"modiounsigned64--modiounsigned64-1",level:3},{value:"ModioUnsigned64 > 0",id:"modiounsigned64--0",level:3},{value:"ModioUnsigned64 > ModioUnsigned64",id:"modiounsigned64--modiounsigned64-2",level:3},{value:"ModioUnsigned64 == ModioUnsigned64",id:"modiounsigned64--modiounsigned64-3",level:3},{value:"ModioUnsigned64 / ModioUnsigned64",id:"modiounsigned64--modiounsigned64-4",level:3},{value:"ModioUnsigned64 / float",id:"modiounsigned64--float",level:3},{value:"ModioUnsigned64 / ModioUnsigned64 (truncate)",id:"modiounsigned64--modiounsigned64-truncate",level:3},{value:"Break to Components",id:"break-to-components",level:3},{value:"ModioUnsigned64 + ModioUnsigned64",id:"modiounsigned64--modiounsigned64-5",level:3},{value:"SubmitNewModFromMemoryAsync",id:"submitnewmodfrommemoryasync",level:3},{value:"Parameters",id:"parameters-124",level:4},{value:"SubmitNewModFileForModFromMemory",id:"submitnewmodfileformodfrommemory",level:3},{value:"Requirements",id:"requirements-41",level:4},{value:"Parameters",id:"parameters-125",level:4},{value:"LoadModFileToMemory",id:"loadmodfiletomemory",level:3},{value:"Requirements",id:"requirements-42",level:4},{value:"Parameters",id:"parameters-126",level:4},{value:"Enums",id:"enums",level:2},{value:"EModioModfilePlatform",id:"EModioModfilePlatform",level:3},{value:"EGameMaturityFlags",id:"EGameMaturityFlags",level:3},{value:"EGameMonetizationFlags",id:"EGameMonetizationFlags",level:3},{value:"EModioModServerSideStatus",id:"EModioModServerSideStatus",level:3},{value:"EModioVirusStatus",id:"EModioVirusStatus",level:3},{value:"EModioVirusScanStatus",id:"EModioVirusScanStatus",level:3},{value:"EModioObjectVisibilityFlags",id:"EModioObjectVisibilityFlags",level:3},{value:"EModioMaturityFlags",id:"EModioMaturityFlags",level:3},{value:"EModioModManagementEventType",id:"EModioModManagementEventType",level:3},{value:"EModioAuthenticationProvider",id:"EModioAuthenticationProvider",level:3},{value:"EModioEnvironment",id:"EModioEnvironment",level:3},{value:"EModioPortal",id:"EModioPortal",level:3},{value:"EModioPlatformName",id:"EModioPlatformName",level:3},{value:"EModioLogoSize",id:"EModioLogoSize",level:3},{value:"EModioAvatarSize",id:"EModioAvatarSize",level:3},{value:"EModioGallerySize",id:"EModioGallerySize",level:3},{value:"EModioLogLevel",id:"EModioLogLevel",level:3},{value:"EModioLanguage",id:"EModioLanguage",level:3},{value:"EModioModChangeType",id:"EModioModChangeType",level:3},{value:"EFileSizeUnit",id:"EFileSizeUnit",level:3},{value:"EModioEntitlementConsumptionState",id:"EModioEntitlementConsumptionState",level:3},{value:"EModioEntitlementType",id:"EModioEntitlementType",level:3},{value:"EModioErrorCondition",id:"EModioErrorCondition",level:3},{value:"EModioSortFieldType",id:"EModioSortFieldType",level:3},{value:"EModioSortDirection",id:"EModioSortDirection",level:3},{value:"EModioRevenueFilterType",id:"EModioRevenueFilterType",level:3},{value:"EModioImageState",id:"EModioImageState",level:3},{value:"EModioModState",id:"EModioModState",level:3},{value:"EModioModProgressState",id:"EModioModProgressState",level:3},{value:"EModioOpenStoreResult",id:"EModioOpenStoreResult",level:3},{value:"EModioRating",id:"EModioRating",level:3},{value:"EModioReportType",id:"EModioReportType",level:3}];function h(e){const d={a:"a",admonition:"admonition",code:"code",del:"del",em:"em",h2:"h2",h3:"h3",h4:"h4",h5:"h5",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,r.R)(),...e.components},{RefTable:n}=d;return n||function(e,d){throw new Error("Expected "+(d?"component":"object")+" `"+e+"` to be defined: you likely forgot to import, pass, or provide it.")}("RefTable",!0),(0,s.jsxs)(s.Fragment,{children:[(0,s.jsx)(d.h2,{id:"classes",children:"Classes"}),"\n",(0,s.jsx)(d.h3,{id:"modiocommontypeslibrary",children:"ModioCommonTypesLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiocreatemodlibrary",children:"ModioCreateModLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-1",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioerrorcodelibrary",children:"ModioErrorCodeLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-2",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioerrorconditionlibrary",children:"ModioErrorConditionLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-3",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioexamplelibrary",children:"ModioExampleLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-4",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioimagelibrary",children:"ModioImageLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-5",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodcollectionlibrary",children:"ModioModCollectionLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-6",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodprogressinfolibrary",children:"ModioModProgressInfoLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-7",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodtagoptionslibrary",children:"ModioModTagOptionsLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-8",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioplatformhelperslibrary",children:"ModioPlatformHelpersLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-9",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiopresetfilterparamslibrary",children:"ModioPresetFilterParamsLibrary"}),"\n",(0,s.jsx)(d.p,{children:"Blueprint library for working with preset filter parameters"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-10",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiosdklibrary",children:"ModioSDKLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-11",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiosubmissionextensionlibrary",children:"ModioSubmissionExtensionLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-12",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.h4,{id:"submitmodchangesasync",children:"SubmitModChangesAsync"}),"\n",(0,s.jsx)(d.p,{children:"Edits the parameters of a mod, by updating any fields set in the Params object to match the passed-in values. Fields left empty on the Params object will not be updated. This method also accepts a Png binary file for uploading a new logo."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubmissionExtensionLibrary_K2_SubmitModChangesFromMemoryAsync.png",src:i(79873).A+"",width:"328",height:"244"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitModChangesFromMemoryAsync(FModioModID Mod, FModioEditModParams Params, TArray PngData, FOnGetModInfoDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubmissionExtensionLibrary"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Mod"})}),(0,s.jsx)(d.td,{children:"The ID of the mod you wish to edit"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Params"})}),(0,s.jsx)(d.td,{children:"Descriptor containing the fields that should be altered."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PngData"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"The callback invoked when the changes have been submitted, containing an optional updated ModInfo object if the edits were performed successfully"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:"GenericError::SDKNotInitialized"}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:"UserDataError::InvalidUser"}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:"NetworkError"}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:"InvalidArgsError"}),(0,s.jsx)(d.td,{children:"Some fields in Params did not pass validation"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiosubsystem",children:"ModioSubsystem"}),"\n",(0,s.jsxs)(d.p,{children:[(0,s.jsx)(d.code,{children:"ModioSubsystem"})," is a thin wrapper around the mod.io SDK, wrapping all the functions available in the SDK's public header ",(0,s.jsx)(d.code,{children:"modio/ModioSDK.h"}),". This subsystem also converts mod.io SDK types to unreal-friendly types and caches some expensive operations."]}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-13",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> EngineSubsystem-> DynamicSubsystem-> Subsystem-> Object"}),"\n",(0,s.jsx)(d.h4,{id:"set-log-level",children:"Set Log Level"}),"\n",(0,s.jsx)(d.p,{children:"Sets the global logging level."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_SetLogLevel.png",src:i(28862).A+"",width:"306",height:"182"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetLogLevel(EModioLogLevel UnrealLogLevel)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-1",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UnrealLogLevel"})}),(0,s.jsx)(d.td,{children:"Determines which messages to include in the log output. Messages with a log level below the specified value will not be displayed."})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"run-pending-handlers",children:"Run Pending Handlers"}),"\n",(0,s.jsx)(d.p,{children:"Runs any pending mod.io work on the calling thread and invokes any callbacks passed to asynchronous operations."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_RunPendingHandlers.png",src:i(63183).A+"",width:"227",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void RunPendingHandlers()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-2",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"query-user-subscriptions",children:"Query User Subscriptions"}),"\n",(0,s.jsx)(d.p,{children:"Fetches the local view of the user's subscribed mods, including mods that are subscribed but not yet installed"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_QueryUserSubscriptions.png",src:i(26349).A+"",width:"230",height:"94"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"TMap QueryUserSubscriptions()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-3",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:[(0,s.jsx)(d.code,{children:"TMap"})," providing information about the subscribed mods"]}),"\n",(0,s.jsx)(d.h4,{id:"query-user-purchased-mods",children:"Query User Purchased Mods"}),"\n",(0,s.jsxs)(d.p,{children:["Returns the user's purchased mods. ",(0,s.jsx)(d.a,{href:"#fetchuserpurchasesasync",children:(0,s.jsx)(d.code,{children:"FetchUserPurchasesAsync"})})," must be called first to populate the cache."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_QueryUserPurchasedMods.png",src:i(703).A+"",width:"247",height:"94"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"TMap QueryUserPurchasedMods()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-1",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-4",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-1",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["A ",(0,s.jsx)(d.code,{children:"TMap"})," of all purchases a user has made."]}),"\n",(0,s.jsx)(d.h4,{id:"query-user-installations",children:"Query User Installations"}),"\n",(0,s.jsx)(d.p,{children:"Fetches the subset of the user's subscribed mods that are installed and ready for loading"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_QueryUserInstallations.png",src:i(94564).A+"",width:"333",height:"126"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"TMap QueryUserInstallations(bool bIncludeOutdatedMods)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-5",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bIncludeOutdatedMods"})}),(0,s.jsx)(d.td,{children:"Include subscribed mods that are installed but have an updated version on the server that has not yet been installed"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-2",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:[(0,s.jsx)(d.code,{children:"TMap"})," providing information about the subscribed mods"]}),"\n",(0,s.jsx)(d.h4,{id:"query-system-installations",children:"Query System Installations"}),"\n",(0,s.jsx)(d.p,{children:"Fetches all mods installed on the system, including those installed by other users."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_QuerySystemInstallations.png",src:i(31214).A+"",width:"239",height:"94"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"TMap QuerySystemInstallations()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-6",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-3",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["A ",(0,s.jsx)(d.code,{children:"TMap"})," of all mods installed on the system, including those installed by other users."]}),"\n",(0,s.jsx)(d.h4,{id:"prioritize-transfer-for-mod",children:"Prioritize Transfer for Mod"}),"\n",(0,s.jsx)(d.p,{children:"Cancels or suspends the current mod update, installation, or upload, and begins processing a pending operation for the specified mod ID"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_PrioritizeTransferForMod.png",src:i(16501).A+"",width:"271",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode PrioritizeTransferForMod(FModioModID ModToPrioritize)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-2",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-7",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModToPrioritize"})}),(0,s.jsx)(d.td,{children:"The ID for the mod to prioritize"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-4",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"Error code indicating the status of the prioritization request. Will be empty if the prioritization was successful or if the mod was already being processed"}),"\n",(0,s.jsx)(d.h5,{id:"error-values-1",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid or not present in the list of pending operations"})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"kill-background-thread",children:"Kill Background Thread"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_KillBackgroundThread.png",src:i(71552).A+"",width:"227",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void KillBackgroundThread()\n"})}),"\n",(0,s.jsx)(d.h4,{id:"verifyuserauthenticationasync",children:"VerifyUserAuthenticationAsync"}),"\n",(0,s.jsx)(d.p,{children:"Queries the server to verify the state of the currently authenticated user if there is one present"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_VerifyUserAuthenticationAsync.png",src:i(93680).A+"",width:"265",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_VerifyUserAuthenticationAsync(FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-3",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-8",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["Callback invoked with the results of the verification process. An empty ",(0,s.jsx)(d.code,{children:"ModioErrorCode"})," indicates successful verification i.e. the mod.io server was contactable and the user's authentication remains valid."]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-2",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"unsubscribefrommodasync",children:"UnsubscribeFromModAsync"}),"\n",(0,s.jsx)(d.p,{children:"Sends a request to the mod.io server to remove the specified mod from the user's list of subscriptions. If no other local users are subscribed to the specified mod this function will also mark the mod for uninstallation by the SDK."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync.png",src:i(28297).A+"",width:"248",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_UnsubscribeFromModAsync(FModioModID ModToUnsubscribeFrom, FOnErrorOnlyDelegate OnUnsubscribeComplete)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-4",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-9",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModToUnsubscribeFrom"})}),(0,s.jsx)(d.td,{children:"Mod ID of the mod requiring unsubscription."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"On Unsubscribe Complete"})}),(0,s.jsx)(d.td,{children:"Callback invoked when the unsubscription request is completed."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-3",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"unmuteuserasync",children:"UnmuteUserAsync"}),"\n",(0,s.jsx)(d.p,{children:"Unmute a user. This allows mod.io to display mods authored by the now unmuted user when performing searches."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_UnmuteUserAsync.png",src:i(60291).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_UnmuteUserAsync(FModioUserID UserID, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-5",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-10",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserID"})}),(0,s.jsx)(d.td,{children:"ID of the user to unmute"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of unmuting the user."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-4",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied user ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"subscribetomodasync",children:"SubscribeToModAsync"}),"\n",(0,s.jsx)(d.p,{children:"Sends a request to the mod.io server to add the specified mod to the user's list of subscriptions, and marks the mod for local installation by the SDK"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubscribeToModAsync.png",src:i(72021).A+"",width:"251",height:"218"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_SubscribeToModAsync(FModioModID ModToSubscribeTo, bool IncludeDependencies, FOnErrorOnlyDelegate OnSubscribeComplete)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-6",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-11",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModToSubscribeTo"})}),(0,s.jsx)(d.td,{children:"Mod ID of the mod requiring a subscription."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"IncludeDependencies"})}),(0,s.jsx)(d.td,{children:"Subscribe to all dependencies as well."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"On Subscribe Complete"})}),(0,s.jsx)(d.td,{children:"Callback invoked when the subscription request is completed."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-5",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"submitnewmodfileformod",children:"SubmitNewModFileForMod"}),"\n",(0,s.jsxs)(d.p,{children:["Queues the upload of a new modfile release for the specified mod using the submitted parameters. This function takes an ",(0,s.jsx)(d.code,{children:"ModioCreateModFileParams"})," object to specify the path to the root folder of the new modfile. The plugin will compress the folder's contents into a .zip archive and queue the result for upload. When the upload completes, a mod management event will be triggered. Note the plugin is also responsible for decompressing the archive upon its installation at a later point in time."]}),"\n",(0,s.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,s.jsx)(d.p,{children:"This function is part of an experimental feature and is subject to change."})}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubmitNewModFileForMod.png",src:i(64303).A+"",width:"237",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitNewModFileForMod(FModioModID Mod, FModioCreateModFileParams Params)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-7",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-12",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Mod"})}),(0,s.jsx)(d.td,{children:"The ID of the mod you are submitting a file for"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Params"})}),(0,s.jsx)(d.td,{children:"Information about the mod file being created, including the root path of the directory that will be archived"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-6",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"submitnewmodasync",children:"SubmitNewModAsync"}),"\n",(0,s.jsx)(d.p,{children:"Requests the creation of a new mod on the server with the specified parameters"}),"\n",(0,s.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,s.jsx)(d.p,{children:"This function is part of an experimental feature and is subject to change."})}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubmitNewModAsync.png",src:i(93124).A+"",width:"227",height:"214"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitNewModAsync(FModioModCreationHandle Handle, FModioCreateModParams Params, FOnSubmitNewModDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-8",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-13",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Handle"})}),(0,s.jsxs)(d.td,{children:["The ",(0,s.jsx)(d.code,{children:"ModioModCreationHandle"})," for this submission. Once this method invokes your callback indicating success, the ",(0,s.jsx)(d.code,{children:"ModioModCreationHandle"})," is invalid for the rest of the session. You should request a new one for the next submission attempt."]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Params"})}),(0,s.jsx)(d.td,{children:"Information about the new mod to create"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,s.jsx)(d.code,{children:"ModioModID"})," for the newly created mod"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-7",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsxs)(d.td,{children:["Some fields in ",(0,s.jsx)(d.code,{children:"Params"})," did not pass validation"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"submitmodratingasync",children:"SubmitModRatingAsync"}),"\n",(0,s.jsx)(d.p,{children:"Submits a rating for a mod on behalf of the current user. Submit a neutral rating to effectively clear a rating already submitted by a user. Submitting other values will overwrite any existing rating submitted by this user."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubmitModRatingAsync.png",src:i(87149).A+"",width:"306",height:"242"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitModRatingAsync(FModioModID Mod, EModioRating Rating, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-9",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-14",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Mod"})}),(0,s.jsx)(d.td,{children:"The mod the user is rating"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Rating"})}),(0,s.jsx)(d.td,{children:"The rating the user wishes to submit"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of the rating submission"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-8",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"submitmodchangesasync-1",children:"SubmitModChangesAsync"}),"\n",(0,s.jsxs)(d.p,{children:["Edits the parameters of a mod by updating any fields set in the ",(0,s.jsx)(d.code,{children:"Params"})," object to match the passed-in values. Fields left empty on the ",(0,s.jsx)(d.code,{children:"Params"})," object will not be updated."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SubmitModChangesAsync.png",src:i(83175).A+"",width:"233",height:"214"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitModChangesAsync(FModioModID Mod, FModioEditModParams Params, FOnGetModInfoDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-10",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-15",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Mod"})}),(0,s.jsx)(d.td,{children:"The ID of the mod you wish to edit"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Params"})}),(0,s.jsx)(d.td,{children:"Descriptor containing the fields that should be altered."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["The callback invoked when the changes have been submitted containing an optional updated ",(0,s.jsx)(d.code,{children:"ModioModInfo"})," object if the edits were performed successfully"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-9",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsxs)(d.td,{children:["Some fields in ",(0,s.jsx)(d.code,{children:"Params"})," did not pass validation"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"shutdownasync",children:"ShutdownAsync"}),"\n",(0,s.jsxs)(d.p,{children:["Cancels any running internal operations, frees SDK resources, and invokes any pending callbacks with an ",(0,s.jsx)(d.code,{children:"OperationCanceled"})," error category. This function will NOT block while the deinitialization occurs."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ShutdownAsync.png",src:i(68768).A+"",width:"232",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_ShutdownAsync(FOnErrorOnlyDelegate OnShutdownComplete)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-16",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"On Shutdown Complete"})}),(0,s.jsxs)(d.td,{children:["Callback invoked when the plugin is shut down and calling ",(0,s.jsx)(d.a,{href:"#run-pending-handlers",children:(0,s.jsx)(d.code,{children:"RunPendingHandlers"})})," is no longer required"]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"setlanguage",children:"SetLanguage"}),"\n",(0,s.jsx)(d.p,{children:"Set language to get corresponding data from the server"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_SetLanguage.png",src:i(53784).A+"",width:"306",height:"182"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_SetLanguage(EModioLanguage Locale)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-17",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Locale"})}),(0,s.jsx)(d.td,{children:"Language to set"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"requestemailauthcodeasync",children:"RequestEmailAuthCodeAsync"}),"\n",(0,s.jsx)(d.p,{children:"Begins email authentication for the current session by requesting a one-time code be sent to the specified email address."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_RequestEmailAuthCodeAsync.png",src:i(67942).A+"",width:"252",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_RequestEmailAuthCodeAsync(FModioEmailAddress EmailAddress, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-11",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-18",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EmailAddress"})}),(0,s.jsx)(d.td,{children:"The email address to send the code to"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code indicating the outcome of the request"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-10",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserAlreadyAuthenticatedError"})}),(0,s.jsxs)(d.td,{children:["Current user is already authenticated. De-authenticate the current user with ",(0,s.jsx)(d.a,{href:"#clearuserdataasync",children:(0,s.jsx)(d.code,{children:"ClearUserDataAsync"})}),", and re-initialize the SDK by calling ",(0,s.jsx)(d.a,{href:"#shutdownasync",children:(0,s.jsx)(d.code,{children:"ShutdownAsync"})})," then ",(0,s.jsx)(d.a,{href:"#initializeasync",children:(0,s.jsx)(d.code,{children:"InitializeAsync"})}),"."]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"reportcontentasync",children:"ReportContentAsync"}),"\n",(0,s.jsxs)(d.p,{children:["Sends a content report to mod.io. When using this function, please inform your users that if they provide their contact name or details in the ",(0,s.jsx)(d.code,{children:"Report"})," parameter, this data may be shared with the person responsible for the content being reported. For more information on what data in a report will be shared with whom, please see ",(0,s.jsx)(d.a,{href:"https://mod.io/report",children:"our website's report form"}),"."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ReportContentAsync.png",src:i(57479).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_ReportContentAsync(FModioReportParams Report, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-12",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-19",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Report"})}),(0,s.jsx)(d.td,{children:"Information about the content being reported and a description of the report."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code to indicate successful submission of the report."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-11",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"Required information in the report did not pass validation"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsxs)(d.td,{children:["The mod ID, game ID, or user ID supplied to ",(0,s.jsx)(d.code,{children:"Report"})," is invalid"]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"removefromtempmodset",children:"RemoveFromTempModSet"}),"\n",(0,s.jsxs)(d.p,{children:["Remove mods from a temp mod set. Every temp mod specified by ",(0,s.jsx)(d.code,{children:"ModIds"})," will be uninstalled unless the user is already subscribed."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_RemoveFromTempModSet.png",src:i(89494).A+"",width:"235",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_RemoveFromTempModSet(TArray ModIds)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-13",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"mod-management-enabled"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"temp-mod-set-initialized"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-20",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModIds"})}),(0,s.jsxs)(d.td,{children:["TArray of ",(0,s.jsx)(d.code,{children:"ModioModID"}),"s to remove as temp mods"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-5",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["Error code indicating the status of the ",(0,s.jsx)(d.code,{children:"TempModSet"}),". Will be empty if it was successful"]}),"\n",(0,s.jsx)(d.h5,{id:"error-values-12",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,s.jsx)(d.td,{children:"Mod management not enabled"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TempModSetNotInitialized"})}),(0,s.jsxs)(d.td,{children:[(0,s.jsx)(d.code,{children:"TempModSet"})," not initialized"]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"refreshuserentitlementsasync",children:"RefreshUserEntitlementsAsync"}),"\n",(0,s.jsx)(d.p,{children:"Requests mod.io refresh the available entitlements for the current user purchased through the portal and currently authenticated."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_RefreshUserEntitlementsAsync.png",src:i(83788).A+"",width:"265",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_RefreshUserEntitlementsAsync(FModioEntitlementParams Params, FOnRefreshUserEntitlementsDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-14",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-21",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Params"})}),(0,s.jsx)(d.td,{children:"Additional parameters."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the refresh operation."})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"queryuserprofile",children:"QueryUserProfile"}),"\n",(0,s.jsx)(d.p,{children:"Fetches the currently authenticated mod.io user profile if there is one"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_QueryUserProfile.png",src:i(56696).A+"",width:"227",height:"92"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioOptionalUser K2_QueryUserProfile()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-22",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-6",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:[(0,s.jsx)(d.code,{children:"ModioOptionalUser"})," object containing profile information"]}),"\n",(0,s.jsx)(d.h4,{id:"querytempmodset",children:"QueryTempModSet"}),"\n",(0,s.jsxs)(d.p,{children:["Query every system and temp mod in ",(0,s.jsx)(d.code,{children:"TempModSet"})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_QueryTempModSet.png",src:i(55060).A+"",width:"227",height:"126"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"TMap K2_QueryTempModSet()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-23",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-7",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["TMap using ",(0,s.jsx)(d.code,{children:"ModioModID"})," as keys and ",(0,s.jsx)(d.code,{children:"ModioModCollectionEntry"})," objects providing information about mods in ",(0,s.jsx)(d.code,{children:"TempModSet"})]}),"\n",(0,s.jsx)(d.h4,{id:"querycurrentmodupdate",children:"QueryCurrentModUpdate"}),"\n",(0,s.jsx)(d.p,{children:"Provides progress information for a mod installation or update operation if one is currently in progress."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_QueryCurrentModUpdate.png",src:i(72340).A+"",width:"227",height:"92"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioOptionalModProgressInfo K2_QueryCurrentModUpdate()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-24",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-8",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:[(0,s.jsx)(d.code,{children:"ModioOptionalModProgressInfo"})," object containing information regarding the progress of the installation operation."]}),"\n",(0,s.jsx)(d.h4,{id:"purchasemodasync",children:"PurchaseModAsync"}),"\n",(0,s.jsx)(d.p,{children:"Purchases a mod for the current player"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_PurchaseModAsync.png",src:i(68537).A+"",width:"227",height:"214"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_PurchaseModAsync(FModioModID ModID, FModioUnsigned64 ExpectedPrice, FOnPurchaseModDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-15",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-25",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModID"})}),(0,s.jsx)(d.td,{children:"ID of the mod to purchase"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ExpectedPrice"})}),(0,s.jsxs)(d.td,{children:["The price the user is expected to pay for the mod, generally ",(0,s.jsx)(d.a,{href:"#modiomodinfo",children:(0,s.jsx)(d.code,{children:"ModioModInfo.Price"})}),". This ensures that there is consistency between the displayed price and the price in the backend. If there is a mismatch, the purchase will fail."]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback invoked with purchase information once the purchase is completed."})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"previewexternalupdatesasync",children:"PreviewExternalUpdatesAsync"}),"\n",(0,s.jsxs)(d.p,{children:["Retrieve a list of updates between the users local mod state, and the server-side state. This allows you to identify which mods will be modified by the next call to ",(0,s.jsx)(d.a,{href:"#fetchexternalupdatesasync",children:(0,s.jsx)(d.code,{children:"FetchExternalUpdatesAsync"})})," in order to perform any content management (such as unloading files) that might be required."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_PreviewExternalUpdatesAsync.png",src:i(99101).A+"",width:"261",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_PreviewExternalUpdatesAsync(FOnPreviewExternalUpdatesDelegate OnPreviewDone)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-26",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"On Preview Done"})}),(0,s.jsx)(d.td,{children:"Callback invoked when the external state has been retrieved. It contains a dictionary with ModID as keys and change maps as values. Empty when there are no differences between local and the mod.io API service"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"muteuserasync",children:"MuteUserAsync"}),"\n",(0,s.jsx)(d.p,{children:"Mute a user. This will prevent mod.io from returning mods authored by the muted user. when performing searches."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MuteUserAsync.png",src:i(96252).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_MuteUserAsync(FModioUserID UserID, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-16",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-27",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserID"})}),(0,s.jsx)(d.td,{children:"ID of the User to mute"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of muting the user."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-13",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied user ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"metricssessionstartasync",children:"MetricsSessionStartAsync"}),"\n",(0,s.jsx)(d.p,{children:"Start a metrics play session"}),"\n",(0,s.jsx)(d.admonition,{title:"Premium Feature",type:"info",children:(0,s.jsxs)(d.p,{children:["This function requires the ",(0,s.jsx)(d.strong,{children:"Metrics"})," premium feature. Please contact your mod.io representative for more details."]})}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MetricsSessionStartAsync.png",src:i(10293).A+"",width:"238",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_MetricsSessionStartAsync(FModioMetricsSessionParams Params, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-28",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Params"})}),(0,s.jsxs)(d.td,{children:[(0,s.jsx)(d.code,{children:"ModioMetricsSessionParams"})," struct containing information of what and how to start a metrics session"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the session start operation"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-14",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RateLimited"})}),(0,s.jsx)(d.td,{children:"Too many frequent calls to the API. Wait some time and try again."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidUser"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionNotInitialized"})}),(0,s.jsx)(d.td,{children:"Metrics session has not yet been initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionIsActive"})}),(0,s.jsx)(d.td,{children:"Metrics session is currently active and running"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"BadParameter"})}),(0,s.jsx)(d.td,{children:"One or more values in the Metric Session Parameters are invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"metricssessionsendheartbeatonceasync",children:"MetricsSessionSendHeartbeatOnceAsync"}),"\n",(0,s.jsx)(d.p,{children:"Sends a single heartbeat to the mod.io server to indicate a session is still active"}),"\n",(0,s.jsx)(d.admonition,{title:"Premium Feature",type:"info",children:(0,s.jsxs)(d.p,{children:["This function requires the ",(0,s.jsx)(d.strong,{children:"Metrics"})," premium feature. Please contact your mod.io representative for more details."]})}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatOnceAsync.png",src:i(44014).A+"",width:"326",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_MetricsSessionSendHeartbeatOnceAsync(FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-29",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the session heartbeat operation"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-15",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidUser"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionNotInitialized"})}),(0,s.jsx)(d.td,{children:"Metrics session has not yet been initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionIsNotActive"})}),(0,s.jsxs)(d.td,{children:["Metrics session is not currently running. Call ",(0,s.jsx)(d.a,{href:"#metricssessionstartasync",children:(0,s.jsx)(d.code,{children:"MetricsSessionStartAsync"})})," before attempting to sending a heartbeat."]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"metricssessionsendheartbeatatintervalasync",children:"MetricsSessionSendHeartbeatAtIntervalAsync"}),"\n",(0,s.jsx)(d.p,{children:"Sends a constant heartbeat at a given interval to the mod.io server to indicate a session is still active"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync.png",src:i(14811).A+"",width:"354",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_MetricsSessionSendHeartbeatAtIntervalAsync(FModioUnsigned64 IntervalSeconds, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-30",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"IntervalSeconds"})}),(0,s.jsx)(d.td,{children:"The frequency in seconds to send a heartbeat to the mod.io server"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the session heartbeat operation"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-16",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidUser"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionNotInitialized"})}),(0,s.jsx)(d.td,{children:"Metrics session has not yet been initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionIsNotActive"})}),(0,s.jsxs)(d.td,{children:["Metrics session is not currently running. Call ",(0,s.jsx)(d.a,{href:"#metricssessionstartasync",children:(0,s.jsx)(d.code,{children:"MetricsSessionStartAsync"})})," before attempting to sending a heartbeat."]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"metricssessionendasync",children:"MetricsSessionEndAsync"}),"\n",(0,s.jsx)(d.p,{children:"Ends a metrics play session"}),"\n",(0,s.jsx)(d.admonition,{title:"Premium Feature",type:"info",children:(0,s.jsxs)(d.p,{children:["This function requires the ",(0,s.jsx)(d.strong,{children:"Metrics"})," premium feature. Please contact your mod.io representative for more details."]})}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_MetricsSessionEndAsync.png",src:i(92124).A+"",width:"229",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_MetricsSessionEndAsync(FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-31",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing an error code indicating success or failure of the session end operation"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-17",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RateLimited"})}),(0,s.jsx)(d.td,{children:"Too many frequent calls to the API. Wait some time and try again."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidUser"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionNotInitialized"})}),(0,s.jsx)(d.td,{children:"Metrics session has not yet been initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionIsNotActive"})}),(0,s.jsxs)(d.td,{children:["Metrics session is not currently running. Call ",(0,s.jsx)(d.a,{href:"#metricssessionstartasync",children:(0,s.jsx)(d.code,{children:"MetricsSessionStartAsync"})})," before attempting to end a session."]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"listusergamesasync",children:"ListUserGamesAsync"}),"\n",(0,s.jsx)(d.p,{children:"Provides a list of games for the current user that match the parameters specified in the filter"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ListUserGamesAsync.png",src:i(6938).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_ListUserGamesAsync(FModioFilterParams Filter, FOnListUserGamesDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-17",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-32",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Filter"})}),(0,s.jsxs)(d.td,{children:[(0,s.jsx)(d.a,{href:"#modiofilterparams",children:(0,s.jsx)(d.code,{children:"ModioFilterParams"})})," object containing any filters that should be applied to the query"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["Callback invoked with a status code and an optional ",(0,s.jsx)(d.code,{children:"ModioGameInfoList"})," providing game profiles"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-18",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"listusercreatedmodsasync",children:"ListUserCreatedModsAsync"}),"\n",(0,s.jsx)(d.p,{children:"Provides a list of mods that the user has submitted or is a team member of for the current game, applying the parameters specified in the filter."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ListUserCreatedModsAsync.png",src:i(22028).A+"",width:"245",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_ListUserCreatedModsAsync(FModioFilterParams Filter, FOnListUserCreatedModsDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-18",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-33",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Filter"})}),(0,s.jsx)(d.td,{children:"Filter to apply when listing the user's created mods."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback invoked when the call succeeds, or when an error occurs."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-19",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RateLimited"})}),(0,s.jsx)(d.td,{children:"Too many frequent calls to the API. Wait some time and try again."})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"listallmodsasync",children:"ListAllModsAsync"}),"\n",(0,s.jsx)(d.p,{children:"Provides a list of mods for the current game that match the parameters specified in the filter"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ListAllModsAsync.png",src:i(85890).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_ListAllModsAsync(FModioFilterParams Filter, FOnListAllModsDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-19",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-34",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Filter"})}),(0,s.jsxs)(d.td,{children:[(0,s.jsx)(d.a,{href:"#modiofilterparams",children:(0,s.jsx)(d.code,{children:"ModioFilterParams"})})," object containing any filters that should be applied to the query"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["Callback invoked with a status code and an optional ",(0,s.jsx)(d.code,{children:"ModioModInfoList"})," providing mod profiles"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-20",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"inittempmodset",children:"InitTempModSet"}),"\n",(0,s.jsxs)(d.p,{children:["Install every temp mod specified by ",(0,s.jsx)(d.code,{children:"ModIds"})," if not already installed."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_InitTempModSet.png",src:i(25684).A+"",width:"227",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_InitTempModSet(TArray ModIds)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-20",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"mod-management-enabled"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-35",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModIds"})}),(0,s.jsxs)(d.td,{children:["TArray of ",(0,s.jsx)(d.code,{children:"ModioModID"}),"s to install as temp mods"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-9",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["Error code indicating the status of the ",(0,s.jsx)(d.code,{children:"TempModSet"}),". Will be empty if it was successful"]}),"\n",(0,s.jsx)(d.h5,{id:"error-values-21",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,s.jsx)(d.td,{children:"Mod management not enabled"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"initializeasync",children:"InitializeAsync"}),"\n",(0,s.jsx)(d.p,{children:"Initializes the SDK for the given user. Loads the state of mods installed on the system as well as the set of mods the specified user has installed on this device"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_InitializeAsync.png",src:i(49280).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_InitializeAsync(FModioInitializeOptions InitializeOptions, FOnErrorOnlyDelegate OnInitComplete)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-36",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InitializeOptions"})}),(0,s.jsx)(d.td,{children:"Parameters to the function packed as a struct where all members needs to be initialized for the call to succeed"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"On Init Complete"})}),(0,s.jsx)(d.td,{children:"Callback which will be invoked with the result of initialization"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-22",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FilesystemError"})}),(0,s.jsx)(d.td,{children:"Couldn't create the user data or common data folders"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ConfigurationError"})}),(0,s.jsxs)(d.td,{children:["InitializeOptions contains an invalid value - inspect ",(0,s.jsx)(d.code,{children:"ec.value()"})," to determine what was incorrect"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKAlreadyInitialized"})}),(0,s.jsx)(d.td,{children:"SDK already initialized"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getuserwalletbalanceasync",children:"GetUserWalletBalanceAsync"}),"\n",(0,s.jsxs)(d.p,{children:["Gets the users current wallet balance. This will also create a wallet for a user if one does not exist. You should ensure this is called prior to calling ",(0,s.jsx)(d.a,{href:"#purchasemodasync",children:(0,s.jsx)(d.code,{children:"PurchaseModAsync"})})," purchase will fail."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync.png",src:i(21964).A+"",width:"248",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetUserWalletBalanceAsync(FOnGetUserWalletBalanceDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-21",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-37",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback invoked with the users wallet balance"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getusermediaasync-avatar",children:"GetUserMediaAsync (Avatar)"}),"\n",(0,s.jsx)(d.p,{children:"Downloads the avatar of the currently authenticated user. Will only perform a download if there is no local cache of the avatar or if that cached copy is out-of-date."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetUserMediaAvatarAsync.png",src:i(96438).A+"",width:"306",height:"212"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetUserMediaAvatarAsync(EModioAvatarSize AvatarSize, FOnGetMediaDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-22",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-38",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"AvatarSize"})}),(0,s.jsx)(d.td,{children:"Parameter specifying the size of avatar image to download"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code for the download and an optional path to the downloaded image"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-23",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getuserdelegationtokenasync",children:"GetUserDelegationTokenAsync"}),"\n",(0,s.jsx)(d.p,{children:"Get a user delegation token that can be used for S2S service calls"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetUserDelegationTokenAsync.png",src:i(40348).A+"",width:"263",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetUserDelegationTokenAsync(FOnGetUserDelegationTokenDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-23",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-39",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback invoked with purchase information once the purchase is completed."})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"gettermsofuseasync",children:"GetTermsOfUseAsync"}),"\n",(0,s.jsx)(d.p,{children:"This function retrieves the information required for a game to display the mod.io terms of use to a player who wishes to create a mod.io account"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetTermsOfUseAsync.png",src:i(65789).A+"",width:"227",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetTermsOfUseAsync(FOnGetTermsOfUseDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-24",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-40",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback invoked with the terms of use data once retrieved from the server"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-24",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getmutedusersasync",children:"GetMutedUsersAsync"}),"\n",(0,s.jsx)(d.p,{children:"List all the users that have been muted by the current user."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetMutedUsersAsync.png",src:i(47193).A+"",width:"227",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetMutedUsersAsync(FOnMuteUsersDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-25",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-41",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of the operation, and an optional containing a list of muted users if successful."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-25",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getmodtagoptionsasync",children:"GetModTagOptionsAsync"}),"\n",(0,s.jsxs)(d.p,{children:["Fetches the available tags used on mods for the current game. These tags can them be used in conjunction with the FilterParams passed to ",(0,s.jsx)(d.a,{href:"#listallmodsasync",children:(0,s.jsx)(d.code,{children:"ListAllModsAsync"})})," Will be cached when first received"]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModTagOptionsAsync.png",src:i(74220).A+"",width:"229",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModTagOptionsAsync(FOnGetModTagOptionsDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-26",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-42",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,s.jsx)(d.code,{children:"ModioModTagOptions"})," object containing the available tags"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-26",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getmodmediaasync-logo",children:"GetModMediaAsync (Logo)"}),"\n",(0,s.jsx)(d.p,{children:"Downloads the logo for the specified mod. Will use existing file if it is already present on disk"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModMediaLogoAsync.png",src:i(90655).A+"",width:"306",height:"242"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModMediaLogoAsync(FModioModID ModId, EModioLogoSize LogoSize, FOnGetMediaDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-27",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-43",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModId"})}),(0,s.jsx)(d.td,{children:"Mod ID for use in logo retrieval"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"LogoSize"})}),(0,s.jsx)(d.td,{children:"Parameter indicating the size of logo that's required"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code and an optional path object pointing to the location of the downloaded image"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-27",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"Specified mod media does not exist or was deleted"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InsufficientSpace"})}),(0,s.jsx)(d.td,{children:"Not enough space for the file"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getmodmediaasync-gallery-image",children:"GetModMediaAsync (Gallery Image)"}),"\n",(0,s.jsx)(d.p,{children:"Get a gallery image for the specified mod ID. If it already exists on disk the file will be reused unless it is outdated"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync.png",src:i(34351).A+"",width:"306",height:"277"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModMediaGalleryImageAsync(FModioModID ModId, EModioGallerySize GallerySize, int32 Index, FOnGetMediaDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-28",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-44",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModId"})}),(0,s.jsx)(d.td,{children:"The mod you want to retrieve an image for"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GallerySize"})}),(0,s.jsx)(d.td,{children:"Size of the image you want to retrieve"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Index"})}),(0,s.jsx)(d.td,{children:"The zero-based index of the image you want to retrieve"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback containing a status code and an Optional containing a path to the image file on disk"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-28",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"Specified mod media does not exist or was deleted"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InsufficientSpace"})}),(0,s.jsx)(d.td,{children:"Not enough space for the file"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getmodmediaasync-avatar",children:"GetModMediaAsync (Avatar)"}),"\n",(0,s.jsx)(d.p,{children:"Downloads the creator avatar for a specified mod. Will use existing file if it is already present on disk and not outdated"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModMediaAvatarAsync.png",src:i(60973).A+"",width:"306",height:"242"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModMediaAvatarAsync(FModioModID ModId, EModioAvatarSize AvatarSize, FOnGetMediaDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-29",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-45",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModId"})}),(0,s.jsx)(d.td,{children:"ID of the mod the creator avatar will be retrieved for"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"AvatarSize"})}),(0,s.jsx)(d.td,{children:"Parameter indicating the size of avatar image that's required"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code and an optional path object pointing to the location of the downloaded image"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-29",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"Specified mod media does not exist or was deleted"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InsufficientSpace"})}),(0,s.jsx)(d.td,{children:"Not enough space for the file"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getmodinfoasync",children:"GetModInfoAsync"}),"\n",(0,s.jsx)(d.p,{children:"Fetches detailed information about the specified mod, including description and file metadata for the most recent release"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModInfoAsync.png",src:i(7434).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModInfoAsync(FModioModID ModId, FOnGetModInfoDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-30",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-46",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModId"})}),(0,s.jsx)(d.td,{children:"Mod ID of the mod to fetch data"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,s.jsx)(d.code,{children:"ModioModInfo"})," object with the mod's extended information"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-30",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getmoddependenciesasync",children:"GetModDependenciesAsync"}),"\n",(0,s.jsx)(d.p,{children:"For a given Mod ID, fetches a list of any mods that the creator has marked as dependencies"}),"\n",(0,s.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,s.jsx)(d.p,{children:"This function is part of an experimental feature and is subject to change."})}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModDependenciesAsync.png",src:i(74955).A+"",width:"245",height:"218"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetModDependenciesAsync(FModioModID ModID, bool Recursive, FOnGetModDependenciesDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-31",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-47",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModID"})}),(0,s.jsx)(d.td,{children:"The mod to retrieve dependencies for"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Recursive"})}),(0,s.jsx)(d.td,{children:"Fetches dependencies recursively up to a depth of 5"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,s.jsx)(d.code,{children:"ModioModTagOptions"})," object containing the available tags"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-31",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"getmodcreationhandle",children:"GetModCreationHandle"}),"\n",(0,s.jsxs)(d.p,{children:["Gets a new mod handle for use with ",(0,s.jsx)(d.a,{href:"#submitnewmodasync",children:(0,s.jsx)(d.code,{children:"SubmitNewModAsync"})}),"."]}),"\n",(0,s.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,s.jsx)(d.p,{children:"This function is part of an experimental feature and is subject to change."})}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetModCreationHandle.png",src:i(56025).A+"",width:"227",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioModCreationHandle K2_GetModCreationHandle()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-48",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"getlanguage",children:"GetLanguage"}),"\n",(0,s.jsx)(d.p,{children:"Get the currently applied language"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetLanguage.png",src:i(82892).A+"",width:"227",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioLanguage K2_GetLanguage()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-49",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-10",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"Current language"}),"\n",(0,s.jsx)(d.h4,{id:"getgameinfoasync",children:"GetGameInfoAsync"}),"\n",(0,s.jsx)(d.p,{children:"Fetches detailed information about the specified game"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetGameInfoAsync.png",src:i(85314).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_GetGameInfoAsync(FModioGameID GameID, FOnGetGameInfoDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-32",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-50",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameID"})}),(0,s.jsx)(d.td,{children:"Game ID of the game data to fetch"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["Callback providing a status code and an optional ",(0,s.jsx)(d.code,{children:"ModioGameInfo"})," object with the game's extended information"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-32",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"Specified game does not exist"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied game ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"forceuninstallmodasync",children:"ForceUninstallModAsync"}),"\n",(0,s.jsxs)(d.p,{children:["Forcibly uninstalls a mod from the system. This can be used when the host application requires additional space for other mods. The current user must not be subscribed to the mod to force uninstall. To remove a mod the current user is subscribed to, first use ",(0,s.jsx)(d.a,{href:"#unsubscribefrommodasync",children:(0,s.jsx)(d.code,{children:"UnsubscribeFromModAsync"})}),". If the mod does not uninstall (due to a different user on the same system remaining subscribed), ",(0,s.jsx)(d.code,{children:"ForceUninstallModAsync"})," can be called next."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ForceUninstallModAsync.png",src:i(2053).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_ForceUninstallModAsync(FModioModID ModToRemove, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-51",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModToRemove"})}),(0,s.jsx)(d.td,{children:"The mod ID to force uninstall."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback invoked indicating success or failure of the uninstallation."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-33",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:"ApiErrorRefSuccess"}),(0,s.jsx)(d.td,{children:"User is still subscribed to the specified mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"fetchuserpurchasesasync",children:"FetchUserPurchasesAsync"}),"\n",(0,s.jsxs)(d.p,{children:["Fetches the user's purchases. This populates a runtime cache of purchase information that can be accessed using ",(0,s.jsx)(d.a,{href:"#query-user-purchased-mods",children:(0,s.jsx)(d.code,{children:"QueryUserPurchasedMods"})}),"."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync.png",src:i(5859).A+"",width:"240",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_FetchUserPurchasesAsync(FOnFetchUserPurchasesDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-33",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-52",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback invoked once the call has been completed."})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"fetchexternalupdatesasync",children:"FetchExternalUpdatesAsync"}),"\n",(0,s.jsx)(d.p,{children:"Synchronises the local list of the current user's subscribed mods with the server. Any mods that have been externally subscribed will be automatically marked for installation, and mods that have been externally removed from the user's subscriptions may be uninstalled if no other local users have a current subscription."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync.png",src:i(86649).A+"",width:"247",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_FetchExternalUpdatesAsync(FOnErrorOnlyDelegate OnFetchDone)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-53",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"On Fetch Done"})}),(0,s.jsx)(d.td,{children:"Callback invoked when the external state has been retrieved and merged with the local data"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"enablemodmanagement",children:"EnableModManagement"}),"\n",(0,s.jsx)(d.p,{children:"Enables the automatic management of installed mods on the system based on the user's subscriptions."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_EnableModManagement.png",src:i(69724).A+"",width:"227",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_EnableModManagement(FOnModManagementDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-54",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsxs)(d.td,{children:["This callback handler will be invoked with a ",(0,s.jsx)(d.a,{href:"#modiomodmanagementevent",children:(0,s.jsx)(d.code,{children:"ModioModManagementEvent"})})," for each mod operation performed by the SDK"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-11",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["An error code indicating success or failure of enabling mod management. Note that this is independent of error codes for mod management events. Inspect the ",(0,s.jsx)(d.code,{children:"Callback"})," for information on each mod management event."]}),"\n",(0,s.jsx)(d.h5,{id:"error-values-34",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModManagementAlreadyEnabled"})}),(0,s.jsx)(d.td,{children:"Mod management was already enabled. The mod management callback has not been changed."})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"closetempmodset",children:"CloseTempModSet"}),"\n",(0,s.jsx)(d.p,{children:"Uninstall every temp mod unless the user is subscribed."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_CloseTempModSet.png",src:i(69252).A+"",width:"227",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_CloseTempModSet()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-34",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"mod-management-enabled"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"temp-mod-set-initialized"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-55",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-12",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["Error code indicating the status of the ",(0,s.jsx)(d.code,{children:"TempModSet"}),". Will be empty if it was successful"]}),"\n",(0,s.jsx)(d.h5,{id:"error-values-35",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,s.jsx)(d.td,{children:"Mod management not enabled"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TempModSetNotInitialized"})}),(0,s.jsxs)(d.td,{children:[(0,s.jsx)(d.code,{children:"TempModSet"})," not initialized"]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"clearuserdataasync",children:"ClearUserDataAsync"}),"\n",(0,s.jsx)(d.p,{children:"De-authenticates the current mod.io user for the current session, and clears all user-specific data stored on the current device. Any subscribed mods that are installed but do not have other local users subscribed will be uninstalled"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ClearUserDataAsync.png",src:i(36106).A+"",width:"227",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_ClearUserDataAsync(FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-35",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-56",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code indicating the outcome of clearing the user data. Error codes returned by this function are informative only - it will always succeed."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-36",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"No authenticated user"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"authenticateuserexternalasync",children:"AuthenticateUserExternalAsync"}),"\n",(0,s.jsx)(d.p,{children:"Uses platform-specific authentication to associate a mod.io user account with the current platform user"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_AuthenticateUserExternalAsync.png",src:i(92223).A+"",width:"306",height:"242"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_AuthenticateUserExternalAsync(FModioAuthenticationParams User, EModioAuthenticationProvider Provider, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-36",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-57",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"User"})}),(0,s.jsx)(d.td,{children:"Authentication payload data to submit to the provider"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Provider"})}),(0,s.jsx)(d.td,{children:"The provider to use to perform the authentication"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback invoked once the authentication request has been made"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-37",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ConfigurationError"})}),(0,s.jsx)(d.td,{children:"The SDK's configuration is not valid"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The arguments passed to the function have failed validation"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserTermsOfUseError"})}),(0,s.jsx)(d.td,{children:"The user has not yet accepted the mod.io Terms of Use"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserAlreadyAuthenticatedError"})}),(0,s.jsxs)(d.td,{children:["Current user is already authenticated. De-authenticate the current user with ",(0,s.jsx)(d.a,{href:"#clearuserdataasync",children:(0,s.jsx)(d.code,{children:"ClearUserDataAsync"})}),", and re-initialize the SDK by calling ",(0,s.jsx)(d.a,{href:"#shutdownasync",children:(0,s.jsx)(d.code,{children:"ShutdownAsync"})})," followed by ",(0,s.jsx)(d.a,{href:"#initializeasync",children:(0,s.jsx)(d.code,{children:"InitializeAsync"})}),"."]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"authenticateuseremailasync",children:"AuthenticateUserEmailAsync"}),"\n",(0,s.jsx)(d.p,{children:"Completes email authentication for the current session by submitting the one-time code sent to the user's email address"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_AuthenticateUserEmailAsync.png",src:i(29010).A+"",width:"251",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_AuthenticateUserEmailAsync(FModioEmailAuthCode AuthenticationCode, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-37",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-58",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"AuthenticationCode"})}),(0,s.jsx)(d.td,{children:"User's authentication code"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code indicating if authentication was successful or not"})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-38",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserAlreadyAuthenticatedError"})}),(0,s.jsxs)(d.td,{children:["Current user is already authenticated. De-authenticate the current user with ",(0,s.jsx)(d.a,{href:"#clearuserdataasync",children:(0,s.jsx)(d.code,{children:"ClearUserDataAsync"})}),", and re-initialize the SDK by calling ",(0,s.jsx)(d.a,{href:"#shutdownasync",children:(0,s.jsx)(d.code,{children:"ShutdownAsync"})})," then ",(0,s.jsx)(d.a,{href:"#initializeasync",children:(0,s.jsx)(d.code,{children:"InitializeAsync"})}),"."]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"archivemodasync",children:"ArchiveModAsync"}),"\n",(0,s.jsx)(d.p,{children:"Archives a mod. This mod will no longer be able to be viewed or retrieved via the SDK, but it will still exist should you choose to restore it at a later date. Archiving is restricted to team managers and administrators only. Note that restoration and permanent deletion of a mod is possible only via web interface."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_ArchiveModAsync.png",src:i(6312).A+"",width:"227",height:"184"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_ArchiveModAsync(FModioModID Mod, FOnErrorOnlyDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-38",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"no-rate-limiting"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-59",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Mod"})}),(0,s.jsx)(d.td,{children:"The mod to be archived."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback providing a status code indicating success or failure of archiving the mod."})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"error-values-39",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InsufficientPermissions"})}),(0,s.jsx)(d.td,{children:"The authenticated user does not have permission to archive this mod. This action is restricted to team managers and administrators only."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"Couldn't connect to mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"Specified mod does not exist or was deleted"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"The supplied mod ID is invalid"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"addtotempmodset",children:"AddToTempModSet"}),"\n",(0,s.jsxs)(d.p,{children:["Add mods to a Temp Mod Set. Every temp mod specified by ",(0,s.jsx)(d.code,{children:"ModIds"})," will be installed if not already installed."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_AddToTempModSet.png",src:i(60006).A+"",width:"227",height:"154"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode K2_AddToTempModSet(TArray ModIds)\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-39",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"mod-management-enabled"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"temp-mod-set-initialized"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-60",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModIds"})}),(0,s.jsxs)(d.td,{children:["TArray of ",(0,s.jsx)(d.code,{children:"ModioModID"}),"s to install as temp mods"]})]})]})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-13",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["Error code indicating the status of the ",(0,s.jsx)(d.code,{children:"TempModSet"}),". Will be empty if it was successful"]}),"\n",(0,s.jsx)(d.h5,{id:"error-values-40",children:"Error Values"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"SDK not initialized"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,s.jsx)(d.td,{children:"Mod management not enabled"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TempModSetNotInitialized"})}),(0,s.jsxs)(d.td,{children:[(0,s.jsx)(d.code,{children:"TempModSet"})," not initialized"]})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"is-using-background-thread",children:"Is Using Background Thread"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_IsUsingBackgroundThread.png",src:i(62286).A+"",width:"243",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool IsUsingBackgroundThread()\n"})}),"\n",(0,s.jsx)(d.h4,{id:"is-mod-management-busy",children:"Is Mod Management Busy"}),"\n",(0,s.jsx)(d.p,{children:"Checks if the automatic management process is currently installing or removing mods"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_IsModManagementBusy.png",src:i(24896).A+"",width:"231",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool IsModManagementBusy()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-61",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-14",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"True if automatic management is currently performing an operation"}),"\n",(0,s.jsx)(d.h4,{id:"get-last-validation-error",children:"Get Last Validation Error"}),"\n",(0,s.jsx)(d.p,{children:"If the last request to the mod.io servers returned a validation failure, this function returns extended information describing the fields that failed validation."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_GetLastValidationError.png",src:i(18781).A+"",width:"227",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"TArray GetLastValidationError()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"requirements-40",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n"]}),"\n",(0,s.jsx)(d.h5,{id:"parameters-62",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.h5,{id:"returns-15",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["Collection of ",(0,s.jsx)(d.code,{children:"ModioValidationError"})," objects, or empty collection if there were no validation failures"]}),"\n",(0,s.jsx)(d.h4,{id:"disable-mod-management",children:"Disable Mod Management"}),"\n",(0,s.jsx)(d.p,{children:"Disables automatic installation or uninstallation of mods based on the user's subscriptions. Allows currently processing installation to complete. Will cancel any pending operations when called."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_DisableModManagement.png",src:i(23297).A+"",width:"231",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void DisableModManagement()\n"})}),"\n",(0,s.jsx)(d.h5,{id:"parameters-63",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModioSubsystem"})})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiotokenpacklibrary",children:"ModioTokenPackLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-14",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiouienumlocalizationlibrary",children:"ModioUIEnumLocalizationLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-15",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiouilocalizationlibrary",children:"ModioUILocalizationLibrary"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-16",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64library",children:"ModioUnsigned64Library"}),"\n",(0,s.jsx)(d.h5,{id:"inheritance-hierarchy-17",children:"Inheritance Hierarchy"}),"\n",(0,s.jsx)(d.p,{children:"-> BlueprintFunctionLibrary-> Object"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h2,{id:"structs",children:"Structs"}),"\n",(0,s.jsx)(d.h3,{id:"modiogameinfolist",children:"ModioGameInfoList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiogameinfo",children:"ModioGameInfo"}),"\n",(0,s.jsx)(d.h5,{id:"variables",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioGameID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameID"})}),(0,s.jsx)(d.td,{children:"Unique Game ID"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateAdded"})}),(0,s.jsx)(d.td,{children:"Unix timestamp of date the game was registered"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateUpdated"})}),(0,s.jsx)(d.td,{children:"Unix timestamp of date the game was updated"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateLive"})}),(0,s.jsx)(d.td,{children:"Unix timestamp of date the game was set live"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UgcName"})}),(0,s.jsx)(d.td,{children:"Word used to describe user-generated content (mods, items, add-ons etc)"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioIcon"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Icon"})}),(0,s.jsx)(d.td,{children:"Contains media URLs to the icon for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioLogo"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Logo"})}),(0,s.jsx)(d.td,{children:"Contains media URLs to the logo for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioHeaderImage"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"HeaderImage"})}),(0,s.jsx)(d.td,{children:"Contains media URLs to the preview header image for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Name"})}),(0,s.jsx)(d.td,{children:"Name of the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Summary"})}),(0,s.jsx)(d.td,{children:"Summary of the game's mod support"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Instructions"})}),(0,s.jsx)(d.td,{children:"A guide about creating and uploading mods for this game to mod.io"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InstructionsUrl"})}),(0,s.jsx)(d.td,{children:"Link to a mod.io guide, modding wiki, or a page where modders can learn how to make and submit mods to this game's profile"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileUrl"})}),(0,s.jsx)(d.td,{children:"URL to the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioTheme"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Theme"})}),(0,s.jsx)(d.td,{children:"Theme color values for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioGameStats"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Stats"})}),(0,s.jsx)(d.td,{children:"Numerous aggregate stats for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"OtherUrls"})}),(0,s.jsx)(d.td,{children:"Creator defined URLs to share"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Platforms"})}),(0,s.jsx)(d.td,{children:"Platforms that are supported by this title"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bAllowNegativeRatings"})}),(0,s.jsx)(d.td,{children:"Whether or not the game allows negative ratings"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EGameMonetizationFlags"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameMonetizationOptions"})}),(0,s.jsx)(d.td,{children:"Monetization options for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EGameMaturityFlags"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameMaturityOptions"})}),(0,s.jsx)(d.td,{children:"Maturity options for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"VirtualTokenName"})}),(0,s.jsx)(d.td,{children:"Name of the Virtual Tokens for this game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PlatformSupport"})}),(0,s.jsx)(d.td,{children:"Platforms that are supported by this title"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TagOptions"})}),(0,s.jsx)(d.td,{children:"Tags available for this game"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodtaginfo",children:"ModioModTagInfo"}),"\n",(0,s.jsx)(d.h5,{id:"variables-1",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TagGroupName"})}),(0,s.jsx)(d.td,{children:"Raw unlocalized tag group name"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TagGroupValues"})}),(0,s.jsx)(d.td,{children:"Valid raw unlocalized tag values this group contains"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TMap"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TagGroupNameLocalizationData"})}),(0,s.jsx)(d.td,{children:"Culture code -> localized tag category name mapping for all configured languages"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TagGroupValueLocalizationData"})}),(0,s.jsx)(d.td,{children:"Localization data for this tag category's values in all configured languages"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bAllowMultipleSelection"})}),(0,s.jsx)(d.td,{children:"True if multiple tags from the group can be used simultaneously"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bHidden"})}),(0,s.jsx)(d.td,{children:"True if only visible by admins"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bLocked"})}),(0,s.jsx)(d.td,{children:"TrueTrue if only editable by admins"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiolocalizedtagcategory",children:"ModioLocalizedTagCategory"}),"\n",(0,s.jsx)(d.p,{children:"Struct containing pre-localized display strings for a tag group"}),"\n",(0,s.jsx)(d.h5,{id:"variables-2",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GroupName"})}),(0,s.jsx)(d.td,{children:"Localized display string for this tag category's name"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Tags"})}),(0,s.jsx)(d.td,{children:"Localized display strings for all valid values in this tag category"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodtaglocalizationdata",children:"ModioModTagLocalizationData"}),"\n",(0,s.jsx)(d.p,{children:"Localization data for an individual tag value"}),"\n",(0,s.jsx)(d.h5,{id:"variables-3",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Tag"})}),(0,s.jsx)(d.td,{children:"The original raw unlocalized tag value used by the REST API"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TMap"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Translations"})}),(0,s.jsx)(d.td,{children:"Culture code -> Localized tag value string mapping for all configured languages."})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiogameplatform",children:"ModioGamePlatform"}),"\n",(0,s.jsx)(d.h5,{id:"variables-4",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioModfilePlatform"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Platform"})}),(0,s.jsx)(d.td,{children:"A platform supported by a title"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Locked"})}),(0,s.jsx)(d.td,{children:"Whether ot not this platform is locked from having files submitted to it by players"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Moderated"})}),(0,s.jsx)(d.td,{children:"Whether or not this platform's file submissions are moderated or not"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiootherurl",children:"ModioOtherUrl"}),"\n",(0,s.jsx)(d.h5,{id:"variables-5",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Label"})}),(0,s.jsx)(d.td,{children:"Label of the link you are sharing"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Url"})}),(0,s.jsx)(d.td,{children:"The URL to be associated with the label"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiogamestats",children:"ModioGameStats"}),"\n",(0,s.jsx)(d.p,{children:"Numerous aggregate stats for the game"}),"\n",(0,s.jsx)(d.h5,{id:"variables-6",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioGameID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameID"})}),(0,s.jsx)(d.td,{children:"Unique game id"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModCountTotal"})}),(0,s.jsx)(d.td,{children:"Available mod count for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModDownloadsToday"})}),(0,s.jsx)(d.td,{children:"Mods downloaded today for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModDownloadsTotal"})}),(0,s.jsx)(d.td,{children:"Total mods downloaded for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModDownloadsDailyAverage"})}),(0,s.jsx)(d.td,{children:"Average mods downloaded on a daily basis"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModSubscribersTotal"})}),(0,s.jsx)(d.td,{children:"Number of total users who have subscribed to the mods for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateExpires"})}),(0,s.jsx)(d.td,{children:"Unix timestamp until this game's statistics are considered stale"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiogameid",children:"ModioGameID"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiotheme",children:"ModioTheme"}),"\n",(0,s.jsx)(d.h5,{id:"variables-7",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Primary"})}),(0,s.jsx)(d.td,{children:"The primary hex color code"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Dark"})}),(0,s.jsx)(d.td,{children:"The dark hex color code"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Light"})}),(0,s.jsx)(d.td,{children:"The light hex color code"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Success"})}),(0,s.jsx)(d.td,{children:"The success hex color code"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Warning"})}),(0,s.jsx)(d.td,{children:"The warning hex color code"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Danger"})}),(0,s.jsx)(d.td,{children:"The danger hex color code"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioheaderimage",children:"ModioHeaderImage"}),"\n",(0,s.jsx)(d.h5,{id:"variables-8",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Filename"})}),(0,s.jsx)(d.td,{children:"Header image filename including extension"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Original"})}),(0,s.jsx)(d.td,{children:"URL to the full-sized header image"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiologo",children:"ModioLogo"}),"\n",(0,s.jsx)(d.h5,{id:"variables-9",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Filename"})}),(0,s.jsx)(d.td,{children:"Logo filename including extension."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Original"})}),(0,s.jsx)(d.td,{children:"URL to the full - sized logo."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb320x180"})}),(0,s.jsx)(d.td,{children:"URL to the small logo thumbnail."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb640x360"})}),(0,s.jsx)(d.td,{children:"URL to the medium logo thumbnail."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb1280x720"})}),(0,s.jsx)(d.td,{children:"URL to the large logo thumbnail."})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioicon",children:"ModioIcon"}),"\n",(0,s.jsx)(d.h5,{id:"variables-10",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Filename"})}),(0,s.jsx)(d.td,{children:"Icon filename including extension."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Original"})}),(0,s.jsx)(d.td,{children:"URL to the full-sized icon."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb64x64"})}),(0,s.jsx)(d.td,{children:"URL to the small icon thumbnail."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb128x128"})}),(0,s.jsx)(d.td,{children:"URL to the medium icon thumbnail."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb256x256"})}),(0,s.jsx)(d.td,{children:"URL to the large icon thumbnail."})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiopagedresult",children:"ModioPagedResult"}),"\n",(0,s.jsx)(d.h5,{id:"variables-11",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int32"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PageIndex"})}),(0,s.jsx)(d.td,{children:"The index of this page"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int32"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PageSize"})}),(0,s.jsx)(d.td,{children:"The amount of results allowed to be displayed within a page"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int32"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PageCount"})}),(0,s.jsx)(d.td,{children:"The total amount of pages"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int32"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TotalResultCount"})}),(0,s.jsx)(d.td,{children:"The total amount of results"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int32"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ResultCount"})}),(0,s.jsx)(d.td,{children:"The amount of results for this page"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodinfolist",children:"ModioModInfoList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodinfo",children:"ModioModInfo"}),"\n",(0,s.jsx)(d.h5,{id:"variables-12",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioModID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModId"})}),(0,s.jsx)(d.td,{children:"Unique Mod ID"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileName"})}),(0,s.jsx)(d.td,{children:"Name of the mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileSummary"})}),(0,s.jsx)(d.td,{children:"Summary of the mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileDescription"})}),(0,s.jsx)(d.td,{children:"Detailed description in HTML format"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileDescriptionPlaintext"})}),(0,s.jsx)(d.td,{children:"Detailed description in plaintext"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileURL"})}),(0,s.jsx)(d.td,{children:"URL to the mod profile"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioUser"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileSubmittedBy"})}),(0,s.jsx)(d.td,{children:"Information on the user who submitted the mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileDateAdded"})}),(0,s.jsx)(d.td,{children:"Unix timestamp of the date the mod was registered"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileDateUpdated"})}),(0,s.jsx)(d.td,{children:"Unix timestamp of the date the mod was updated"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileDateLive"})}),(0,s.jsx)(d.td,{children:"Unix timestamp of the date the mod was marked live"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioMaturityFlags"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileMaturityOption"})}),(0,s.jsx)(d.td,{children:"Flags for maturity options. Maturity options are flagged by the mod developer. This is only relevant if the parent game allows mods to be labeled as mature."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.del,{children:(0,s.jsx)(d.code,{children:"bVisible_DEPRECATED"})})}),(0,s.jsxs)(d.td,{children:["Deprecated as of 2023.6 release. Please use ",(0,s.jsx)(d.code,{children:"EModioObjectVisibilityFlags Visibility"})," instead."]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioObjectVisibilityFlags"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Visibility"})}),(0,s.jsx)(d.td,{children:"Enum parameter to signal the backend if the mod to upload would be publicly visible. Default value is Public"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Dependencies"})}),(0,s.jsx)(d.td,{children:"If this mod has any dependencies"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MetadataBlob"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioFileMetadata"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FileInfo"})}),(0,s.jsx)(d.td,{children:"Information about the mod's most recent public release"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MetadataKvp"})}),(0,s.jsx)(d.td,{children:"Arbitrary key-value metadata set for this mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Tags"})}),(0,s.jsx)(d.td,{children:"Tags this mod has set"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int32"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NumGalleryImages"})}),(0,s.jsx)(d.td,{children:"Number of images in the mod's media gallery"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioYoutubeURLList"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"YoutubeURLs"})}),(0,s.jsx)(d.td,{children:"List of youtube links provided by the creator of the mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioSketchfabURLList"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SketchfabURLs"})}),(0,s.jsx)(d.td,{children:"List of sketchfab links provided by the creator of the mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioModStats"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Stats"})}),(0,s.jsx)(d.td,{children:"Stats and rating information for the mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioModServerSideStatus"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModStatus"})}),(0,s.jsx)(d.td,{children:"Status of the mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioUnsigned64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Price"})}),(0,s.jsx)(d.td,{children:"Price of this mod"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64",children:"ModioUnsigned64"}),"\n",(0,s.jsx)(d.p,{children:"Trivial Blueprint-compatible wrapper around an unsigned 64-bit integer"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodstats",children:"ModioModStats"}),"\n",(0,s.jsx)(d.p,{children:"Contains download stats and ratings for a mod"}),"\n",(0,s.jsx)(d.h5,{id:"variables-13",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PopularityRankPosition"})}),(0,s.jsx)(d.td,{children:"Current rank of the mod."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PopularityRankTotalMods"})}),(0,s.jsx)(d.td,{children:"Number of ranking spots the current rank is measured against."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DownloadsTotal"})}),(0,s.jsx)(d.td,{children:"Number of total mod downloads."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SubscribersTotal"})}),(0,s.jsx)(d.td,{children:"Number of total users who have subscribed to the mod."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RatingTotal"})}),(0,s.jsx)(d.td,{children:"Number of times this mod has been rated."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RatingPositive"})}),(0,s.jsx)(d.td,{children:"Number of positive ratings."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RatingNegative"})}),(0,s.jsx)(d.td,{children:"Number of negative ratings."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RatingPercentagePositive"})}),(0,s.jsx)(d.td,{children:"Number of positive ratings, divided by the total ratings to determine it\u2019s percentage score."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"float"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RatingWeightedAggregate"})}),(0,s.jsxs)(d.td,{children:["Overall rating of this item calculated using the ",(0,s.jsx)(d.a,{href:"https://www.evanmiller.org/how-not-to-sort-by-average-Ratinghtml",children:"Wilson score confidence interval"}),". This column is good to sort on, as it will order items based on number of ratings and will place items with many positive ratings above those with a higher score but fewer ratings. We actually get a double back from the server, but it's converted to a float for blueprint support"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RatingDisplayText"})}),(0,s.jsx)(d.td,{children:"Textual representation of the rating in format: Overwhelmingly Positive -> Very Positive -> Positive -> Mostly Positive -> Mixed -> Negative -> Mostly Negative -> Very Negative -> Overwhelmingly Negative -> Unrated"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiosketchfaburllist",children:"ModioSketchfabURLList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioyoutubeurllist",children:"ModioYoutubeURLList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodtag",children:"ModioModTag"}),"\n",(0,s.jsx)(d.h5,{id:"variables-14",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Tag"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FText"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TagLocalized"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiometadata",children:"ModioMetadata"}),"\n",(0,s.jsx)(d.h5,{id:"variables-15",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Key"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Value"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiofilemetadata",children:"ModioFileMetadata"}),"\n",(0,s.jsx)(d.p,{children:"Metadata for a release archive for a mod"}),"\n",(0,s.jsx)(d.h5,{id:"variables-16",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioFileMetadataID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MetadataId"})}),(0,s.jsx)(d.td,{children:"Unique modfile id."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioModID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModId"})}),(0,s.jsx)(d.td,{children:"Unique mod id."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateAdded"})}),(0,s.jsx)(d.td,{children:"Unix timestamp of date file was added."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioVirusScanStatus"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"CurrentVirusScanStatus"})}),(0,s.jsx)(d.td,{children:"Current virus scan status of the file. For newly added files that have yet to be scanned this field will change frequently until a scan is complete"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioVirusStatus"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"CurrentVirusStatus"})}),(0,s.jsx)(d.td,{children:"Was a virus detected?"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Filesize"})}),(0,s.jsx)(d.td,{children:"Size of the file in bytes."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FilesizeUncompressed"})}),(0,s.jsx)(d.td,{children:"Total size of all files in the mod after installation."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Filename"})}),(0,s.jsx)(d.td,{children:"Filename including extension."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Version"})}),(0,s.jsx)(d.td,{children:"Release version this file represents."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Changelog"})}),(0,s.jsx)(d.td,{children:"Changelog for the file."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MetadataBlob"})}),(0,s.jsx)(d.td,{children:"Metadata stored by the game developer for this file."})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodid",children:"ModioModID"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiofilemetadataid",children:"ModioFileMetadataID"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiouser",children:"ModioUser"}),"\n",(0,s.jsx)(d.h5,{id:"variables-17",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioUserID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserId"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Username"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateOnline"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ProfileUrl"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DisplayNamePortal"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiouserid",children:"ModioUserID"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodtagoptions",children:"ModioModTagOptions"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiotokenpacklist",children:"ModioTokenPackList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiotokenpack",children:"ModioTokenPack"}),"\n",(0,s.jsx)(d.p,{children:"A wrapper type around the Unreal Engine type FOnlineStoreOffer, representing a platform-agnostic store offer/product, e.g a Premium Currency pack."}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioerrorcode",children:"ModioErrorCode"}),"\n",(0,s.jsxs)(d.p,{children:["Wrapper around ",(0,s.jsx)(d.code,{children:"Modio::ErrorCode"})]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalgameinfo",children:"ModioOptionalGameInfo"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalimage",children:"ModioOptionalImage"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalmoddependencylist",children:"ModioOptionalModDependencyList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalmodinfo",children:"ModioOptionalModInfo"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalmodtagoptions",children:"ModioOptionalModTagOptions"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalterms",children:"ModioOptionalTerms"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionaluint64",children:"ModioOptionalUInt64"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalmodinfolist",children:"ModioOptionalModInfoList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalgameinfolist",children:"ModioOptionalGameInfoList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodmanagementevent",children:"ModioModManagementEvent"}),"\n",(0,s.jsx)(d.p,{children:"Simple struct representing the outcome of a mod management operation"}),"\n",(0,s.jsx)(d.h5,{id:"variables-18",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioModID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ID"})}),(0,s.jsx)(d.td,{children:"ID for the mod that the event occurred on"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioModManagementEventType"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Event"})}),(0,s.jsx)(d.td,{children:"What type of event occurred"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioErrorCode"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Status"})}),(0,s.jsx)(d.td,{children:"Empty if operation completed successfully, truthy/contains error code if operation failed"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionaluserlist",children:"ModioOptionalUserList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalmodchangemap",children:"ModioOptionalModChangeMap"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionaltransactionrecord",children:"ModioOptionalTransactionRecord"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalentitlementconsumptionstatuslist",children:"ModioOptionalEntitlementConsumptionStatusList"}),"\n",(0,s.jsx)(d.p,{children:"Optional class representing a list of entitlement consumption statuses that may be a page from a larger set of results"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalmodid",children:"ModioOptionalModID"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioauthenticationparams",children:"ModioAuthenticationParams"}),"\n",(0,s.jsx)(d.p,{children:"Simple struct to encapsulate data passed to external authentication systems"}),"\n",(0,s.jsx)(d.h5,{id:"variables-19",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"AuthToken"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserEmail"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bUserHasAcceptedTerms"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TMap"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ExtendedParameters"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioapikey",children:"ModioApiKey"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioguid",children:"ModioGuid"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalguid",children:"ModioOptionalGuid"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioemailaddress",children:"ModioEmailAddress"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioemailauthcode",children:"ModioEmailAuthCode"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioentitlementparams",children:"ModioEntitlementParams"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiometricssessionparams",children:"ModioMetricsSessionParams"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiotokenpackid",children:"ModioTokenPackID"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiocreatemodfileparams",children:"ModioCreateModFileParams"}),"\n",(0,s.jsx)(d.h5,{id:"variables-20",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PathToModRootDirectory"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiocreatemodparams",children:"ModioCreateModParams"}),"\n",(0,s.jsx)(d.h5,{id:"variables-21",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PathToLogoFile"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Name"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Summary"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioeditmodparams",children:"ModioEditModParams"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioentitlementwalletbalance",children:"ModioEntitlementWalletBalance"}),"\n",(0,s.jsx)(d.p,{children:"Updated wallet balance from the sync entitlements call"}),"\n",(0,s.jsx)(d.h5,{id:"variables-22",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioUnsigned64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Balance"})}),(0,s.jsx)(d.td,{children:"The updated balance of the wallet"})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalentitlementwalletbalance",children:"ModioOptionalEntitlementWalletBalance"}),"\n",(0,s.jsx)(d.p,{children:"Optional updated wallet balance from the sync entitlements call"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioentitlementconsumptionvirtualcurrencydetails",children:"ModioEntitlementConsumptionVirtualCurrencyDetails"}),"\n",(0,s.jsx)(d.p,{children:"Further details about a Virtual Currency entitlement that was consumed"}),"\n",(0,s.jsx)(d.h5,{id:"variables-23",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int32"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TokensAllocated"})}),(0,s.jsx)(d.td,{children:"Amount of tokens that were issued for this specific entitlement consumption"})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"entitlementconsumptionstatus",children:"EntitlementConsumptionStatus"}),"\n",(0,s.jsx)(d.p,{children:"The result of an entitlement's consumption"}),"\n",(0,s.jsx)(d.h5,{id:"variables-24",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TransactionId"})}),(0,s.jsx)(d.td,{children:"ID of the transaction to redeem this entitlement"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioEntitlementConsumptionState"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TransactionState"})}),(0,s.jsx)(d.td,{children:"State of the transaction"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SkuId"})}),(0,s.jsx)(d.td,{children:"ID of the SKU that we attempted to consume"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntitlementConsumed"})}),(0,s.jsx)(d.td,{children:"Whether this entitlement was consumed or not"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioEntitlementType"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntitlementType"})}),(0,s.jsx)(d.td,{children:"Type of Entitlement that was consumed"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioEntitlementConsumptionVirtualCurrencyDetails"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"VirtualCurrencyDetails"})}),(0,s.jsx)(d.td,{children:"Details about virtual currency entitlement consumption"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioentitlementconsumptionstatuslist",children:"ModioEntitlementConsumptionStatusList"}),"\n",(0,s.jsx)(d.p,{children:"Class representing a list of entitlement consumption statuses that may be a page from a larger set of results"}),"\n",(0,s.jsx)(d.h5,{id:"variables-25",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioPagedResult"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PagedResult"})}),(0,s.jsx)(d.td,{children:"Stored property for a paged result, which provides context on the entitlement consumption status list"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InternalList"})}),(0,s.jsx)(d.td,{children:"Stored property for the dependency list"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioOptionalEntitlementWalletBalance"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"WalletBalance"})}),(0,s.jsx)(d.td,{children:"Updated wallet balance from syncing entitlements"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiofilterparams",children:"ModioFilterParams"}),"\n",(0,s.jsxs)(d.p,{children:["Class storing a set of filter parameters for use in ",(0,s.jsx)(d.a,{href:"#listallmodsasync",children:(0,s.jsx)(d.code,{children:"ListAllModsAsync"})})]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiopresetfilterparams",children:"ModioPresetFilterParams"}),"\n",(0,s.jsx)(d.p,{children:"Helper struct for named preset filter parameters"}),"\n",(0,s.jsx)(d.h5,{id:"variables-26",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FText"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PresetName"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Tags"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ExcludedTags"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioSortDirection"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Direction"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioSortFieldType"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SortField"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Count"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioimagewrapper",children:"ModioImageWrapper"}),"\n",(0,s.jsx)(d.h5,{id:"variables-27",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ImagePath"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioinitializeoptions",children:"ModioInitializeOptions"}),"\n",(0,s.jsx)(d.h5,{id:"variables-28",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioGameID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameId"})}),(0,s.jsx)(d.td,{children:"The mod.io-provided ID for the game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioApiKey"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ApiKey"})}),(0,s.jsx)(d.td,{children:"The mod.io-provided API key for your application or game"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioEnvironment"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameEnvironment"})}),(0,s.jsx)(d.td,{children:"The mod.io environment you want to run the SDK on"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioPortal"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PortalInUse"})}),(0,s.jsx)(d.td,{children:"The portal your title is running through"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TMap"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ExtendedInitializationParameters"})}),(0,s.jsx)(d.td,{children:"Extended platform-specific initialization parameters. Refer to the platform documentation for valid keys and their values. Unrecognized values will be ignored"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bUseBackgroundThread"})}),(0,s.jsx)(d.td,{children:"Set mod.io to run with background thread"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodchangemap",children:"ModioModChangeMap"}),"\n",(0,s.jsx)(d.h5,{id:"variables-29",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TMap"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Changes"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodcollectionentry",children:"ModioModCollectionEntry"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodcreationhandle",children:"ModioModCreationHandle"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomoddependency",children:"ModioModDependency"}),"\n",(0,s.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,s.jsx)(d.p,{children:"This class is part of an experimental feature and is subject to change."})}),"\n",(0,s.jsx)(d.h5,{id:"variables-30",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioModID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModID"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModName"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateAdded"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FDateTime"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateUpdated"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"uint8"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DependencyDepth"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioLogo"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Logo"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioFileMetadata"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FileInfo"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioModServerSideStatus"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Status"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EModioObjectVisibilityFlags"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Visibility"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomoddependencylist",children:"ModioModDependencyList"}),"\n",(0,s.jsx)(d.admonition,{title:"Experimental Feature",type:"warning",children:(0,s.jsx)(d.p,{children:"This class is part of an experimental feature and is subject to change."})}),"\n",(0,s.jsx)(d.h5,{id:"variables-31",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioPagedResult"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PagedResult"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InternalList"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TotalFilesize"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"int64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TotalFilesizeUncompressed"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodprogressinfo",children:"ModioModProgressInfo"}),"\n",(0,s.jsx)(d.h5,{id:"variables-32",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioModID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ID"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionalmodprogressinfo",children:"ModioOptionalModProgressInfo"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioreportparams",children:"ModioReportParams"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiolink",children:"ModioLink"}),"\n",(0,s.jsx)(d.h5,{id:"variables-33",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Text"})}),(0,s.jsx)(d.td,{children:"The user-facing text for the link"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"URL"})}),(0,s.jsx)(d.td,{children:"The actual URL for the link"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bool"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bRequired"})}),(0,s.jsx)(d.td,{children:"Is displaying this link mandatory?"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modioterms",children:"ModioTerms"}),"\n",(0,s.jsx)(d.h5,{id:"variables-34",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"AgreeButtonText"})}),(0,s.jsx)(d.td,{children:"Text to display on the affirmative/OK button"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DisagreeButtonText"})}),(0,s.jsx)(d.td,{children:"Text to display on the negative/cancel button"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioLink"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"WebsiteLink"})}),(0,s.jsx)(d.td,{children:"Link to the mod.io website"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioLink"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TermsLink"})}),(0,s.jsx)(d.td,{children:"Link to the mod.io terms of use"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioLink"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PrivacyLink"})}),(0,s.jsx)(d.td,{children:"Link to the mod.io Privacy Policy"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioLink"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ManageLink"})}),(0,s.jsx)(d.td,{children:"Link to the mod.io Manage User Account page"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TermsText"})}),(0,s.jsx)(d.td,{children:"The plaintext version of the mod.io terms of use"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionaltokenpack",children:"ModioOptionalTokenPack"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionaltokenpacklist",children:"ModioOptionalTokenPackList"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiotransactionrecord",children:"ModioTransactionRecord"}),"\n",(0,s.jsx)(d.h5,{id:"variables-35",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioModID"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"AssociatedModID"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioUnsigned64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Price"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioUnsigned64"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UpdatedUserWalletBalance"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiooptionaluser",children:"ModioOptionalUser"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiouserlist",children:"ModioUserList"}),"\n",(0,s.jsx)(d.h5,{id:"variables-36",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FModioPagedResult"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PagedResult"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InternalList"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiovalidationerror",children:"ModioValidationError"}),"\n",(0,s.jsx)(d.p,{children:"Wrapper struct containing information about a field validation error"}),"\n",(0,s.jsx)(d.h5,{id:"variables-37",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FieldName"})}),(0,s.jsx)(d.td,{children:"String description of the field that failed validation"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FString"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ValidationFailureDescription"})}),(0,s.jsx)(d.td,{children:"String description of the validation failure"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiocreatemodfilememoryparams",children:"ModioCreateModFileMemoryParams"}),"\n",(0,s.jsx)(d.h5,{id:"variables-38",children:"Variables"}),"\n",(0,s.jsx)(n,{colWidths:["10%","10%","20%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TArray"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModMemory"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h2,{id:"functions",children:"Functions"}),"\n",(0,s.jsx)(d.h3,{id:"set-session-identifier",children:"Set Session Identifier"}),"\n",(0,s.jsxs)(d.p,{children:["Changes the session identifier for the provided set of ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetSessionIdentifier.png",src:i(73569).A+"",width:"297",height:"111"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetSessionIdentifier(FModioInitializeOptions Options, FString SessionIdentifier)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-64",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Options"})}),(0,s.jsxs)(d.td,{children:["The template ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionIdentifier"})}),(0,s.jsx)(d.td,{children:"The new session id to use"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-16",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["New ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the session identifier set to the desired value"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-session-id",children:"Set Session Id"}),"\n",(0,s.jsx)(d.p,{children:"Changes the session identifier for the provided Metrics Session parameters"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetSessionId.png",src:i(9731).A+"",width:"224",height:"106"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioMetricsSessionParams SetSessionId(FModioMetricsSessionParams Params, FModioGuid Id)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-65",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Params"})}),(0,s.jsx)(d.td,{children:"The template Metrics Session parameters"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Id"})}),(0,s.jsx)(d.td,{children:"The intended Guid to store in the Metrics Session parameters"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-portal",children:"Set Portal"}),"\n",(0,s.jsxs)(d.p,{children:["Changes the portal for the provided set of ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetPortal.png",src:i(21880).A+"",width:"379",height:"134"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetPortal(FModioInitializeOptions Options, EModioPortal PortalToUse)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-66",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Options"})}),(0,s.jsxs)(d.td,{children:["The template ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PortalToUse"})}),(0,s.jsx)(d.td,{children:"The new portal to use"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-17",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["New ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the portal set to the desired value"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-game-id",children:"Set Game Id"}),"\n",(0,s.jsxs)(d.p,{children:["Changes the game id for the provided set of ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetGameId.png",src:i(45671).A+"",width:"249",height:"111"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetGameId(FModioInitializeOptions Options, int64 GameId)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-67",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Options"})}),(0,s.jsxs)(d.td,{children:["The template ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameId"})}),(0,s.jsx)(d.td,{children:"The new game id to use"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-18",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["New ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the game id set to the desired value"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-game-environment",children:"Set Game Environment"}),"\n",(0,s.jsxs)(d.p,{children:["Changes the game environment for the provided set of ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetGameEnvironment.png",src:i(80201).A+"",width:"379",height:"134"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetGameEnvironment(FModioInitializeOptions Options, EModioEnvironment GameEnvironment)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-68",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Options"})}),(0,s.jsxs)(d.td,{children:["The template ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameEnvironment"})}),(0,s.jsx)(d.td,{children:"The new environment to use"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-19",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["New ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the game environment set to the desired value"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-extended-initialization-parameters",children:"Set Extended Initialization Parameters"}),"\n",(0,s.jsxs)(d.p,{children:["Sets extended initialization parameters for the provided set of ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetExtendedInitializationParameters.png",src:i(38945).A+"",width:"304",height:"108"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetExtendedInitializationParameters(FModioInitializeOptions Options, TMap ExtendedParameters)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-69",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Options"})}),(0,s.jsxs)(d.td,{children:["The template ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ExtendedParameters"})}),(0,s.jsx)(d.td,{children:"The new extended parameters to use (will overwrite existing values)"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-20",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["New ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the extended parameters set to the desired value"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-background-thread",children:"Set Background Thread"}),"\n",(0,s.jsxs)(d.p,{children:["Changes the background thread usage for the provided set of ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetBackgroundThread.png",src:i(92422).A+"",width:"312",height:"124"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetBackgroundThread(FModioInitializeOptions Options, bool bUseBackgroundThread)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-70",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Options"})}),(0,s.jsxs)(d.td,{children:["The template ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bUseBackgroundThread"})}),(0,s.jsx)(d.td,{children:"Whether to use a background thread"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-21",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["New ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the background thread usage set to the desired value"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-api-key",children:"Set API Key"}),"\n",(0,s.jsxs)(d.p,{children:["Changes the API key for the provided set of ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_SetAPIKey.png",src:i(16659).A+"",width:"244",height:"111"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions SetAPIKey(FModioInitializeOptions Options, FString APIKey)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-71",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Options"})}),(0,s.jsxs)(d.td,{children:["The template ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"APIKey"})}),(0,s.jsx)(d.td,{children:"The new API key to use"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-22",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["New ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})})," object with the API key set to the desired value"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodid--modiomodid",children:"ModioModID != ModioModID"}),"\n",(0,s.jsx)(d.p,{children:"Compares two mod IDs to check whether they're not equal"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_NotEqualTo.png",src:i(42690).A+"",width:"159",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool NotEqualTo(FModioModID A, FModioModID B)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-72",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"A"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"B"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"make-metrics-session-params",children:"Make Metrics Session Params"}),"\n",(0,s.jsx)(d.p,{children:"Create Metrics Session parameters"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeMetricsSessionParams.png",src:i(80709).A+"",width:"258",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioMetricsSessionParams MakeMetricsSessionParams(TArray Ids)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-73",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Ids"})}),(0,s.jsx)(d.td,{children:"The list of mods to store within the Metrics Session parameters"})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"make-initialize-options",children:"Make Initialize Options"}),"\n",(0,s.jsxs)(d.p,{children:["Make ",(0,s.jsx)(d.a,{href:"#modioinitializeoptions",children:(0,s.jsx)(d.code,{children:"ModioInitializeOptions"})}),". Should only be used in conjunction with ",(0,s.jsx)(d.a,{href:"#initializeasync",children:(0,s.jsx)(d.code,{children:"InitializeAsync"})}),"."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeInitializeOptions.png",src:i(50900).A+"",width:"379",height:"280"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions MakeInitializeOptions(int64 GameId, FString APIKey, EModioEnvironment GameEnvironment, EModioPortal PortalInUse, bool bUseBackgroundThread)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-74",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameId"})}),(0,s.jsxs)(d.td,{children:["A positive integer that maps to your game. This can be found in the admin section of your game's page at ",(0,s.jsx)(d.a,{href:"https://mod.io/",children:"https://mod.io/"})]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"APIKey"})}),(0,s.jsxs)(d.td,{children:["The API key for your game. This can be found in the admin section of your game's page at ",(0,s.jsx)(d.a,{href:"https://mod.io/",children:"https://mod.io/"})]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameEnvironment"})}),(0,s.jsx)(d.td,{children:"The environment your game has been set up on: test or live."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PortalInUse"})}),(0,s.jsxs)(d.td,{children:["The ",(0,s.jsx)(d.a,{href:"#EModioPortal",children:(0,s.jsx)(d.code,{children:"EModioPortal"})})," representing the store or service your game is being"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bUseBackgroundThread"})}),(0,s.jsxs)(d.td,{children:["Whether to run the mod.io SDK on a background thread. Defaults to ",(0,s.jsx)(d.code,{children:"true"}),". distributed through. Defaults to ",(0,s.jsx)(d.code,{children:"EModioPortal::None"}),"."]})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"make-guid",children:"Make Guid"}),"\n",(0,s.jsx)(d.p,{children:"Create a Guid from a string"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeGuid.png",src:i(1237).A+"",width:"230",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioGuid MakeGuid(FString Guid)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-75",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Guid"})}),(0,s.jsx)(d.td,{children:"A string to wrap within a Guid struct"})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"make-game-id",children:"Make Game Id"}),"\n",(0,s.jsxs)(d.p,{children:["Create a game id from a integer. Should only be used in conjunction with ",(0,s.jsx)(d.a,{href:"#initializeasync",children:(0,s.jsx)(d.code,{children:"InitializeAsync"})})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeGameId.png",src:i(94923).A+"",width:"249",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioGameID MakeGameId(int64 GameId)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-76",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameId"})}),(0,s.jsx)(d.td,{children:"A positive integer that maps to your game"})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"make-entitlement-params",children:"Make Entitlement Params"}),"\n",(0,s.jsx)(d.p,{children:"Create entitlement parameters"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeEntitlementParams.png",src:i(21735).A+"",width:"303",height:"78"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioEntitlementParams MakeEntitlementParams(TMap ExtendedParameters)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-77",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ExtendedParameters"})}),(0,s.jsx)(d.td,{children:"A map to store extended parameters required by some portals"})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"make-auth-params",children:"Make Auth Params"}),"\n",(0,s.jsxs)(d.p,{children:["Creates a ",(0,s.jsx)(d.code,{children:"ModioAuthenticationParams"})," object"]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeAuthParams.png",src:i(36342).A+"",width:"306",height:"150"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioAuthenticationParams MakeAuthParams(FString AuthToken, FString EmailAddress, bool bHasAcceptedTOS)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-78",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"AuthToken"})}),(0,s.jsx)(d.td,{children:"Authentication provider-supplied OAuth token"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EmailAddress"})}),(0,s.jsx)(d.td,{children:"User email address, can be left blank"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bHasAcceptedTOS"})}),(0,s.jsx)(d.td,{children:"Has the user been shown the Terms of Service and accepted the Terms of Service?"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-23",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["The constructed ",(0,s.jsx)(d.code,{children:"ModioAuthenticationParams"})," object for use with ",(0,s.jsx)(d.a,{href:"#authenticateuserexternalasync",children:(0,s.jsx)(d.code,{children:"AuthenticateUserExternalAsync"})})]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"make-api-key",children:"Make Api Key"}),"\n",(0,s.jsxs)(d.p,{children:["Create an ApiKey id from a string. Should only be used in conjunction with ",(0,s.jsx)(d.a,{href:"#initializeasync",children:(0,s.jsx)(d.code,{children:"InitializeAsync"})})]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_MakeApiKey.png",src:i(89887).A+"",width:"246",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioApiKey MakeApiKey(FString ApiKey)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-79",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ApiKey"})}),(0,s.jsx)(d.td,{children:"The api key from your settings panel on mod.io"})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-raw-value-from-mod-id",children:"Get Raw Value from Mod ID"}),"\n",(0,s.jsxs)(d.p,{children:["Retrieves the raw underlying value from an ",(0,s.jsx)(d.code,{children:"ModioModID"}),". ",(0,s.jsx)(d.code,{children:"ModioModID"}),"s are intended as opaque types, so use with care."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_GetRawValueFromModID.png",src:i(59892).A+"",width:"240",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"int64 GetRawValueFromModID(FModioModID In)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-80",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"In"})}),(0,s.jsxs)(d.td,{children:["The ",(0,s.jsx)(d.code,{children:"ModioModID"})," to retrieve the value for"]})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-24",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"The underlying value"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiomodid--modiomodid-1",children:"ModioModID == ModioModID"}),"\n",(0,s.jsx)(d.p,{children:"Compares two mod IDs to check whether they're equal"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCommonTypesLibrary_EqualTo.png",src:i(73149).A+"",width:"159",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool EqualTo(FModioModID A, FModioModID B)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-81",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"A"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"B"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-version-string",children:"Set Version String"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetVersionString.png",src:i(71105).A+"",width:"186",height:"143"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetVersionString(FModioCreateModFileParams In, FString Version)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-tags",children:"Set Tags"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetTags.png",src:i(40007).A+"",width:"137",height:"138"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetTags(FModioCreateModParams In, TArray Tags)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-modfile-platforms",children:"Set Modfile Platforms"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetModfilePlatforms.png",src:i(94646).A+"",width:"208",height:"138"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetModfilePlatforms(FModioCreateModFileParams In, TArray Platforms)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-mod-file-metadata-blob",children:"Set Mod File Metadata Blob"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetModFileMetadataBlob.png",src:i(32566).A+"",width:"239",height:"143"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetModFileMetadataBlob(FModioCreateModFileParams In, FString MetadataBlob)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-metadata-blob",children:"Set Metadata Blob"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetMetadataBlob.png",src:i(10318).A+"",width:"212",height:"143"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetMetadataBlob(FModioCreateModParams In, FString MetadataBlob)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-mark-as-active-release",children:"Set Mark as Active Release"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetMarkAsActiveRelease.png",src:i(74678).A+"",width:"258",height:"142"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetMarkAsActiveRelease(FModioCreateModFileParams In, bool bMarkAsActiveRelease)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-initial-visibility-deprecated",children:"Set Initial Visibility DEPRECATED"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetInitialVisibility_DEPRECATED.png",src:i(65268).A+"",width:"269",height:"142"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetInitialVisibility_DEPRECATED(FModioCreateModParams In, bool InitialVisibility)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-initial-visibility",children:"Set Initial Visibility"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetInitialVisibility.png",src:i(40594).A+"",width:"306",height:"166"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetInitialVisibility(FModioCreateModParams In, EModioObjectVisibilityFlags InitialVisibility)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-homepage-url",children:"Set Homepage URL"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetHomepageURL.png",src:i(58589).A+"",width:"216",height:"143"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetHomepageURL(FModioCreateModParams In, FString HomepageURL)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-description",children:"Set Description"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetDescription.png",src:i(21900).A+"",width:"193",height:"143"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetDescription(FModioCreateModParams In, FString Description)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"set-changelog-string",children:"Set Changelog String"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioCreateModLibrary_SetChangelogString.png",src:i(62283).A+"",width:"202",height:"143"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void SetChangelogString(FModioCreateModFileParams In, FString Changelog)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-localized-text-for-enum-by-name",children:"Get Localized Text for Enum by Name"}),"\n",(0,s.jsxs)(d.p,{children:["Returns the string table ",(0,s.jsx)(d.code,{children:"FText"})," for a given enum value's ",(0,s.jsx)(d.code,{children:"FName"}),". Only works with enums registered via ",(0,s.jsx)(d.code,{children:"ModioUI::RegisterEnumAsLocalizable"}),"."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUIEnumLocalizationLibrary_GetLocalizedTextForEnumByName.png",src:i(89670).A+"",width:"298",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText GetLocalizedTextForEnumByName(FName EnumName)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-82",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EnumName"})}),(0,s.jsx)(d.td,{children:"The Name from a given enum value"})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-25",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"localized Text for the specified enum value, or dummy FText if not found"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"filesizetotext-unsigned64",children:"FileSizeToText (Unsigned64)"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUIEnumLocalizationLibrary_FileSizeUnsigned64_ToText.png",src:i(42575).A+"",width:"441",height:"197"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText FileSizeUnsigned64_ToText(FModioUnsigned64 FileSize, int32 MinDecimals, int32 MaxDecimals, TEnumAsByte Unit, bool bIncludeUnitName)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-83",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FileSize"})}),(0,s.jsx)(d.td,{children:"Filesize in bytes"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MinDecimals"})}),(0,s.jsx)(d.td,{children:"Minimum number of decimals to display for the filesize"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MaxDecimals"})}),(0,s.jsx)(d.td,{children:"Maximum number of decimals to display for the filesize"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Unit"})}),(0,s.jsxs)(d.td,{children:["If ",(0,s.jsx)(d.code,{children:"Largest"}),", it tries to display the size in the largest unit that will have a integral part > 0, else it displays the filesize in the specified unit"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bIncludeUnitName"})}),(0,s.jsx)(d.td,{children:"Whether or not to include the unit name"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-26",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["An ",(0,s.jsx)(d.code,{children:"FText"})," formatted with your specifications"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-localized-text-from-default-table-by-key",children:"Get Localized Text from Default Table by Key"}),"\n",(0,s.jsxs)(d.p,{children:["Returns the string table ",(0,s.jsx)(d.code,{children:"FText"})," for a given string key"]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUILocalizationLibrary_GetLocalizedTextFromDefaultTableByKey.png",src:i(39200).A+"",width:"342",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText GetLocalizedTextFromDefaultTableByKey(FString StringKey)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-84",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"StringKey"})}),(0,s.jsx)(d.td,{children:"The key to look up in the table"})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-27",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["Localized Text for the specified key, or ",(0,s.jsx)(d.code,{children:"StringKey"})," if not found"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"reconstruct-error",children:"Reconstruct Error"}),"\n",(0,s.jsx)(d.p,{children:"Helper method to reconstruct a mod.io error passed via code that cannot reference mod.io types"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioErrorCodeLibrary_ReconstructError.png",src:i(23039).A+"",width:"254",height:"116"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioErrorCode ReconstructError(int32 Value, int32 Category)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-85",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Value"})}),(0,s.jsx)(d.td,{children:"The numeric value of the code"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Category"})}),(0,s.jsx)(d.td,{children:"The category ID (populated by native code)"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"iserror",children:"IsError"}),"\n",(0,s.jsx)(d.p,{children:"Checks if an error code contains a error"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioErrorCodeLibrary_IsErrorAsExec.png",src:i(78906).A+"",width:"164",height:"110"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool IsErrorAsExec(FModioErrorCode Error)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-86",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Error"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-28",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"true if the error code is an error"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-value",children:"Get Value"}),"\n",(0,s.jsx)(d.p,{children:"Get underlying error code for an FModioErrorCode."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioErrorCodeLibrary_GetValue.png",src:i(85392).A+"",width:"208",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"int32 GetValue(FModioErrorCode Error)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-87",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Error"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-29",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"The underlying error code. 0 represents no error."}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-message",children:"Get Message"}),"\n",(0,s.jsx)(d.p,{children:"Get the textual representation of the error"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioErrorCodeLibrary_GetMessage.png",src:i(270).A+"",width:"208",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FString GetMessage(FModioErrorCode Error)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-88",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Error"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-30",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["An ",(0,s.jsx)(d.code,{children:"FString"})," message describing the error"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"error-code-matches",children:"Error Code Matches"}),"\n",(0,s.jsx)(d.p,{children:"Checks if the passed-in ErrorCode matches the specified error condition"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioErrorConditionLibrary_ErrorCodeMatches.png",src:i(4457).A+"",width:"379",height:"166"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool ErrorCodeMatches(FModioErrorCode ErrorCode, EModioErrorCondition Condition)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-89",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ErrorCode"})}),(0,s.jsx)(d.td,{children:"The code to check"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Condition"})}),(0,s.jsx)(d.td,{children:"The error condition to check against"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-31",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"True if the code matches the condition"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"list-user-subscription-async",children:"List User Subscription Async"}),"\n",(0,s.jsx)(d.p,{children:"Runs a filter over the user's subscription list"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioExampleLibrary_ListUserSubscriptionAsync.png",src:i(89e3).A+"",width:"249",height:"138"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void ListUserSubscriptionAsync(FModioFilterParams FilterParams, FOnListAllModsDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-90",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FilterParams"})}),(0,s.jsx)(d.td,{children:"The filters to use on the user's subscription list"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Called when mod list has been processed"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-logo-thumbnail-size",children:"Get Logo Thumbnail Size"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioExampleLibrary_GetLogoThumbnailSize.png",src:i(77120).A+"",width:"223",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioLogoSize GetLogoThumbnailSize()\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-logo-full-size",children:"Get Logo Full Size"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioExampleLibrary_GetLogoFullSize.png",src:i(15899).A+"",width:"184",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioLogoSize GetLogoFullSize()\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-avatar-thumbnail-size",children:"Get Avatar Thumbnail Size"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioExampleLibrary_GetAvatarThumbnailSize.png",src:i(58370).A+"",width:"234",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioAvatarSize GetAvatarThumbnailSize()\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"to-filter-params",children:"To Filter Params"}),"\n",(0,s.jsx)(d.p,{children:"Converts a preset filter into a concrete set of filter parameters that can be passed to the mod.io plugin"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioPresetFilterParamsLibrary_ToFilterParams.png",src:i(64454).A+"",width:"216",height:"108"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioFilterParams ToFilterParams(FModioPresetFilterParams Preset)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-91",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Preset"})}),(0,s.jsx)(d.td,{children:"The preset to convert"})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-32",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"The converted filter params"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"load-async",children:"Load Async"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_LoadAsync.png",src:i(60565).A+"",width:"200",height:"138"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void LoadAsync(FModioImageWrapper Image, FOnLoadImageDelegate OnImageLoaded)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-texture",children:"Get Texture"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetTexture.png",src:i(92108).A+"",width:"215",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"UTexture2DDynamic* GetTexture(FModioImageWrapper Image)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-state",children:"Get State"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetState.png",src:i(46998).A+"",width:"215",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioImageState GetState(FModioImageWrapper Image)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-logo-size",children:"Get Logo Size"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetLogoSize.png",src:i(63647).A+"",width:"379",height:"158"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FVector2D GetLogoSize(UTexture* Logo, EModioLogoSize LogoSize)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-92",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Logo"})}),(0,s.jsxs)(d.td,{children:["If null and ",(0,s.jsx)(d.code,{children:"EModioLogoSize::Original"})," is passed, then (0, 0) is returned"]})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"LogoSize"})}),(0,s.jsx)(d.td,{children:"The size of the logo we want to return"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-33",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"Dimensions of the logo if displayed in a 1:1 pixel ratio"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-gallery-size",children:"Get Gallery Size"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetGallerySize.png",src:i(24940).A+"",width:"379",height:"158"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FVector2D GetGallerySize(UTexture* GalleryImage, EModioGallerySize GallerySize)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-93",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GalleryImage"})}),(0,s.jsx)(d.td,{children:"If null and EModioGallerySize::Original is passed, then (0, 0) is returned"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GallerySize"})}),(0,s.jsx)(d.td,{children:"The size of the gallery image we want to return"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-34",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"Dimensions of the gallery image if displayed in a 1:1 pixel ratio"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-avatar-size",children:"Get Avatar Size"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioImageLibrary_GetAvatarSize.png",src:i(86557).A+"",width:"379",height:"158"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FVector2D GetAvatarSize(UTexture* Avatar, EModioAvatarSize AvatarSize)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-94",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Avatar"})}),(0,s.jsx)(d.td,{children:"If null and EModioAvatarSize::Original is passed, then (0, 0) is returned"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"AvatarSize"})}),(0,s.jsx)(d.td,{children:"The size of the avatar we want to return"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-35",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"Dimensions of the avatar if displayed in a 1:1 pixel ratio"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-path",children:"Get Path"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioModCollectionLibrary_GetPath.png",src:i(65953).A+"",width:"210",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FString GetPath(FModioModCollectionEntry Entry)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-95",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Entry"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-36",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:["Path to the mod's installation folder on disk. If the mod is not yet installed this path may not yet exist. Check ",(0,s.jsx)(d.code,{children:"GetModState"})," before trying to load files in this location"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-mod-state",children:"Get Mod State"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioModCollectionLibrary_GetModState.png",src:i(56543).A+"",width:"210",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioModState GetModState(FModioModCollectionEntry Entry)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-96",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Entry"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-37",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"EModioModState enum representing current state of the mod"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-mod-profile",children:"Get Mod Profile"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioModCollectionLibrary_GetModProfile.png",src:i(39711).A+"",width:"210",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioModInfo GetModProfile(FModioModCollectionEntry Entry)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-97",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Entry"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-38",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"FModioModInfo containing mod profile data"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-id",children:"Get ID"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioModCollectionLibrary_GetID.png",src:i(52915).A+"",width:"210",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioModID GetID(FModioModCollectionEntry Entry)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-98",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Entry"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-39",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"Mod ID"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-total-progress",children:"Get Total Progress"}),"\n",(0,s.jsx)(d.p,{children:"Retrieves the total amount of progress required for the specified state."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioModProgressInfoLibrary_GetTotalProgress.png",src:i(66964).A+"",width:"379",height:"134"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 GetTotalProgress(FModioModProgressInfo Info, EModioModProgressState State)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-99",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Info"})}),(0,s.jsx)(d.td,{children:"the progress struct to query"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"State"})}),(0,s.jsx)(d.td,{children:"which state to query total progress for"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-40",children:"Returns"}),"\n",(0,s.jsxs)(d.p,{children:[(0,s.jsx)(d.code,{children:"Modio::FileSize"})," for total progress in bytes"]}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-current-state",children:"Get Current State"}),"\n",(0,s.jsx)(d.p,{children:"Returns a EModioModProgressState indicating which state the mod operation is in"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioModProgressInfoLibrary_GetCurrentState.png",src:i(84009).A+"",width:"203",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioModProgressState GetCurrentState(FModioModProgressInfo Info)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-100",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Info"})}),(0,s.jsx)(d.td,{children:"The progress struct to query"})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-current-progress",children:"Get Current Progress"}),"\n",(0,s.jsxs)(d.p,{children:["Retrieves the progress value for the specified state. ",(0,s.jsx)(d.code,{children:"CurrentProgress == TotalProgress"})," for states which have completed, for example if a mod is currently Extracting, then passing in Downloading would give you a value equal to the total download size because the download has completed"]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioModProgressInfoLibrary_GetCurrentProgress.png",src:i(5873).A+"",width:"379",height:"134"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 GetCurrentProgress(FModioModProgressInfo Info, EModioModProgressState State)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-101",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Info"})}),(0,s.jsx)(d.td,{children:"the progress struct to query"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"State"})}),(0,s.jsx)(d.td,{children:"which state to query progress information for"})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-41",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"FModioUnsigned64 containing current progress in bytes"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-tags",children:"Get Tags"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioModTagOptionsLibrary_GetTags.png",src:i(7593).A+"",width:"232",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"TArray GetTags(FModioModTagOptions ModTags)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-102",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModTags"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-42",children:"Returns"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-paged-result",children:"Get Paged Result"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioModTagOptionsLibrary_GetPagedResult.png",src:i(17572).A+"",width:"236",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioPagedResult GetPagedResult(FModioModTagOptions ModTags)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-103",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModTags"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-43",children:"Returns"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-default-portal-for-current-platform",children:"Get Default Portal for Current Platform"}),"\n",(0,s.jsx)(d.p,{children:"Get the default portal for the platform the game is running on."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioPlatformHelpersLibrary_GetDefaultPortalForCurrentPlatform.png",src:i(22178).A+"",width:"307",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioPortal GetDefaultPortalForCurrentPlatform()\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-104",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsx)(d.table,{children:(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})})})}),"\n",(0,s.jsx)(d.h4,{id:"returns-44",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"EModioPortal of the portal to use"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-default-auth-provider-for-current-platform",children:"Get Default Auth Provider for Current Platform"}),"\n",(0,s.jsx)(d.p,{children:"Get the default Authentication Provider for the current platform the game is running on"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioPlatformHelpersLibrary_GetDefaultAuthProviderForCurrentPlatform.png",src:i(2957).A+"",width:"351",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioAuthenticationProvider GetDefaultAuthProviderForCurrentPlatform()\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-105",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsx)(d.table,{children:(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})})})}),"\n",(0,s.jsx)(d.h4,{id:"returns-45",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"EModioAuthenticationProvider to use for authentication calls"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-current-platform",children:"Get Current Platform"}),"\n",(0,s.jsx)(d.p,{children:"Gets the current platform that the game is running on"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioPlatformHelpersLibrary_GetCurrentPlatform.png",src:i(21596).A+"",width:"203",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioPlatformName GetCurrentPlatform()\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-106",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsx)(d.table,{children:(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})})})}),"\n",(0,s.jsx)(d.h4,{id:"returns-46",children:"Returns"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"round-number-string",children:"Round Number String"}),"\n",(0,s.jsx)(d.p,{children:"Sets the correct decimals depending on the file size or speed"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_RoundNumberString.png",src:i(1836).A+"",width:"287",height:"84"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText RoundNumberString(FText inputText)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-107",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"inputText"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-percent-integer64integer64",children:"Get Percent (integer64/integer64)"}),"\n",(0,s.jsxs)(d.p,{children:["Calculates percentage using two ",(0,s.jsx)(d.code,{children:"int64"})," params"]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_Pct_Int64Int64.png",src:i(84564).A+"",width:"267",height:"92"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"float Pct_Int64Int64(int64 Dividend, int64 Divisor)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-108",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Dividend"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Divisor"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-47",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"The floating point result of Dividend/Divisor with no checks"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"is-valid-security-code-format",children:"Is Valid Security Code Format"}),"\n",(0,s.jsx)(d.p,{children:"Checks if the string has the same format as the mod.io security code"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_IsValidSecurityCodeFormat.png",src:i(16566).A+"",width:"252",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool IsValidSecurityCodeFormat(FString String)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-109",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"String"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-48",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"True if the security code has a valid format"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"is-valid-email-address-format",children:"Is Valid Email Address Format"}),"\n",(0,s.jsx)(d.p,{children:"Does a basic validation if the email address supplied has a valid form"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_IsValidEmailAddressFormat.png",src:i(72003).A+"",width:"253",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool IsValidEmailAddressFormat(FString String)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-110",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"String"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-49",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"True if the email address has a valid format"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-time-span-as-string",children:"Get Time Span as String"}),"\n",(0,s.jsx)(d.p,{children:"Gets the time span between present and specified past date FString"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetTimeSpanAsString.png",src:i(60532).A+"",width:"293",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FString GetTimeSpanAsString(FString PastDateString)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-111",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PastDateString"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-shortened-number-as-string",children:"Get Shortened Number as String"}),"\n",(0,s.jsx)(d.p,{children:"Shortens the specified large number"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetShortenedNumberAsString.png",src:i(58510).A+"",width:"269",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FString GetShortenedNumberAsString(int64 Number)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-112",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Number"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-project-initialize-options-for-session-id",children:"Get Project Initialize Options for Session Id"}),"\n",(0,s.jsx)(d.p,{children:"Get the options needed to initialize the mod.io SDK specified in the project settings"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetProjectInitializeOptionsForSessionId.png",src:i(17753).A+"",width:"330",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioInitializeOptions GetProjectInitializeOptionsForSessionId(FString SessionId)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-113",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SessionId"})}),(0,s.jsx)(d.td,{children:"The LocalSessionIdentifier option to initialize project with"})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-project-game-id",children:"Get Project Game Id"}),"\n",(0,s.jsx)(d.p,{children:"Get the game id specified in the mod.io project settings"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetProjectGameId.png",src:i(98488).A+"",width:"198",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioGameID GetProjectGameId()\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-114",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsx)(d.table,{children:(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})})})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-project-environment",children:"Get Project Environment"}),"\n",(0,s.jsx)(d.p,{children:"Get the environment specified in the mod.io project settings"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetProjectEnvironment.png",src:i(46898).A+"",width:"223",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioEnvironment GetProjectEnvironment()\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-115",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsx)(d.table,{children:(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})})})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-project-api-key",children:"Get Project Api Key"}),"\n",(0,s.jsx)(d.p,{children:"Get the api key specified in the mod.io project settings"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetProjectApiKey.png",src:i(54088).A+"",width:"193",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioApiKey GetProjectApiKey()\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-116",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsx)(d.table,{children:(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})})})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-monetization-purchase-category",children:"Get Monetization Purchase Category"}),"\n",(0,s.jsx)(d.p,{children:"Get the purchase category to pass to the store overlay UI for a given portal."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetMonetizationPurchaseCategory.png",src:i(91574).A+"",width:"379",height:"104"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FString GetMonetizationPurchaseCategory(EModioPortal Portal)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-117",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Portal"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-50",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"string of ID for the purchase category for a given Portal in use, returns empty if the given key was not found/not defined"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-language-code-string",children:"Get Language Code String"}),"\n",(0,s.jsx)(d.p,{children:"Get language code string. This can be used for localization purposes"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetLanguageCodeString.png",src:i(89138).A+"",width:"379",height:"104"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FString GetLanguageCodeString(EModioLanguage Language)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-118",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Language"})}),(0,s.jsx)(d.td,{children:"The language code to convert to string"})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-51",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:'The language code as a string (e.g. "en", "fr", "de")'}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-language-code-from-string",children:"Get Language Code from String"}),"\n",(0,s.jsx)(d.p,{children:"Get language code enum from string in ISO 639-1 format. This can be used for localization purposes"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetLanguageCodeFromString.png",src:i(81254).A+"",width:"291",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"EModioLanguage GetLanguageCodeFromString(FString LanguageCode)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-119",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"LanguageCode"})}),(0,s.jsx)(d.td,{children:'The language code as a string (e.g. "en", "fr", "de")'})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-52",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"The language code as an enum"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-desired-file-size-unit",children:"Get Desired File Size Unit"}),"\n",(0,s.jsx)(d.p,{children:"Get desired file size unit based on the size of the file"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetDesiredFileSizeUnit.png",src:i(60391).A+"",width:"249",height:"81"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"TEnumAsByte GetDesiredFileSizeUnit(int64 FileSize)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-120",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FileSize"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-53",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"the desired file size unit"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-default-session-id-windows",children:"Get Default Session Id Windows"}),"\n",(0,s.jsx)(d.p,{children:"Get Session Id for Windows for initialization of the SDK"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_GetDefaultSessionIdWindows.png",src:i(71175).A+"",width:"266",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FString GetDefaultSessionIdWindows()\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-121",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsx)(d.table,{children:(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})})})}),"\n",(0,s.jsx)(d.h4,{id:"returns-54",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"The windows session id, or an empty string if you are not on Windows"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"tostring-filesize",children:"ToString (Filesize)"}),"\n",(0,s.jsx)(d.p,{children:"Converts a filesize to a human readable string with the appropriate unit"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSDKLibrary_Filesize_ToString.png",src:i(12536).A+"",width:"441",height:"205"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText Filesize_ToString(int64 FileSize, int32 MinDecimals, int32 MaxDecimals, TEnumAsByte Unit, bool bIncludeUnitName)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-122",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FileSize"})}),(0,s.jsx)(d.td,{children:"Filesize in bytes"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MinDecimals"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MaxDecimals"})}),(0,s.jsx)(d.td,{children:"Maximum amount of decimals to display of the filesize"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Unit"})}),(0,s.jsx)(d.td,{children:"If Largest, then it tries to display the size in the largest unit that will have a integral part > 0, else it displays the filesize in the specified unit"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"bIncludeUnitName"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-55",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"A text formatted from your specifications"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"getdefaultmodinstallationdirectory",children:"GetDefaultModInstallationDirectory"}),"\n",(0,s.jsx)(d.p,{children:"Returns the default mod installation directory for this game and platform, ignoring overrides and without requiring the SDK to be initialized."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubsystem_K2_GetDefaultModInstallationDirectory.png",src:i(92554).A+"",width:"290",height:"108"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FString K2_GetDefaultModInstallationDirectory(FModioGameID GameID)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-123",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GameID"})}),(0,s.jsxs)(d.td,{children:["The ",(0,s.jsx)(d.code,{children:"ModioGameID"})," of the game we're fetching the default mod installation directory for."]})]})})]})}),"\n",(0,s.jsx)(d.h4,{id:"returns-56",children:"Returns"}),"\n",(0,s.jsx)(d.p,{children:"The default mod installation directory for the specified game on the current platform"}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"is-purchasable",children:"Is Purchasable"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_IsPurchasable.png",src:i(72668).A+"",width:"192",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool IsPurchasable(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-title",children:"Get Title"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_GetTitle.png",src:i(15032).A+"",width:"192",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText GetTitle(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-regular-price",children:"Get Regular Price"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_GetRegularPrice.png",src:i(88371).A+"",width:"192",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText GetRegularPrice(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-numeric-price",children:"Get Numeric Price"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_GetNumericPrice.png",src:i(96756).A+"",width:"192",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"int64 GetNumericPrice(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-modio-id",children:"Get Modio Id"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_GetModioId.png",src:i(83505).A+"",width:"192",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioTokenPackID GetModioId(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-long-description",children:"Get Long Description"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_GetLongDescription.png",src:i(66182).A+"",width:"203",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText GetLongDescription(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-id-1",children:"Get Id"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_GetId.png",src:i(88883).A+"",width:"192",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FString GetId(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-fields",children:"Get Fields"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_GetFields.png",src:i(74615).A+"",width:"193",height:"78"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"TMap GetFields(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-display-price",children:"Get Display Price"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_GetDisplayPrice.png",src:i(29301).A+"",width:"192",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText GetDisplayPrice(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"get-description",children:"Get Description"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioTokenPackLibrary_GetDescription.png",src:i(53312).A+"",width:"192",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FText GetDescription(FModioTokenPack In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64---modiounsigned64",children:"ModioUnsigned64 - ModioUnsigned64"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_Subtract.png",src:i(6119).A+"",width:"139",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 Subtract(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"percentage-unsigned-64",children:"Percentage Unsigned 64"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_Percentage_Unsigned64.png",src:i(47991).A+"",width:"222",height:"106"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"float Percentage_Unsigned64(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64",children:"ModioUnsigned64 != ModioUnsigned64"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_NotEqualTo.png",src:i(60609).A+"",width:"159",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool NotEqualTo(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"make-from-components",children:"Make from Components"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_MakeFromComponents.png",src:i(37173).A+"",width:"220",height:"106"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 MakeFromComponents(int32 High, int32 Low)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-1",children:"ModioUnsigned64 < ModioUnsigned64"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_LessThan.png",src:i(29929).A+"",width:"139",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool LessThan(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64--0",children:"ModioUnsigned64 > 0"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_GreaterThanZero.png",src:i(42506).A+"",width:"179",height:"54"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool GreaterThanZero(FModioUnsigned64 In)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-2",children:"ModioUnsigned64 > ModioUnsigned64"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_GreaterThan.png",src:i(84996).A+"",width:"139",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool GreaterThan(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-3",children:"ModioUnsigned64 == ModioUnsigned64"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_EqualTo.png",src:i(62576).A+"",width:"179",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool EqualTo(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-4",children:"ModioUnsigned64 / ModioUnsigned64"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_DivideToFloat.png",src:i(1381).A+"",width:"139",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"float DivideToFloat(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64--float",children:"ModioUnsigned64 / float"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_DivideFloat.png",src:i(74136).A+"",width:"171",height:"84"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"float DivideFloat(FModioUnsigned64 LHS, float RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-truncate",children:"ModioUnsigned64 / ModioUnsigned64 (truncate)"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_Divide.png",src:i(1512).A+"",width:"139",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 Divide(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"break-to-components",children:"Break to Components"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_BreakToComponents.png",src:i(9105).A+"",width:"206",height:"106"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void BreakToComponents(FModioUnsigned64 In, int32 High, int32 Low)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"modiounsigned64--modiounsigned64-5",children:"ModioUnsigned64 + ModioUnsigned64"}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioUnsigned64Library_Add.png",src:i(99794).A+"",width:"139",height:"76"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"FModioUnsigned64 Add(FModioUnsigned64 LHS, FModioUnsigned64 RHS)\n"})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"submitnewmodfrommemoryasync",children:"SubmitNewModFromMemoryAsync"}),"\n",(0,s.jsx)(d.p,{children:"Submit a new mod, with its logo data coming from an in-memory buffer rather than a file."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync.png",src:i(34022).A+"",width:"287",height:"198"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitNewModFromMemoryAsync(FModioModCreationHandle Handle, FModioCreateModParams Params, TArray PngData, FOnSubmitNewModDelegate Callback)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"parameters-124",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Handle"})}),(0,s.jsx)(d.td,{children:"Mod creation handle"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Params"})}),(0,s.jsx)(d.td,{children:"Parameters to use when creating the mod"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PngData"})}),(0,s.jsx)(d.td,{children:"In-memory buffer, representative of a PNG file to be used for upload"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Callback"})}),(0,s.jsx)(d.td,{children:"Callback once operation has completed"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"submitnewmodfileformodfrommemory",children:"SubmitNewModFileForModFromMemory"}),"\n",(0,s.jsxs)(d.p,{children:["Queues the upload of a new mod file release for the specified mod, using the submitted parameters. This upload method accepts a block of memory ",(0,s.jsx)(d.code,{children:"TArray"})," rather than a file path. The upload's progress can be tracked in the same way as downloads; when completed, a Mod Management Event will be triggered with the result code for the upload."]}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory.png",src:i(32601).A+"",width:"315",height:"192"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"void K2_SubmitNewModFileForModFromMemory(UModioSubsystem* Target, FModioModID Mod, FModioCreateModFileMemoryParams Params)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"requirements-41",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h4,{id:"parameters-125",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UModioSubsystem*"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Mod"})}),(0,s.jsx)(d.td,{children:"The ID of the mod you are submitting a file for"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Params"})}),(0,s.jsx)(d.td,{children:"Information about the mod file being created, including the memory that wiull be uploaded as a mod"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"loadmodfiletomemory",children:"LoadModFileToMemory"}),"\n",(0,s.jsx)(d.p,{children:"Loads an installed mod file into memory."}),"\n",(0,s.jsx)(d.p,{children:(0,s.jsx)(d.img,{alt:"nd_img_ModioSubmissionExtensionLibrary_K2_LoadModFileToMemory.png",src:i(20595).A+"",width:"318",height:"162"})}),"\n",(0,s.jsx)(d.pre,{children:(0,s.jsx)(d.code,{className:"language-cpp",children:"bool K2_LoadModFileToMemory(UModioSubsystem* Target, FModioModID ModId, TArray ModData)\n"})}),"\n",(0,s.jsx)(d.h4,{id:"requirements-42",children:"Requirements"}),"\n",(0,s.jsxs)(d.ul,{children:["\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"initialized-sdk"})}),"\n",(0,s.jsx)(d.li,{children:(0,s.jsx)(d.em,{children:"authenticated-user"})}),"\n"]}),"\n",(0,s.jsx)(d.h4,{id:"parameters-126",children:"Parameters"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Target"})}),(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UModioSubsystem*"})})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModId"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Mod Data"})}),(0,s.jsx)(d.td,{children:"A byte array of the mod that has been loaded"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h2,{id:"enums",children:"Enums"}),"\n",(0,s.jsx)(d.h3,{id:"EModioModfilePlatform",children:"EModioModfilePlatform"}),"\n",(0,s.jsx)(d.p,{children:"Enum representing the platform(s) that a modfile is enabled for"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Windows"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Mac"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Linux"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Android"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"iOS"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"XboxOne"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"XboxSeriesX"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PS4"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PS5"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Switch"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Oculus"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Source"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EGameMaturityFlags",children:"EGameMaturityFlags"}),"\n",(0,s.jsx)(d.p,{children:"Maturity options for a game"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"None"})}),(0,s.jsx)(d.td,{children:"Don't allow mature content in mods (default)"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MatureModsAllowed"})}),(0,s.jsx)(d.td,{children:"This game allows mods containing mature content"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MatureAudiencesOnly"})}),(0,s.jsx)(d.td,{children:"This game is for mature audiences only"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EGameMonetizationFlags",children:"EGameMonetizationFlags"}),"\n",(0,s.jsx)(d.p,{children:"Monetization properties of a game"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"None"})}),(0,s.jsx)(d.td,{children:"None set (default)"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Monetization"})}),(0,s.jsx)(d.td,{children:"Monetization is enabled"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Marketplace"})}),(0,s.jsx)(d.td,{children:"Marketplace is enabled"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PartnerProgram"})}),(0,s.jsx)(d.td,{children:"Partner Program is enabled"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioModServerSideStatus",children:"EModioModServerSideStatus"}),"\n",(0,s.jsx)(d.p,{children:"Enum representing a mod's server side status"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NotAccepted"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Accepted"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Deleted"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioVirusStatus",children:"EModioVirusStatus"}),"\n",(0,s.jsx)(d.p,{children:"If the file has been found to be malicious or not"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NoThreat"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Malicious"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioVirusScanStatus",children:"EModioVirusScanStatus"}),"\n",(0,s.jsx)(d.p,{children:"Current state of the scanned file"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NotScanned"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ScanComplete"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InProgress"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TooLargeToScan"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FileNotFound"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ErrorScanning"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioObjectVisibilityFlags",children:"EModioObjectVisibilityFlags"}),"\n",(0,s.jsx)(d.p,{children:"Enum representing whether or not a mod is visible to users"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Hidden"})}),(0,s.jsx)(d.td,{children:"Mod is concealed from users"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Public"})}),(0,s.jsx)(d.td,{children:"Mod is openly available"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioMaturityFlags",children:"EModioMaturityFlags"}),"\n",(0,s.jsx)(d.p,{children:"Enum representing mature content that a mod may contain"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"None"})}),(0,s.jsx)(d.td,{children:"No maturity"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Alcohol"})}),(0,s.jsx)(d.td,{children:"Content contains alcohol references"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Drugs"})}),(0,s.jsx)(d.td,{children:"Content contains drug references"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Violence"})}),(0,s.jsx)(d.td,{children:"Content contains violence references"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Explicit"})}),(0,s.jsx)(d.td,{children:"Content contains sexual references"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioModManagementEventType",children:"EModioModManagementEventType"}),"\n",(0,s.jsx)(d.p,{children:"What type of event occurred"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Installed"})}),(0,s.jsx)(d.td,{children:"Mod installation to local storage completed"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Uninstalled"})}),(0,s.jsx)(d.td,{children:"Mod uninstallation from local storage completed"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Updated"})}),(0,s.jsx)(d.td,{children:"Mod local installation updated to latest version"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Uploaded"})}),(0,s.jsx)(d.td,{children:"Mod file was uploaded"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"BeginInstall"})}),(0,s.jsx)(d.td,{children:"Mod download and installation has started"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"BeginUninstall"})}),(0,s.jsx)(d.td,{children:"Mod uninstallation has started"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"BeginUpdate"})}),(0,s.jsx)(d.td,{children:"Mod download and update has started"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"BeginUpload"})}),(0,s.jsx)(d.td,{children:"Mod upload has started"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioAuthenticationProvider",children:"EModioAuthenticationProvider"}),"\n",(0,s.jsx)(d.p,{children:"Simple struct to encapsulate data passed to external authentication systems"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"XboxLive"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Steam"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GoG"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Itch"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Switch"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Discord"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PSN"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Epic"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Oculus"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"OpenID"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GoogleIDToken"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GoogleServerSideToken"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioEnvironment",children:"EModioEnvironment"}),"\n",(0,s.jsxs)(d.p,{children:["Enum representing which environment the game is deployed to: ",(0,s.jsx)(d.code,{children:"Test"})," or ",(0,s.jsx)(d.code,{children:"Live"}),"."]}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Test"})}),(0,s.jsx)(d.td,{children:"Test (private) environment"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Live"})}),(0,s.jsx)(d.td,{children:"Live (public) environment"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioPortal",children:"EModioPortal"}),"\n",(0,s.jsx)(d.p,{children:"Enum representing the store or service your game is being distributed through"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"None"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Apple"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EpicGamesStore"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GOG"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Google"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Itchio"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Nintendo"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PSN"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Steam"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"XboxLive"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioPlatformName",children:"EModioPlatformName"}),"\n",(0,s.jsx)(d.p,{children:"Enum representing a named platform that the plugin is running on."}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Windows"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Mac"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Linux"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PS4"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PS5"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"XBoxOne"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"XSX"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Switch"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Unknown"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Android"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"iOS"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioLogoSize",children:"EModioLogoSize"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb320"})}),(0,s.jsx)(d.td,{children:"320x180px"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb640"})}),(0,s.jsx)(d.td,{children:"640x360px"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb1280"})}),(0,s.jsx)(d.td,{children:"1280x720px"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Original"})}),(0,s.jsx)(d.td,{children:"Original Size"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioAvatarSize",children:"EModioAvatarSize"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Original"})}),(0,s.jsx)(d.td,{children:"Original Size"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb50"})}),(0,s.jsx)(d.td,{children:"50x50px Thumbnail"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb100"})}),(0,s.jsx)(d.td,{children:"100x100px Thumbnail"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioGallerySize",children:"EModioGallerySize"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Original"})}),(0,s.jsx)(d.td,{children:"Original Size"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb320"})}),(0,s.jsx)(d.td,{children:"320x180px Thumbnail"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thumb1280"})}),(0,s.jsx)(d.td,{children:"1280x720 Thumbnail"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioLogLevel",children:"EModioLogLevel"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Trace"})}),(0,s.jsx)(d.td,{children:"Detailed low-level debugging output. Not intended for general use"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Detailed"})}),(0,s.jsx)(d.td,{children:"Detailed but not low-level. Generally useful for some mid-level information for debugging."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Info"})}),(0,s.jsx)(d.td,{children:"Informational output containing status messages"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Warning"})}),(0,s.jsx)(d.td,{children:"Warnings about incorrect plugin usage, timeouts"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Error"})}),(0,s.jsx)(d.td,{children:"Only errors"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioLanguage",children:"EModioLanguage"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"English"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Bulgarian"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"French"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"German"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Italian"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Polish"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Portuguese"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Hungarian"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Japanese"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Korean"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Russian"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Spanish"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Thai"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ChineseSimplified"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ChineseTraditional"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Turkish"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Ukrainian"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Arabic"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioModChangeType",children:"EModioModChangeType"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Added"})}),(0,s.jsx)(d.td,{children:"The user's list has a new mod to synchronize"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Removed"})}),(0,s.jsx)(d.td,{children:"The user's list must remove a mod to synchronize"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Updated"})}),(0,s.jsx)(d.td,{children:"The user's list must update a mod to synchronize"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EFileSizeUnit",children:"EFileSizeUnit"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Largest"})}),(0,s.jsx)(d.td,{children:"Will take the largest one that becomes a number larger than 1 (i.e, 1300mb becomes 1.3gb)"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"B"})}),(0,s.jsx)(d.td,{children:"A single byte"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"KB"})}),(0,s.jsx)(d.td,{children:"Kilobytes"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MB"})}),(0,s.jsx)(d.td,{children:"Megabytes"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"GB"})}),(0,s.jsx)(d.td,{children:"Gigabytes"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioEntitlementConsumptionState",children:"EModioEntitlementConsumptionState"}),"\n",(0,s.jsx)(d.p,{children:"State of an entitlement consumption transaction"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Failed"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Pending"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Fulfilled"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ConsumeLimitExceeded"})}),(0,s.jsx)(d.td,{})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioEntitlementType",children:"EModioEntitlementType"}),"\n",(0,s.jsx)(d.p,{children:"Type of entitlement that was consumed"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsx)(d.tbody,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"VirtualCurrency"})}),(0,s.jsx)(d.td,{})]})})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioErrorCondition",children:"EModioErrorCondition"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NoError"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NetworkError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code represents a connection or HTTP error between the client and the mod.io server."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ConfigurationError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates the SDK's configuration is not valid - the game ID or API key are incorrect or the game has been deleted."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InvalidArgsError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates the arguments passed to the function have failed validation or were otherwise invalid."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FilesystemError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates a permission or IO error when accessing local filesystem data."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InternalError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code represents an internal SDK error - please inform mod.io of the error code value."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ApiErrorRefSuccess"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error ref returned by the API indicates an implicit success because the operation has already been performed (ie a no-op is success)."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EntityNotFoundError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that a specified game, mod, user, media file or mod file was not found."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserTermsOfUseError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that the user has not yet accepted the mod.io Terms of Use."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SubmitReportError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that a report for the specified content could not be submitted."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserNotAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that a user is not authenticated."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKNotInitialized"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that the SDK has not been initialized."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UserAlreadyAuthenticatedError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that the user is already authenticated."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SystemError"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that a low-level system error occurred outside of mod.io SDK control."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"OperationCanceled"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that the asynchronous operation was cancelled before it completed."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModManagementDisabled"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that Mod Management has not been enabled."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RateLimited"})}),(0,s.jsx)(d.td,{children:"Too many requests made to the mod.io API within the rate-limiting window. Please wait and try again."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModBeingProcessed"})}),(0,s.jsx)(d.td,{children:"The specified mod's files are currently being updated by the SDK. Please try again later."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InsufficientSpace"})}),(0,s.jsx)(d.td,{children:"There is insufficient space to install the mod. Please free up space and try again."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SDKAlreadyInitialized"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that the SDK has already been initialized."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ModManagementAlreadyEnabled"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that Mod Management has already been enabled."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InsufficientPermissions"})}),(0,s.jsx)(d.td,{children:"When this condition is true, the error code indicates that the current user does not have the required permissions for this operation."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EmailLoginCodeInvalid"})}),(0,s.jsx)(d.td,{children:"The email login code is incorrect, has expired, or has already been used."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"AlreadySubscribed"})}),(0,s.jsx)(d.td,{children:"The specified mod is already subscribed to."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InstallOrUpdateCancelled"})}),(0,s.jsx)(d.td,{children:"The current mod installation or update was cancelled."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UploadCancelled"})}),(0,s.jsx)(d.td,{children:"The current modfile upload was cancelled."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"TempModSetNotInitialized"})}),(0,s.jsx)(d.td,{children:"TempModSet need to be initialized first, call InitTempModSet."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MonetizationOperationError"})}),(0,s.jsx)(d.td,{children:"An error occurred while performing a monetization operation."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PaymentTransactionFailed"})}),(0,s.jsx)(d.td,{children:"The transaction requires a payment but it could not be fulfilled. Please retry with funds on the wallet"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"IncorrectPrice"})}),(0,s.jsx)(d.td,{children:"The display price for the mod is out-of-date or incorrect. Please retry with the correct display price."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ItemAlreadyOwned"})}),(0,s.jsx)(d.td,{children:"The authenticated user already has acquired this item"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ParentalControlRestrictions"})}),(0,s.jsx)(d.td,{children:"Parental control restrictions prevent this account from accessing UGC."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MetricsSessionNotInitialized"})}),(0,s.jsx)(d.td,{children:"Metrics session has not yet been initialized. Ensure that you have a metrics secret key set for your project."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MetricsSessionAlreadyInitialized"})}),(0,s.jsx)(d.td,{children:"Metrics session has already been been initialized."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MetricsSessionIsActive"})}),(0,s.jsx)(d.td,{children:"Metrics session has been started."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MetricsSessionIsNotActive"})}),(0,s.jsx)(d.td,{children:"Metrics session has not been started. Please call MetricsSessionStartAsync."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"MetricsSessionHasNoMods"})}),(0,s.jsx)(d.td,{children:"No mods have been added to the session."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"PremiumFeatureNotAvailable"})}),(0,s.jsx)(d.td,{children:"This premium feature is not available for your project."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"EmailExchangeCodeAlreadyRedeemed"})}),(0,s.jsx)(d.td,{children:"The email security code has already been redeemed."})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioSortFieldType",children:"EModioSortFieldType"}),"\n",(0,s.jsx)(d.p,{children:"Enum indicating which field should be used to sort the results"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"ID"})}),(0,s.jsx)(d.td,{children:"Use mod ID (default)"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DownloadsToday"})}),(0,s.jsx)(d.td,{children:'Use number of downloads in last 24 (exposed in REST API as "popular")'})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"SubscriberCount"})}),(0,s.jsx)(d.td,{children:"Use number of subscribers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Rating"})}),(0,s.jsx)(d.td,{children:"Use mod rating"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateMarkedLive"})}),(0,s.jsx)(d.td,{children:"Use date mod was marked live"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DateUpdated"})}),(0,s.jsx)(d.td,{children:"Use date mod was last updated"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DownloadsTotal"})}),(0,s.jsx)(d.td,{children:"Use downloads total"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Alphabetical"})}),(0,s.jsx)(d.td,{children:"Use mod name"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioSortDirection",children:"EModioSortDirection"}),"\n",(0,s.jsx)(d.p,{children:"Enum indicating which direction sorting should be applied"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Ascending"})}),(0,s.jsx)(d.td,{})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Descending"})}),(0,s.jsx)(d.td,{children:"(default)"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioRevenueFilterType",children:"EModioRevenueFilterType"}),"\n",(0,s.jsx)(d.p,{children:"Enum indicating filtering options based off revenue type"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Free"})}),(0,s.jsx)(d.td,{children:"Return only free mods"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Paid"})}),(0,s.jsx)(d.td,{children:"Return only paid mods"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FreeAndPaid"})}),(0,s.jsx)(d.td,{children:"Return both free and paid mods"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioImageState",children:"EModioImageState"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"OnDisc"})}),(0,s.jsx)(d.td,{children:"Image data is located on hard drive"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"LoadingIntoMemory"})}),(0,s.jsx)(d.td,{children:"Image data is transferring to a memory location"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InMemory"})}),(0,s.jsx)(d.td,{children:"Image data is located in memory"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Corrupted"})}),(0,s.jsx)(d.td,{children:"Image data is not readable"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioModState",children:"EModioModState"}),"\n",(0,s.jsx)(d.p,{children:"Enum representing the current state of a mod"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"InstallationPending"})}),(0,s.jsx)(d.td,{children:"The mod is pending installation. This state is not saved."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Installed"})}),(0,s.jsx)(d.td,{children:"The mod is installed."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UpdatePending"})}),(0,s.jsx)(d.td,{children:"The mod is pending an update. This state is saved as installed."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Downloading"})}),(0,s.jsx)(d.td,{children:"The mod is downloading as part of the installation process. This state is not saved."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Extracting"})}),(0,s.jsx)(d.td,{children:"The mod is extracting as part of the installation process. This state is not saved."})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"UninstallPending"})}),(0,s.jsx)(d.td,{children:"The mod is pending uninstallation. This state is saved as installed."})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioModProgressState",children:"EModioModProgressState"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Initializing"})}),(0,s.jsx)(d.td,{children:"Download information is being retrieved from mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Downloading"})}),(0,s.jsx)(d.td,{children:"Mod archive is downloading from mod.io servers"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Extracting"})}),(0,s.jsx)(d.td,{children:"Mod archive is downloaded and now extracting"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Compressing"})}),(0,s.jsx)(d.td,{children:"Mod archive is being compressed from files on disk"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Uploading"})}),(0,s.jsx)(d.td,{children:"Mod archive is uploading to mod.io servers"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioOpenStoreResult",children:"EModioOpenStoreResult"}),"\n",(0,s.jsx)(d.p,{children:"Enumerator of potential results on attempting to open a native platform store"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Success"})}),(0,s.jsx)(d.td,{children:"Store opened successfully *"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FailedInactive"})}),(0,s.jsx)(d.td,{children:"Monetization not active *"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FailedUnsupportedPlatform"})}),(0,s.jsx)(d.td,{children:"The current platform does not support a native store *"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FailedUnknown"})}),(0,s.jsx)(d.td,{children:"Failed to open for an unknown reason *"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioRating",children:"EModioRating"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Neutral"})}),(0,s.jsx)(d.td,{children:"A neutral rating"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Positive"})}),(0,s.jsx)(d.td,{children:"A positive rating"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Negative"})}),(0,s.jsx)(d.td,{children:"A negative rating"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{}),"\n",(0,s.jsx)(d.h3,{id:"EModioReportType",children:"EModioReportType"}),"\n",(0,s.jsx)(n,{colWidths:["30%","70%"],stripes:"odd",children:(0,s.jsxs)(d.table,{children:[(0,s.jsx)(d.thead,{children:(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.th,{}),(0,s.jsx)(d.th,{})]})}),(0,s.jsxs)(d.tbody,{children:[(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Generic"})}),(0,s.jsx)(d.td,{children:"A generic mod report"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"DMCA"})}),(0,s.jsx)(d.td,{children:"Digital Millennium Copyright Act mod report"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"NotWorking"})}),(0,s.jsx)(d.td,{children:"Not working mod report"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"RudeContent"})}),(0,s.jsx)(d.td,{children:"Rude content mod report"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"IllegalContent"})}),(0,s.jsx)(d.td,{children:"Illegal content mod report"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"StolenContent"})}),(0,s.jsx)(d.td,{children:"Stolen content mod report"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"FalseInformation"})}),(0,s.jsx)(d.td,{children:"False information mod report"})]}),(0,s.jsxs)(d.tr,{children:[(0,s.jsx)(d.td,{children:(0,s.jsx)(d.code,{children:"Other"})}),(0,s.jsx)(d.td,{children:"Other type of mod report"})]})]})]})}),"\n",(0,s.jsx)(d.hr,{})]})}function a(e={}){const{wrapper:d}={...(0,r.R)(),...e.components};return d?(0,s.jsx)(d,{...e,children:(0,s.jsx)(h,{...e})}):h(e)}},73149:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},59892:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},89887:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},36342:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCommonTypesLibrary_MakeAuthParams-9889ff972f1927cd2d65350a785da1b8.png"},21735:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCommonTypesLibrary_MakeEntitlementParams-ca9ee94871d4bcb815ae1afeadb102d4.png"},94923:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},1237:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},50900:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions-76f5950618826972f9a40bde21e49d4d.png"},80709:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},42690:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},16659:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},92422:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetBackgroundThread-e499bbf6ceac1c576056f97dbfce48d3.png"},38945:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetExtendedInitializationParameters-b675994c5f5117d737da5f13331ec33b.png"},80201:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetGameEnvironment-9bf28c658ca9a8175e27f7f1e2e6ce09.png"},45671:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetGameId-733baa40804b516d3926e06d2e5ffed7.png"},21880:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetPortal-f95434662488c3c1596eda6b76b8d3de.png"},9731:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},73569:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier-48bd0814f5db0ec0f8a97d2560a396f7.png"},62283:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},21900:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},58589:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},40594:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCreateModLibrary_SetInitialVisibility-ea630d72471876be3f92ec2a7f62411e.png"},65268:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCreateModLibrary_SetInitialVisibility_DEPRECATED-f8acc13f3be4d819e740b51766dd4ed7.png"},74678:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCreateModLibrary_SetMarkAsActiveRelease-d40c905fb2ac942db96dcf0e0982b680.png"},10318:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},32566:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob-935a953f1fa434ce4627432800c74b48.png"},94646:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},40007:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},71105:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},270:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},85392:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},78906:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},23039:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioErrorCodeLibrary_ReconstructError-fd34e6c485430ac5359952f8ad438ffb.png"},4457:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioErrorConditionLibrary_ErrorCodeMatches-0577e7c35976e2a2d2733042ba1b514d.png"},58370:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},15899:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},77120:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},89e3:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync-695ef7f0e8b10ac403d0fd21329dfcf0.png"},86557:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioImageLibrary_GetAvatarSize-082b1eeffcd2b743f94a66210f340b0b.png"},24940:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioImageLibrary_GetGallerySize-9626c66c39dbc11535dffd35e279ef87.png"},63647:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioImageLibrary_GetLogoSize-e5d6147cf8343e10fa23bdb08d0b646a.png"},46998:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},92108:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},60565:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},52915:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},39711:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},56543:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},65953:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},5873:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioModProgressInfoLibrary_GetCurrentProgress-cf02a1331822e1a6d164eb0af3fe972f.png"},84009:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},66964:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioModProgressInfoLibrary_GetTotalProgress-bf65b6793b2e6f41218751755daf9136.png"},17572:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},7593:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},21596:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},2957:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioPlatformHelpersLibrary_GetDefaultAuthProviderForCurrentPlatform-8684a5c74a56ec16d57182102488818f.png"},22178:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioPlatformHelpersLibrary_GetDefaultPortalForCurrentPlatform-2f5206eeda17f3ae9b21f090893f58e9.png"},64454:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},12536:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},71175:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},60391:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},81254:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSDKLibrary_GetLanguageCodeFromString-07f6fa34ac42129bacc95488775b986f.png"},89138:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSDKLibrary_GetLanguageCodeString-47da87bf1ca6d0b3444e4ea1eaac742e.png"},91574:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSDKLibrary_GetMonetizationPurchaseCategory-51bd9f941653bd18b5e75f34e6c26c39.png"},54088:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},46898:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},98488:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},17753:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSDKLibrary_GetProjectInitializeOptionsForSessionId-794e92929faeee6a2fb661456d91d1d7.png"},58510:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSDKLibrary_GetShortenedNumberAsString-4d9bf0258754df026d7a2082964ffba4.png"},60532:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSDKLibrary_GetTimeSpanAsString-b09ea2de257e9f6134e49e2bae984be6.png"},72003:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},16566:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSDKLibrary_IsValidSecurityCodeFormat-f2b25b06b6647d66e1e55a880bf08f59.png"},84564:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},1836:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},20595:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_LoadModFileToMemory-880f5d8c1ba728bf068807cb3559919b.png"},79873:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitModChangesFromMemoryAsync-4edd860ab7540b93959c8c0f7f2a5137.png"},32601:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory-43d8d48bf01808ca9f889c44c78586e4.png"},34022:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync-31680cc49f35d2f21d91b6d2fbb9c98d.png"},23297:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_DisableModManagement-5f0eb0a41951b35be46cb68387079953.png"},18781:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_GetLastValidationError-248302bd685ccf133c8b35edafba9bb3.png"},24896:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_IsModManagementBusy-124248b31972e2a2a47932b93466cdda.png"},62286:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_IsUsingBackgroundThread-09942c052a7a41c9f8586bc302807014.png"},60006:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_AddToTempModSet-bdc1c1978e245ba3122b0de58c45c5d0.png"},6312:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_ArchiveModAsync-301820098783a60cf0bb90984e24ea48.png"},29010:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_AuthenticateUserEmailAsync-d8860759b43511919fe6a4dd23b4d16b.png"},92223:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_AuthenticateUserExternalAsync-20730b90c571ce00ba33752eab60040b.png"},36106:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_ClearUserDataAsync-c2ead944b1bcdee2aa8612351796be8d.png"},69252:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_CloseTempModSet-cd8ef129b913668bdc811207a7c16224.png"},69724:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_EnableModManagement-9fdcb415f8dac2cadba8b3ef4619e8ea.png"},86649:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync-83a8301ecba540f8a8592822d3184974.png"},5859:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync-d188629fa0a5c265b944f648d53b6440.png"},2053:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_ForceUninstallModAsync-f67b8d699d627e8c5788ca293d0194fd.png"},92554:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetDefaultModInstallationDirectory-358e4440ead73240cbcc7e1351ddd81a.png"},85314:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetGameInfoAsync-c33df4da02b4806e818624cea3158637.png"},82892:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetLanguage-825d1af09bb500dac57886a5a78ca64a.png"},56025:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetModCreationHandle-556d4efd18f7c7c76c23437062cac4d2.png"},74955:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetModDependenciesAsync-bb9a17927f1c969c6787d4dd9d42c76e.png"},7434:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetModInfoAsync-c141dbb87248abd169f33ef254251c9f.png"},60973:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetModMediaAvatarAsync-1fdb3c58a27f52670c442f3aa1bbf016.png"},34351:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync-be9588fa4dfac1ac91a102ce0ebd6671.png"},90655:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetModMediaLogoAsync-0a0a1ed024a69d97fcc0a10bdae9b811.png"},74220:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetModTagOptionsAsync-ed35670d985059ef2f2775e597d94fe9.png"},47193:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetMutedUsersAsync-b889b19125d0360cf4d520ade2ab18c5.png"},65789:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetTermsOfUseAsync-c6e22c2b2868122144640004d2394b3b.png"},40348:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetUserDelegationTokenAsync-80a54ec053e5b99bafc5bf071b5fbc97.png"},96438:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetUserMediaAvatarAsync-2a773ca98cccb4b7e926aaf48bc7ff7e.png"},21964:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync-5fe3ab434879bb8d9a79b0ee50739d4c.png"},25684:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_InitTempModSet-34fdcb787f07da43afb64dd9d76ea4d1.png"},49280:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_InitializeAsync-11220774e42901b7ce5d83720c58c28d.png"},85890:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_ListAllModsAsync-1ad7bc96d31f8ae60bfae27078fc2b9f.png"},22028:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_ListUserCreatedModsAsync-ff9ef2ec2b8111cc4674281c005b4d0e.png"},6938:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_ListUserGamesAsync-3914aa13ec1f36ad56d3058600385d7d.png"},92124:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_MetricsSessionEndAsync-830d93bd55175fb12d6ada839686ad5d.png"},14811:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync-554e56025bd496213a76378b849c929b.png"},44014:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatOnceAsync-5bfbdb4e8f6cb7d8572b81c5dccc84dc.png"},10293:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_MetricsSessionStartAsync-651730c41eec661e03832ebfea99162f.png"},96252:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_MuteUserAsync-95092e29e2790d8e96bc2cdc90d1ffb1.png"},99101:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_PreviewExternalUpdatesAsync-2855ce40346338726cdf2fc3f6d4c10a.png"},68537:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_PurchaseModAsync-d05b70f5ff78fce8ad562c2c6597af48.png"},72340:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_QueryCurrentModUpdate-ca721cd3920c490f3d8f27a0e87a9721.png"},55060:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_QueryTempModSet-b0ede8242625cb572430cc348883f3c0.png"},56696:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_QueryUserProfile-290a47c2579a6de4b0e8b4c843dad5cf.png"},83788:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_RefreshUserEntitlementsAsync-0c70260b80c34c71c10bf74061fb4ea2.png"},89494:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_RemoveFromTempModSet-6a7cd2b22c7cb4bd05c036821655d49a.png"},57479:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_ReportContentAsync-eb434d3157fb7884dc999828eb2d6a6e.png"},67942:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_RequestEmailAuthCodeAsync-72567693be0c3a74f8829f100dd7b9d5.png"},53784:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_SetLanguage-64c43dcff334858421a96bdf06ba16f9.png"},68768:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_ShutdownAsync-ad359e71788c08397a38cf61bcb0be00.png"},83175:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_SubmitModChangesAsync-4ad4759be516c2f3d9b45e350c3eaab7.png"},87149:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_SubmitModRatingAsync-a0e171b1e0c563c3cfdf3ad28389564b.png"},93124:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_SubmitNewModAsync-95991a02d50891bf9f8e62f8aa9c4161.png"},64303:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_SubmitNewModFileForMod-24f84c6a68b38837f8b0dfd6bf081b98.png"},72021:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_SubscribeToModAsync-93fd3340617ee4d44dea7078fb1199d4.png"},60291:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_UnmuteUserAsync-3d5fb9c4326a9e0083404c7122e2a1c8.png"},28297:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync-ffcaaed44b92c819c915849b1931814a.png"},93680:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_K2_VerifyUserAuthenticationAsync-5a26791679686730e9dcc7880e18c978.png"},71552:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_KillBackgroundThread-7568cfd2a189b287b269ceec5db5d079.png"},16501:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_PrioritizeTransferForMod-1eb6142c4dd34c75b675bb6246a0d6d8.png"},31214:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_QuerySystemInstallations-dff87f3a0a9c23416e81db47ca0f25da.png"},94564:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_QueryUserInstallations-ad0e0b0aa88244144d9da4210ee3d5d1.png"},703:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_QueryUserPurchasedMods-066afb38d53390a7926561bafc794593.png"},26349:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_QueryUserSubscriptions-73885d17f3df616346f77111ae8b8ad1.png"},63183:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_RunPendingHandlers-50f98a4a74afd398ef05e160d7717e88.png"},28862:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioSubsystem_SetLogLevel-183d6fd9ceeb266aefcf08587b244729.png"},53312:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},29301:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},74615:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},88883:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},66182:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},83505:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},96756:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},88371:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},15032:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},72668:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},42575:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioUIEnumLocalizationLibrary_FileSizeUnsigned64_ToText-4b00fdbdb167e45ea0acaffd8e54b333.png"},89670:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioUIEnumLocalizationLibrary_GetLocalizedTextForEnumByName-5849fa04c36e80e669f1faae64312cb1.png"},39200:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioUILocalizationLibrary_GetLocalizedTextFromDefaultTableByKey-f07c4cb6bbe86f7285e78f7564f65111.png"},99794:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},9105:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},1512:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},74136:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},1381:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},62576:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},84996:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},42506:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},29929:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},37173:(e,d,i)=>{i.d(d,{A:()=>s});const s=i.p+"assets/images/nd_img_ModioUnsigned64Library_MakeFromComponents-5bd42089022b89256170fe2c464b932d.png"},60609:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},47991:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},6119:(e,d,i)=>{i.d(d,{A:()=>s});const s=""},28453:(e,d,i)=>{i.d(d,{R:()=>t,x:()=>l});var s=i(96540);const r={},n=s.createContext(r);function t(e){const d=s.useContext(n);return s.useMemo((function(){return"function"==typeof e?e(d):{...d,...e}}),[d,e])}function l(e){let d;return d=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:t(e.components),s.createElement(n.Provider,{value:d},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/433ca74c.5daf400e.js b/Doc/assets/js/433ca74c.5daf400e.js new file mode 100644 index 00000000..82e4477a --- /dev/null +++ b/Doc/assets/js/433ca74c.5daf400e.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[5973],{45786:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>a,contentTitle:()=>r,default:()=>u,frontMatter:()=>s,metadata:()=>l,toc:()=>d});var i=t(74848),o=t(28453);const s={id:"ue-index",title:"Overview",slug:"/unreal/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/index.mdx"},r=void 0,l={id:"ue-index",title:"Overview",description:"The mod.io Unreal Engine plugin enables game developers to easily integrate mods into their Unreal Engine 5 games.",source:"@site/public/en-us/index.mdx",sourceDirName:".",slug:"/unreal/",permalink:"/unreal/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/index.mdx",tags:[],version:"current",frontMatter:{id:"ue-index",title:"Overview",slug:"/unreal/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/index.mdx"},sidebar:"sidebar",previous:{title:"Home",permalink:"/"},next:{title:"Installation & Setup",permalink:"/unreal/installation-and-setup/"}},a={},d=[{value:"Engine & platform compatability",id:"engine--platform-compatability",level:2},{value:"Engine compatability",id:"engine-compatability",level:3},{value:"Platform compatability",id:"platform-compatability",level:3},{value:"Features",id:"features",level:2}];function c(e){const n={a:"a",h2:"h2",h3:"h3",li:"li",p:"p",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,o.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.p,{children:"The mod.io Unreal Engine plugin enables game developers to easily integrate mods into their Unreal Engine 5 games."}),"\n",(0,i.jsxs)(n.p,{children:["The plugin provides a C++ and Blueprint interface around the ",(0,i.jsx)(n.a,{href:"/cppsdk/",children:"mod.io SDK"})," to connect to the ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/restapiref/",children:"mod.io REST API"}),". Also included is a set of UI components to easily create a custom user interface that matches your game's style, plus a comprehensive template UI to get you started."]}),"\n",(0,i.jsxs)(n.p,{children:["You can download the ",(0,i.jsx)(n.a,{href:"https://github.com/modio/modio-ue",children:"mod.io Unreal Engine plugin from GitHub"}),". Once you have, follow the ",(0,i.jsx)(n.a,{href:"installation-and-setup/",children:"installation guide"})," for installing and configuring the plugin. A series of ",(0,i.jsx)(n.a,{href:"getting-started/",children:"quick-start guides"})," are also available to familiarize yourself with the plugin's core features."]}),"\n",(0,i.jsx)(n.h2,{id:"engine--platform-compatability",children:"Engine & platform compatability"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io plugin is generally maintained to support the 3 most recent versions of Unreal Engine. If you are using an older version of the engine, you can access the last supported release of the engine from our ",(0,i.jsx)(n.a,{href:"https://github.com/modio/modio-ue/releases",children:"releases page"})," on GitHub."]}),"\n",(0,i.jsx)(n.h3,{id:"engine-compatability",children:"Engine compatability"}),"\n",(0,i.jsxs)(n.table,{children:[(0,i.jsx)(n.thead,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.th,{children:"Engine Version"}),(0,i.jsx)(n.th,{children:"Last Release"})]})}),(0,i.jsxs)(n.tbody,{children:[(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE4.26"}),(0,i.jsx)(n.td,{children:"2023.11"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE4.27"}),(0,i.jsx)(n.td,{children:"2023.11"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.0"}),(0,i.jsx)(n.td,{children:"2023.11"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.1"}),(0,i.jsx)(n.td,{children:"2024.6"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.2"}),(0,i.jsx)(n.td,{children:"Current"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.3"}),(0,i.jsx)(n.td,{children:"Current"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.4"}),(0,i.jsx)(n.td,{children:"Current"})]})]})]}),"\n",(0,i.jsx)(n.h3,{id:"platform-compatability",children:"Platform compatability"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine plugin supports Windows, Linux, Android and macOS as part of the public release. For access to Windows (GDK), Xbox, PlayStation\xae4, PlayStation\xae5 or Switch, follow the ",(0,i.jsx)(n.a,{href:"/platforms/console-sdks/",children:"instructions here"}),"."]}),"\n",(0,i.jsx)(n.h2,{id:"features",children:"Features"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"C++ and Blueprint support"}),"\n",(0,i.jsx)(n.li,{children:"UGC browsing and filtering"}),"\n",(0,i.jsx)(n.li,{children:"Automatic downloads and updates"}),"\n",(0,i.jsx)(n.li,{children:"Customizable component-based user interface"}),"\n",(0,i.jsx)(n.li,{children:"Permissive MIT/BSL-license"}),"\n",(0,i.jsx)(n.li,{children:"Async delegate-based interface"}),"\n",(0,i.jsx)(n.li,{children:"Non-blocking IO"}),"\n",(0,i.jsx)(n.li,{children:"SSO authentication for Steam, Epic Games Store, console platform and custom implementations"}),"\n"]})]})}function u(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(c,{...e})}):c(e)}},28453:(e,n,t)=>{t.d(n,{R:()=>r,x:()=>l});var i=t(96540);const o={},s=i.createContext(o);function r(e){const n=i.useContext(s);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function l(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:r(e.components),i.createElement(s.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/433ca74c.ab83666c.js b/Doc/assets/js/433ca74c.ab83666c.js deleted file mode 100644 index 08320e17..00000000 --- a/Doc/assets/js/433ca74c.ab83666c.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[5973],{45786:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>a,contentTitle:()=>l,default:()=>h,frontMatter:()=>s,metadata:()=>r,toc:()=>d});var i=t(74848),o=t(28453);const s={id:"ue-index",title:"Overview",slug:"/unreal/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/index.mdx"},l=void 0,r={id:"ue-index",title:"Overview",description:"The mod.io Unreal Engine plugin enables game developers to easily integrate mods into their Unreal Engine 5 games.",source:"@site/public/en-us/index.mdx",sourceDirName:".",slug:"/unreal/",permalink:"/unreal/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/index.mdx",tags:[],version:"current",frontMatter:{id:"ue-index",title:"Overview",slug:"/unreal/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/index.mdx"},sidebar:"sidebar",previous:{title:"Home",permalink:"/"},next:{title:"Installation & Setup",permalink:"/unreal/installation-and-setup/"}},a={},d=[{value:"Engine & platform compatability",id:"engine--platform-compatability",level:2},{value:"Engine compatability",id:"engine-compatability",level:3},{value:"Platform compatability",id:"platform-compatability",level:3},{value:"Features",id:"features",level:2}];function c(e){const n={a:"a",h2:"h2",h3:"h3",li:"li",p:"p",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,o.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.p,{children:"The mod.io Unreal Engine plugin enables game developers to easily integrate mods into their Unreal Engine 5 games."}),"\n",(0,i.jsxs)(n.p,{children:["The plugin provides a C++ and Blueprint interface around the ",(0,i.jsx)(n.a,{href:"/cppsdk/",children:"mod.io SDK"})," to connect to the ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/restapiref/",children:"mod.io REST API"}),". Also included is a set of UI components to easily create a custom user interface that matches your game's style, plus a comprehensive template UI to get you started."]}),"\n",(0,i.jsxs)(n.p,{children:["You can download the Unreal Engine plugin from the ",(0,i.jsx)(n.a,{href:"https://github.com/modio/modio-ue",children:"mod.io GitHub repository"})," and follow the ",(0,i.jsx)(n.a,{href:"installation-and-setup/",children:"installation guide"})," for installing and configuring the plugin. A series of ",(0,i.jsx)(n.a,{href:"getting-started/",children:"quick-start guides"})," are also available to familiarize yourself with the plugin's core features."]}),"\n",(0,i.jsx)(n.h2,{id:"engine--platform-compatability",children:"Engine & platform compatability"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io plugin is generally maintained to support the 3 most recent versions of Unreal Engine. If you are using an older version of the engine, you can access the last supported release of the engine from our ",(0,i.jsx)(n.a,{href:"https://github.com/modio/modio-ue/releases",children:"releases page"})," on GitHub."]}),"\n",(0,i.jsx)(n.h3,{id:"engine-compatability",children:"Engine compatability"}),"\n",(0,i.jsxs)(n.table,{children:[(0,i.jsx)(n.thead,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.th,{children:"Engine Version"}),(0,i.jsx)(n.th,{children:"Last Release"})]})}),(0,i.jsxs)(n.tbody,{children:[(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE4.26"}),(0,i.jsx)(n.td,{children:"2023.11"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE4.27"}),(0,i.jsx)(n.td,{children:"2023.11"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.0"}),(0,i.jsx)(n.td,{children:"2023.11"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.1"}),(0,i.jsx)(n.td,{children:"2024.6"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.2"}),(0,i.jsx)(n.td,{children:"Current"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.3"}),(0,i.jsx)(n.td,{children:"Current"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"UE5.4"}),(0,i.jsx)(n.td,{children:"Current"})]})]})]}),"\n",(0,i.jsx)(n.h3,{id:"platform-compatability",children:"Platform compatability"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine plugin supports Windows, Linux, Android and macOS as part of the public release. For access to Windows (GDK), Xbox, PlayStation\xae4, PlayStation\xae5 or Switch, follow the ",(0,i.jsx)(n.a,{href:"/platforms/console-sdks/",children:"instructions here"}),"."]}),"\n",(0,i.jsx)(n.h2,{id:"features",children:"Features"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"C++ and Blueprint support"}),"\n",(0,i.jsx)(n.li,{children:"UGC browsing and filtering"}),"\n",(0,i.jsx)(n.li,{children:"Automatic downloads and updates"}),"\n",(0,i.jsx)(n.li,{children:"Customizable component-based user interface"}),"\n",(0,i.jsx)(n.li,{children:"Permissive MIT/BSL-license"}),"\n",(0,i.jsx)(n.li,{children:"Async delegate-based interface"}),"\n",(0,i.jsx)(n.li,{children:"Non-blocking IO"}),"\n",(0,i.jsx)(n.li,{children:"SSO authentication for Steam, Epic Games Store, console platform and custom implementations"}),"\n"]})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(c,{...e})}):c(e)}},28453:(e,n,t)=>{t.d(n,{R:()=>l,x:()=>r});var i=t(96540);const o={},s=i.createContext(o);function l(e){const n=i.useContext(s);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:l(e.components),i.createElement(s.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/4b65d6a2.3e711868.js b/Doc/assets/js/4b65d6a2.3e711868.js new file mode 100644 index 00000000..6ac37ee5 --- /dev/null +++ b/Doc/assets/js/4b65d6a2.3e711868.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[433],{12291:(e,n,o)=>{o.r(n),o.d(n,{assets:()=>d,contentTitle:()=>l,default:()=>c,frontMatter:()=>r,metadata:()=>s,toc:()=>a});var t=o(74848),i=o(28453);const r={id:"ue-android-configuration",title:"Android Configuration",slug:"/unreal/android-configuration/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/android.mdx"},l=void 0,s={id:"ue-android-configuration",title:"Android Configuration",description:"Single Sign-On (5.4 onwards)",source:"@site/public/en-us/android.mdx",sourceDirName:".",slug:"/unreal/android-configuration/",permalink:"/unreal/android-configuration/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/android.mdx",tags:[],version:"current",frontMatter:{id:"ue-android-configuration",title:"Android Configuration",slug:"/unreal/android-configuration/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/android.mdx"},sidebar:"sidebar",previous:{title:"Mod Creation Tool",permalink:"/unreal/mod-creation-tool/"},next:{title:"Unreal Plugin API Reference",permalink:"/unreal/refdocs/"}},d={},a=[{value:"Single Sign-On (5.4 onwards)",id:"single-sign-on-54-onwards",level:2},{value:"Configuration",id:"configuration",level:3},{value:"Getting the Token",id:"getting-the-token",level:3},{value:"Single Sign-On (Pre-5.4)",id:"single-sign-on-pre-54",level:2},{value:"Configuration",id:"configuration-1",level:3},{value:"Getting the Token",id:"getting-the-token-1",level:3}];function u(e){const n={code:"code",h2:"h2",h3:"h3",p:"p",pre:"pre",...(0,i.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.h2,{id:"single-sign-on-54-onwards",children:"Single Sign-On (5.4 onwards)"}),"\n",(0,t.jsx)(n.p,{children:"As of UE5.4, there is better support for obtaining an IdToken for Google Authentication. However there are still some changes that need to be made."}),"\n",(0,t.jsxs)(n.p,{children:["If you are on a Source build, open ",(0,t.jsx)(n.code,{children:"Engine\\Plugins\\Online\\OnlineSubsystemGoogle\\OnlineSubsystemGoogle.uplugin"})," and add Android to the ",(0,t.jsx)(n.code,{children:"PlatformAllowList"})," array. If you are on a Launcher build, you can simply copy this plugin from this directory into your projects Plugins directory and make the appropriate change."]}),"\n",(0,t.jsx)(n.h3,{id:"configuration",children:"Configuration"}),"\n",(0,t.jsx)(n.p,{children:"Next, update your AndroidEngine.ini file to contain the following:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:'[OnlineSubsystem]\nDefaultPlatformService=GooglePlay\nNativePlatformService=Google\n\n[OnlineSubsystemGoogle]\nbEnabled=True\nClientId=your-client-id-here\nServerClientId=your-server-client-id-here\n\n\n[OnlineSubsystemGoogle.OnlineIdentityGoogle]\n+ScopeFields="email"\n+ScopeFields="profile"\n+ScopeFields="https://www.googleapis.com/auth/userinfo.email"\n+ScopeFields="https://www.googleapis.com/auth/userinfo.profile"\nbRequestIdToken=true\nbRequestOfflineAccess=true\n'})}),"\n",(0,t.jsx)(n.p,{children:"This will ensure that you are using OnlineSubsystemGoogle as the native platform OSS in order to perform SSO."}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"bRequestIdToken"})," will request an IdToken that you can pass to mod.io's authentication service. ",(0,t.jsx)(n.code,{children:"bRequestOfflineAccess"})," will request a server-side token that can be used as an alternative to the IdToken."]}),"\n",(0,t.jsx)(n.p,{children:"Ensure that you replace the ClientId and ServerClientId with your client IDs from Google Cloud Console. The ClientId should be the Android OAuth Credential that you have linked to your title in Google Play Console. ServerClientId should be the Web Application OAuth Credential that you configured."}),"\n",(0,t.jsx)(n.h3,{id:"getting-the-token",children:"Getting the Token"}),"\n",(0,t.jsx)(n.p,{children:"Once you have configured the above options, you should be able to obtain either an IdToken or ServerAuthCode that you can use with mod.io authentication."}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-cpp",children:"const IOnlineSubsystem* OnlineSubsystem = IOnlineSubsystem::GetByPlatform();\nconst IOnlineIdentityPtr OnlineIdentity = OnlineSubsystem->GetIdentityInterface();\n\nOnlineIdentity->AddOnLoginCompleteDelegate_Handle(0, NativeLoginComplete);\nOnlineIdentity->Login(0, FOnlineAccountCredentials());\n\n// In the NativeLoginComplete delegate, however you set it up, if it was a successful login, you can then obtain the Server Auth Token as follows:\nFString IdToken, ServerAuthToken;\n\nUserAccount.Get()->GetAuthAttribute(AUTH_ATTR_ID_TOKEN, IdToken);\nUserAccount.Get()->GetAuthAttribute(AUTH_ATTR_AUTHORIZATION_CODE, ServerAuthToken);\n\n// Now perform Auth to the mod.io service\nAuthParams.AuthToken = ServerAuthToken;\nAuthParams.bUserHasAcceptedTerms = true;\n \n// Alternatively you could set AuthParams.AuthToken = IdToken and use EModioAuthenticationProvider::GoogleIDToken\nModioSubsystem->AuthenticateUserExternalAsync(AuthParams, EModioAuthenticationProvider::GoogleServerSideToken, FOnErrorOnlyDelegateFast::CreateUObject(this, OnAuthenticationComplete));\n"})}),"\n",(0,t.jsx)(n.h2,{id:"single-sign-on-pre-54",children:"Single Sign-On (Pre-5.4)"}),"\n",(0,t.jsx)(n.p,{children:"For Unreal Engine versions before 5.4, you need to make some engine-level modifications in order for Android SSO to work. You must be using a source build rather than an engine build."}),"\n",(0,t.jsxs)(n.p,{children:["Open ",(0,t.jsx)(n.code,{children:"Engine\\Plugins\\Online\\OnlineSubsystemGoogle\\Source\\ThirdParty\\Android\\Java\\GoogleLogin.java"})," and change the ",(0,t.jsx)(n.code,{children:"init"})," method to uncomment ",(0,t.jsx)(n.code,{children:".requestServerAuthCode(serverClientId)"}),"."]}),"\n",(0,t.jsxs)(n.p,{children:["Change the ",(0,t.jsx)(n.code,{children:"getAuthTokenJsonStr"})," method to the following:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-cpp",children:'private String getAuthTokenJsonStr(GoogleSignInAccount acct)\n{\n if (acct != null)\n {\n return "{\\"access_token\\":\\""+ acct.getServerAuthCode() + "\\"," +\n "\\"refresh_token\\":\\"androidInternal\\"," +\n "\\"id_token\\":\\""+ acct.getIdToken() + "\\"}";\n }\n return "";\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Finally, Open OnlineSubsystemGoogle.uplugin and add Android to the ",(0,t.jsx)(n.code,{children:"PlatformAllowList"})," array."]}),"\n",(0,t.jsx)(n.h3,{id:"configuration-1",children:"Configuration"}),"\n",(0,t.jsx)(n.p,{children:"Once you have made the above changes, update your AndroidEngine.ini to contain the following:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:'[OnlineSubsystem]\nDefaultPlatformService=GooglePlay\nNativePlatformService=Google\n\n[OnlineSubsystemGoogle]\nbEnabled=True\nClientId=your-client-id-here\nServerClientId=your-server-client-id-here\n\n[OnlineSubsystemGoogle.OnlineIdentityGoogle]\n+ScopeFields="email"\n+ScopeFields="profile"\n+ScopeFields="https://www.googleapis.com/auth/userinfo.email"\n+ScopeFields="https://www.googleapis.com/auth/userinfo.profile"\n'})}),"\n",(0,t.jsx)(n.p,{children:"This will ensure that you are using OnlineSubsystemGoogle as the native platform OSS in order to perform SSO."}),"\n",(0,t.jsx)(n.p,{children:"Ensure that you replace the ClientId and ServerClientId with your client IDs from Google Cloud Console. The ClientId should be the Android OAuth Credential that you have linked to your title in Google Play Console. ServerClientId should be the Web Application OAuth Credential that you configured."}),"\n",(0,t.jsx)(n.h3,{id:"getting-the-token-1",children:"Getting the Token"}),"\n",(0,t.jsx)(n.p,{children:"Once you have configured the above options, you should be able to obtain a server side token that you can use for auth with mod.io."}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-cpp",children:"const IOnlineSubsystem* OnlineSubsystem = IOnlineSubsystem::GetByPlatform();\nconst IOnlineIdentityPtr OnlineIdentity = OnlineSubsystem->GetIdentityInterface();\n\nOnlineIdentity->AddOnLoginCompleteDelegate_Handle(0, NativeLoginComplete);\nOnlineIdentity->Login(0, FOnlineAccountCredentials());\n\n// In the NativeLoginComplete delegate, however you set it up, if it was a successful login, you can then obtain the Server Auth Token as follows:\nconst FString PlatformToken = OnlineIdentity->GetAuthToken(0);\n\n// Now perform Auth to the mod.io service\nAuthParams.AuthToken = PlatformToken;\nAuthParams.bUserHasAcceptedTerms = true;\n \nModioSubsystem->AuthenticateUserExternalAsync(AuthParams, EModioAuthenticationProvider::GoogleServerSideToken, FOnErrorOnlyDelegateFast::CreateUObject(this, OnAuthenticationComplete));\n"})})]})}function c(e={}){const{wrapper:n}={...(0,i.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(u,{...e})}):u(e)}},28453:(e,n,o)=>{o.d(n,{R:()=>l,x:()=>s});var t=o(96540);const i={},r=t.createContext(i);function l(e){const n=t.useContext(r);return t.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function s(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:l(e.components),t.createElement(r.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/4b65d6a2.b96e2355.js b/Doc/assets/js/4b65d6a2.b96e2355.js deleted file mode 100644 index aeafaa69..00000000 --- a/Doc/assets/js/4b65d6a2.b96e2355.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[433],{12291:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>d,contentTitle:()=>l,default:()=>c,frontMatter:()=>r,metadata:()=>s,toc:()=>a});var o=t(74848),i=t(28453);const r={id:"ue-android-configuration",title:"Android Configuration",slug:"/unreal/android-configuration/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/android.mdx"},l=void 0,s={id:"ue-android-configuration",title:"Android Configuration",description:"Single Sign-On (5.4 onwards)",source:"@site/public/en-us/android.mdx",sourceDirName:".",slug:"/unreal/android-configuration/",permalink:"/unreal/android-configuration/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/android.mdx",tags:[],version:"current",frontMatter:{id:"ue-android-configuration",title:"Android Configuration",slug:"/unreal/android-configuration/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/android.mdx"},sidebar:"sidebar",previous:{title:"Mod Creation Tool",permalink:"/unreal/mod-creation-tool/"},next:{title:"Unreal Plugin API Reference",permalink:"/unreal/refdocs/"}},d={},a=[{value:"Single Sign-On (5.4 onwards)",id:"single-sign-on-54-onwards",level:2},{value:"Configuration",id:"configuration",level:3},{value:"Getting the Token",id:"getting-the-token",level:3},{value:"Single Sign-On (Pre-5.4)",id:"single-sign-on-pre-54",level:2},{value:"Configuration",id:"configuration-1",level:3},{value:"Getting the Token",id:"getting-the-token-1",level:3}];function u(e){const n={code:"code",h2:"h2",h3:"h3",p:"p",pre:"pre",...(0,i.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(n.h2,{id:"single-sign-on-54-onwards",children:"Single Sign-On (5.4 onwards)"}),"\n",(0,o.jsx)(n.p,{children:"As of UE5.4, there is better support for obtaining an IdToken for Google Authentication. However there are still some changes that need to be made."}),"\n",(0,o.jsxs)(n.p,{children:["If you are on a Source build, open ",(0,o.jsx)(n.code,{children:"Engine\\Plugins\\Online\\OnlineSubsystemGoogle\\OnlineSubsystemGoogle.uplugin"})," and add Android to the ",(0,o.jsx)(n.code,{children:"PlatformAllowList"})," array. If you are on a Launcher build, you can simply copy this plugin from this directory into your projects Plugins directory and make the appropriate change."]}),"\n",(0,o.jsx)(n.h3,{id:"configuration",children:"Configuration"}),"\n",(0,o.jsx)(n.p,{children:"Next, update your AndroidEngine.ini file to contain the following:"}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{children:'[OnlineSubsystem]\nDefaultPlatformService=GooglePlay\nNativePlatformService=Google\n\n[OnlineSubsystemGoogle]\nbEnabled=True\nClientId=your-client-id-here\nServerClientId=your-server-client-id-here\n\n\n[OnlineSubsystemGoogle.OnlineIdentityGoogle]\n+ScopeFields="email"\n+ScopeFields="profile"\n+ScopeFields="https://www.googleapis.com/auth/userinfo.email"\n+ScopeFields="https://www.googleapis.com/auth/userinfo.profile"\nbRequestIdToken=true\nbRequestOfflineAccess=true\n'})}),"\n",(0,o.jsx)(n.p,{children:"This will ensure that you are using OnlineSubsystemGoogle as the native platform OSS in order to perform SSO."}),"\n",(0,o.jsxs)(n.p,{children:[(0,o.jsx)(n.code,{children:"bRequestIdToken"})," will request an IdToken that you can pass to mod.io's authentication service. ",(0,o.jsx)(n.code,{children:"bRequestOfflineAccess"})," will request a server-side token that can be used as an alternative to the IdToken."]}),"\n",(0,o.jsx)(n.p,{children:"Ensure that you replace the ClientId and ServerClientId with your client IDs from Google Cloud Console. The ClientId should be the Android OAuth Credential that you have linked to your title in Google Play Console. ServerClientId should be the Web Application OAuth Credential that you configured."}),"\n",(0,o.jsx)(n.h3,{id:"getting-the-token",children:"Getting the Token"}),"\n",(0,o.jsx)(n.p,{children:"Once you have configured the above options, you should be able to obtain either an IdToken or ServerAuthCode that you can use with mod.io authentication."}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-cpp",children:"const IOnlineSubsystem* OnlineSubsystem = IOnlineSubsystem::GetByPlatform();\nconst IOnlineIdentityPtr OnlineIdentity = OnlineSubsystem->GetIdentityInterface();\n\nOnlineIdentity->AddOnLoginCompleteDelegate_Handle(0, NativeLoginComplete);\nOnlineIdentity->Login(0, FOnlineAccountCredentials());\n\n// In the NativeLoginComplete delegate, however you set it up, if it was a successful login, you can then obtain the Server Auth Token as follows:\nFString IdToken, ServerAuthToken;\n\nUserAccount.Get()->GetAuthAttribute(AUTH_ATTR_ID_TOKEN, IdToken);\nUserAccount.Get()->GetAuthAttribute(AUTH_ATTR_AUTHORIZATION_CODE, ServerAuthToken);\n\n// Now perform Auth to the mod.io service\nAuthParams.AuthToken = ServerAuthToken;\nAuthParams.bUserHasAcceptedTerms = true;\n \n// Alternatively you could set AuthParams.AuthToken = IdToken and use EModioAuthenticationProvider::GoogleIDToken\nModioSubsystem->AuthenticateUserExternalAsync(AuthParams, EModioAuthenticationProvider::GoogleServerSideToken, FOnErrorOnlyDelegateFast::CreateUObject(this, OnAuthenticationComplete));\n"})}),"\n",(0,o.jsx)(n.h2,{id:"single-sign-on-pre-54",children:"Single Sign-On (Pre-5.4)"}),"\n",(0,o.jsx)(n.p,{children:"For Unreal Engine versions before 5.4, you need to make some engine-level modifications in order for Android SSO to work. You must be using a source build rather than an engine build."}),"\n",(0,o.jsxs)(n.p,{children:["Open ",(0,o.jsx)(n.code,{children:"Engine\\Plugins\\Online\\OnlineSubsystemGoogle\\Source\\ThirdParty\\Android\\Java\\GoogleLogin.java"})," and change the ",(0,o.jsx)(n.code,{children:"init"})," method to uncomment ",(0,o.jsx)(n.code,{children:".requestServerAuthCode(serverClientId)"}),"."]}),"\n",(0,o.jsxs)(n.p,{children:["Change the ",(0,o.jsx)(n.code,{children:"getAuthTokenJsonStr"})," method to the following:"]}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-cpp",children:'private String getAuthTokenJsonStr(GoogleSignInAccount acct)\n{\n if (acct != null)\n {\n return "{\\"access_token\\":\\""+ acct.getServerAuthCode() + "\\"," +\n "\\"refresh_token\\":\\"androidInternal\\"," +\n "\\"id_token\\":\\""+ acct.getIdToken() + "\\"}";\n }\n return "";\n}\n'})}),"\n",(0,o.jsxs)(n.p,{children:["Finally, Open OnlineSubsystemGoogle.uplugin and add Android to the ",(0,o.jsx)(n.code,{children:"PlatformAllowList"})," array."]}),"\n",(0,o.jsx)(n.h3,{id:"configuration-1",children:"Configuration"}),"\n",(0,o.jsx)(n.p,{children:"Once you have made the above changes, update your AndroidEngine.ini to contain the following:"}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{children:'[OnlineSubsystem]\nDefaultPlatformService=GooglePlay\nNativePlatformService=Google\n\n[OnlineSubsystemGoogle]\nbEnabled=True\nClientId=your-client-id-here\nServerClientId=your-server-client-id-here\n\n[OnlineSubsystemGoogle.OnlineIdentityGoogle]\n+ScopeFields="email"\n+ScopeFields="profile"\n+ScopeFields="https://www.googleapis.com/auth/userinfo.email"\n+ScopeFields="https://www.googleapis.com/auth/userinfo.profile"\n'})}),"\n",(0,o.jsx)(n.p,{children:"This will ensure that you are using OnlineSubsystemGoogle as the native platform OSS in order to perform SSO."}),"\n",(0,o.jsx)(n.p,{children:"Ensure that you replace the ClientId and ServerClientId with your client IDs from Google Cloud Console. The ClientId should be the Android OAuth Credential that you have linked to your title in Google Play Console. ServerClientId should be the Web Application OAuth Credential that you configured."}),"\n",(0,o.jsx)(n.h3,{id:"getting-the-token-1",children:"Getting the Token"}),"\n",(0,o.jsx)(n.p,{children:"Once you have configured the above options, you should be able to obtain a server side token that you can use for auth with mod.io."}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-cpp",children:"const IOnlineSubsystem* OnlineSubsystem = IOnlineSubsystem::GetByPlatform();\nconst IOnlineIdentityPtr OnlineIdentity = OnlineSubsystem->GetIdentityInterface();\n\nOnlineIdentity->AddOnLoginCompleteDelegate_Handle(0, NativeLoginComplete);\nOnlineIdentity->Login(0, FOnlineAccountCredentials());\n\n// In the NativeLoginComplete delegate, however you set it up, if it was a successful login, you can then obtain the Server Auth Token as follows:\nconst FString PlatformToken = OnlineIdentity->GetAuthToken(0);\n\n// Now perform Auth to the mod.io service\nAuthParams.AuthToken = PlatformToken;\nAuthParams.bUserHasAcceptedTerms = true;\n \nModioSubsystem->AuthenticateUserExternalAsync(AuthParams, EModioAuthenticationProvider::GoogleServerSideToken, FOnErrorOnlyDelegateFast::CreateUObject(this, OnAuthenticationComplete));\n"})})]})}function c(e={}){const{wrapper:n}={...(0,i.R)(),...e.components};return n?(0,o.jsx)(n,{...e,children:(0,o.jsx)(u,{...e})}):u(e)}},28453:(e,n,t)=>{t.d(n,{R:()=>l,x:()=>s});var o=t(96540);const i={},r=o.createContext(i);function l(e){const n=o.useContext(r);return o.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function s(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:l(e.components),o.createElement(r.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/4dd573c0.3ed96b98.js b/Doc/assets/js/4dd573c0.3ed96b98.js new file mode 100644 index 00000000..fa71bdfb --- /dev/null +++ b/Doc/assets/js/4dd573c0.3ed96b98.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[5533],{49455:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>a,contentTitle:()=>l,default:()=>u,frontMatter:()=>r,metadata:()=>s,toc:()=>c});var i=n(74848),o=n(28453);const r={id:"ue-marketplace",title:"Marketplace",slug:"/unreal/marketplace/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/marketplace.mdx"},l=void 0,s={id:"ue-marketplace",title:"Marketplace",description:"Setup",source:"@site/public/en-us/marketplace.mdx",sourceDirName:".",slug:"/unreal/marketplace/",permalink:"/unreal/marketplace/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/marketplace.mdx",tags:[],version:"current",frontMatter:{id:"ue-marketplace",title:"Marketplace",slug:"/unreal/marketplace/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/marketplace.mdx"},sidebar:"sidebar",previous:{title:"Metrics Play Sessions",permalink:"/unreal/getting-started/metrics-play-sessions"},next:{title:"Asset Support for Mods",permalink:"/unreal/asset-support-for-mods/"}},a={},c=[{value:"Setup",id:"setup",level:2},{value:"Implementation",id:"implementation",level:2},{value:"References",id:"references",level:2}];function d(e){const t={a:"a",code:"code",h2:"h2",li:"li",p:"p",ul:"ul",...(0,o.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(t.h2,{id:"setup",children:"Setup"}),"\n",(0,i.jsx)(t.p,{children:"The mod.io Unreal Engine Plugin supports in-game integration of the mod.io Marketplace - exposing the data and functions required to build a UGC store in-game."}),"\n",(0,i.jsx)(t.p,{children:"The mod.io monetization features are need to be enabled through the web UI and, if selling the virtual currency through a platform store, will need a service to consume the virtual currency entitlements and apply them to player's mod.io account."}),"\n",(0,i.jsx)(t.h2,{id:"implementation",children:"Implementation"}),"\n",(0,i.jsx)(t.p,{children:"Marketplace functionality is accessible via the mod.io Subsystem along with all other functionality and as Blueprint nodes. Functions specific to monetization include:"}),"\n",(0,i.jsxs)(t.ul,{children:["\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.code,{children:"GetUserWalletBalanceAsync"})}),"\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.code,{children:"FetchUserPurchasesAsync"})}),"\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.code,{children:"PurchaseModAsync"})}),"\n"]}),"\n",(0,i.jsx)(t.h2,{id:"references",children:"References"}),"\n",(0,i.jsxs)(t.ul,{children:["\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.a,{href:"/monetization/",children:"Monetization Introduction"})}),"\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.a,{href:"/cppsdk/getting-started/#monetization",children:"SDK Monetization Guide"})}),"\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.a,{href:"/web-services/marketplace/overview/",children:"Purchase Server Implementation"})}),"\n"]})]})}function u(e={}){const{wrapper:t}={...(0,o.R)(),...e.components};return t?(0,i.jsx)(t,{...e,children:(0,i.jsx)(d,{...e})}):d(e)}},28453:(e,t,n)=>{n.d(t,{R:()=>l,x:()=>s});var i=n(96540);const o={},r=i.createContext(o);function l(e){const t=i.useContext(r);return i.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function s(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:l(e.components),i.createElement(r.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/4dd573c0.c971e0ef.js b/Doc/assets/js/4dd573c0.c971e0ef.js deleted file mode 100644 index 900ec7ac..00000000 --- a/Doc/assets/js/4dd573c0.c971e0ef.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[5533],{49455:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>a,contentTitle:()=>l,default:()=>u,frontMatter:()=>r,metadata:()=>s,toc:()=>c});var i=n(74848),o=n(28453);const r={id:"ue-marketplace",title:"Marketplace",slug:"/unreal/marketplace/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/marketplace.mdx"},l=void 0,s={id:"ue-marketplace",title:"Marketplace",description:"Setup",source:"@site/public/en-us/marketplace.mdx",sourceDirName:".",slug:"/unreal/marketplace/",permalink:"/unreal/marketplace/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/marketplace.mdx",tags:[],version:"current",frontMatter:{id:"ue-marketplace",title:"Marketplace",slug:"/unreal/marketplace/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/marketplace.mdx"},sidebar:"sidebar",previous:{title:"Metrics Play Sessions",permalink:"/unreal/getting-started/metrics-play-sessions"},next:{title:"Asset Support for Mods",permalink:"/unreal/asset-support-for-mods/"}},a={},c=[{value:"Setup",id:"setup",level:2},{value:"Implementation",id:"implementation",level:2},{value:"References",id:"references",level:2}];function d(e){const t={a:"a",code:"code",h2:"h2",li:"li",p:"p",ul:"ul",...(0,o.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(t.h2,{id:"setup",children:"Setup"}),"\n",(0,i.jsx)(t.p,{children:"The mod.io Unreal Engine Plugin supports in-game integration of the mod.io Marketplace - exposing the data and functions required to build a UGC store in-game."}),"\n",(0,i.jsx)(t.p,{children:"The mod.io monetization features are need to be enabled through the web UI and, if selling the virtual currency through a platform store, will need a service to consume the virtual currency entitlements and apply them to player's mod.io account."}),"\n",(0,i.jsx)(t.h2,{id:"implementation",children:"Implementation"}),"\n",(0,i.jsx)(t.p,{children:"Marketplace functionality is accessible via the mod.io Subsystem along with all other functionality and as Blueprint nodes. Functions specific to monetization include:"}),"\n",(0,i.jsxs)(t.ul,{children:["\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.code,{children:"GetUserWalletBalanceAsync"})}),"\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.code,{children:"FetchUserPurchasesAsync"})}),"\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.code,{children:"PurchaseModAsync"})}),"\n"]}),"\n",(0,i.jsx)(t.h2,{id:"references",children:"References"}),"\n",(0,i.jsxs)(t.ul,{children:["\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.a,{href:"/monetization/",children:"Monetization Introduction"})}),"\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.a,{href:"/cppsdk/getting-started/#monetization",children:"SDK Monetization Guide"})}),"\n",(0,i.jsx)(t.li,{children:(0,i.jsx)(t.a,{href:"/web-services/marketplace/overview/",children:"Purchase Server Implementation"})}),"\n"]})]})}function u(e={}){const{wrapper:t}={...(0,o.R)(),...e.components};return t?(0,i.jsx)(t,{...e,children:(0,i.jsx)(d,{...e})}):d(e)}},28453:(e,t,n)=>{n.d(t,{R:()=>l,x:()=>s});var i=n(96540);const o={},r=i.createContext(o);function l(e){const t=i.useContext(r);return i.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function s(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:l(e.components),i.createElement(r.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/4f8ccd8c.4c953df7.js b/Doc/assets/js/4f8ccd8c.4c953df7.js new file mode 100644 index 00000000..e86952a5 --- /dev/null +++ b/Doc/assets/js/4f8ccd8c.4c953df7.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[3124],{19169:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>l,default:()=>b,frontMatter:()=>r,metadata:()=>d,toc:()=>c});var o=n(74848),i=n(28453),a=n(27064),s=n(89236);const r={id:"ue-submit-mods",title:"In-Game Mod Submission",slug:"/unreal/getting-started/submit-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/submit-mods.mdx"},l=void 0,d={id:"getting-started/ue-submit-mods",title:"In-Game Mod Submission",description:"Submitting a mod from inside your game and making it visible to other players involves two steps:",source:"@site/public/en-us/getting-started/submit-mods.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/submit-mods",permalink:"/unreal/getting-started/submit-mods",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/submit-mods.mdx",tags:[],version:"current",frontMatter:{id:"ue-submit-mods",title:"In-Game Mod Submission",slug:"/unreal/getting-started/submit-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/submit-mods.mdx"},sidebar:"sidebar",previous:{title:"Mod Subscriptions & Management",permalink:"/unreal/getting-started/mod-subscriptions"},next:{title:"Edit an Existing Mod",permalink:"/unreal/getting-started/edit-mods"}},u={},c=[{value:"Submitting a new mod",id:"submitting-a-new-mod",level:2},{value:"Submitting a file for a mod",id:"submitting-a-file-for-a-mod",level:2}];function m(e){const t={a:"a",admonition:"admonition",code:"code",h2:"h2",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,i.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(t.p,{children:"Submitting a mod from inside your game and making it visible to other players involves two steps:"}),"\n",(0,o.jsxs)(t.ul,{children:["\n",(0,o.jsx)(t.li,{children:"Submission of the mod"}),"\n",(0,o.jsx)(t.li,{children:'Submission of the mod\u2019s data (aka "the mod file")'}),"\n"]}),"\n",(0,o.jsxs)(t.p,{children:["These steps are outlined below. Mods can also be edited after submission, as ",(0,o.jsx)(t.a,{href:"edit-mods",children:"detailed here"})]}),"\n",(0,o.jsx)(t.h2,{id:"submitting-a-new-mod",children:"Submitting a new mod"}),"\n",(0,o.jsxs)(t.p,{children:["To submit a mod, you must first create a mod handle using ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#getmodcreationhandle",children:(0,o.jsx)(t.code,{children:"GetModCreationHandle"})}),", and use that handle when calling ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#submitnewmodasync",children:(0,o.jsx)(t.code,{children:"SubmitNewModAsync"})}),". Note that the newly created mod will remain hidden until a mod file is added in the next step."]}),"\n",(0,o.jsxs)(a.A,{"group-id":"languages",children:[(0,o.jsx)(s.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(t.p,{children:(0,o.jsx)(t.img,{alt:"submit_new_mod",src:n(92695).A+"",width:"870",height:"519"})})}),(0,o.jsx)(s.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-cpp",children:'void UModioManagerSubsystem::SubmitNewMod()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tFModioModCreationHandle Handle = Subsystem->GetModCreationHandle();\n\n \tFModioCreateModParams Params;\n \tParams.Name = TEXT("My Amazing Mod");\n \tParams.Description = TEXT("This mod does amazing things");\n \tParams.PathToLogoFile = TEXT("C:\\\\path\\\\to\\\\image.png");\n\n \tSubsystem->SubmitNewModAsync(Handle, Params, FOnSubmitNewModDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnSubmitNewModComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnSubmitNewModComplete(FModioErrorCode ErrorCode, TOptional ModId)\n{\n if (!ErrorCode)\n {\n \t// Mod was successfully submitted \n \t// We can now call SubmitNewModFileForMod with this ModId\n }\n}\n'})})})]}),"\n",(0,o.jsx)(t.h2,{id:"submitting-a-file-for-a-mod",children:"Submitting a file for a mod"}),"\n",(0,o.jsxs)(t.p,{children:["Once you have successfully submitted a mod, you can submit a mod file for that mod using ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#submitnewmodfileformod",children:(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"})}),". When you submit a mod file, you pass a ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#modiocreatemodfileparams",children:(0,o.jsx)(t.code,{children:"ModioCreateModFileParams"})})," containing the directory containing all the files that you want to submit. The plugin will compress this folder into a .zip file and upload it as the active version of the mod."]}),"\n",(0,o.jsxs)(t.p,{children:["In the future, if the mod is updated and requires a new mod file, ",(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"})," can be called again. The most recent mod file uploaded by ",(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"})," will be set as the active version."]}),"\n",(0,o.jsx)(t.admonition,{type:"note",children:(0,o.jsxs)(t.p,{children:["There is no callback for ",(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"}),"; you\u2019ll be notified of the completed upload by your ",(0,o.jsx)(t.a,{href:"mod-subscriptions#installation-management",children:(0,o.jsx)(t.strong,{children:"mod management callback"})}),"."]})}),"\n",(0,o.jsxs)(a.A,{"group-id":"languages",children:[(0,o.jsxs)(s.A,{value:"blueprint",label:"Blueprint",children:[(0,o.jsx)(t.p,{children:"After the callback for submitting a mod has completed, you can get the Mod Id to use for file submission."}),(0,o.jsx)(t.p,{children:(0,o.jsx)(t.img,{alt:"submit_new_mod_file",src:n(12214).A+"",width:"862",height:"483"})})]}),(0,o.jsx)(s.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-cpp",children:'void UModioManagerSubsystem::SubmitNewModFile(FModioModID ModId)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tFModioCreateModFileParams Params;\n \tParams.PathToModRootDirectory = TEXT("C:\\\\path\\\\to\\\\mod-folder");\n \tSubsystem->SubmitNewModFileForMod(ModId, Params);\n }\n}\n'})})})]})]})}function b(e={}){const{wrapper:t}={...(0,i.R)(),...e.components};return t?(0,o.jsx)(t,{...e,children:(0,o.jsx)(m,{...e})}):m(e)}},19365:(e,t,n)=>{n.d(t,{A:()=>s});n(96540);var o=n(18215);const i={tabItem:"tabItem_Ymn6"};var a=n(74848);function s(e){let{children:t,hidden:n,className:s}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,o.A)(i.tabItem,s),hidden:n,children:t})}},11470:(e,t,n)=>{n.d(t,{A:()=>y});var o=n(96540),i=n(18215),a=n(23104),s=n(56347),r=n(205),l=n(57485),d=n(31682),u=n(70679);function c(e){return o.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:n}=e;return(0,o.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:n,attributes:o,default:i}}=e;return{value:t,label:n,attributes:o,default:i}}))}(n);return function(e){const t=(0,d.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function b(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:n}=e;const i=(0,s.W6)(),a=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l.aZ)(a),(0,o.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(i.location.search);t.set(a,e),i.replace({...i.location,search:t.toString()})}),[a,i])]}function f(e){const{defaultValue:t,queryString:n=!1,groupId:i}=e,a=m(e),[s,l]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!b({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const o=n.find((e=>e.default))??n[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:t,tabValues:a}))),[d,c]=h({queryString:n,groupId:i}),[f,g]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[i,a]=(0,u.Dv)(n);return[i,(0,o.useCallback)((e=>{n&&a.set(e)}),[n,a])]}({groupId:i}),p=(()=>{const e=d??f;return b({value:e,tabValues:a})?e:null})();(0,r.A)((()=>{p&&l(p)}),[p]);return{selectedValue:s,selectValue:(0,o.useCallback)((e=>{if(!b({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),g(e)}),[c,g,a]),tabValues:a}}var g=n(92303);const p={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var v=n(74848);function x(e){let{className:t,block:n,selectedValue:o,selectValue:s,tabValues:r}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,a.a_)(),u=e=>{const t=e.currentTarget,n=l.indexOf(t),i=r[n].value;i!==o&&(d(t),s(i))},c=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const n=l.indexOf(e.currentTarget)+1;t=l[n]??l[0];break}case"ArrowLeft":{const n=l.indexOf(e.currentTarget)-1;t=l[n]??l[l.length-1];break}}t?.focus()};return(0,v.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,i.A)("tabs",{"tabs--block":n},t),children:r.map((e=>{let{value:t,label:n,attributes:a}=e;return(0,v.jsx)("li",{role:"tab",tabIndex:o===t?0:-1,"aria-selected":o===t,ref:e=>l.push(e),onKeyDown:c,onClick:u,...a,className:(0,i.A)("tabs__item",p.tabItem,a?.className,{"tabs__item--active":o===t}),children:n??t},t)}))})}function w(e){let{lazy:t,children:n,selectedValue:i}=e;const a=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===i));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return(0,v.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==i})))})}function M(e){const t=f(e);return(0,v.jsxs)("div",{className:(0,i.A)("tabs-container",p.tabList),children:[(0,v.jsx)(x,{...t,...e}),(0,v.jsx)(w,{...t,...e})]})}function y(e){const t=(0,g.A)();return(0,v.jsx)(M,{...e,children:c(e.children)},String(t))}},89236:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(19365),i=(n(96540),n(74848));function a(e){return(0,i.jsx)(i.Fragment,{children:(0,i.jsx)(o.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(11470),i=(n(96540),n(74848));function a(e){return(0,i.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,i.jsx)(o.A,{...e})})}},92695:(e,t,n)=>{n.d(t,{A:()=>o});const o=n.p+"assets/images/submit_new_mod-d860f68584f2f0b67dd74c6e95467fcb.png"},12214:(e,t,n)=>{n.d(t,{A:()=>o});const o=n.p+"assets/images/submit_new_mod_file-e1cde8dca73f1d2d5985433981e8c5a3.png"},28453:(e,t,n)=>{n.d(t,{R:()=>s,x:()=>r});var o=n(96540);const i={},a=o.createContext(i);function s(e){const t=o.useContext(a);return o.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function r(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:s(e.components),o.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/4f8ccd8c.be71a477.js b/Doc/assets/js/4f8ccd8c.be71a477.js deleted file mode 100644 index 574e14a4..00000000 --- a/Doc/assets/js/4f8ccd8c.be71a477.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[3124],{19169:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>d,default:()=>b,frontMatter:()=>r,metadata:()=>l,toc:()=>c});var o=n(74848),i=n(28453),a=n(27064),s=n(89236);const r={id:"ue-submit-mods",title:"In-Game Mod Submission",slug:"/unreal/getting-started/submit-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/submit-mods.mdx"},d=void 0,l={id:"getting-started/ue-submit-mods",title:"In-Game Mod Submission",description:"Submitting a mod from inside your game and making it visible to other players involves two steps:",source:"@site/public/en-us/getting-started/submit-mods.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/submit-mods",permalink:"/unreal/getting-started/submit-mods",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/submit-mods.mdx",tags:[],version:"current",frontMatter:{id:"ue-submit-mods",title:"In-Game Mod Submission",slug:"/unreal/getting-started/submit-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/submit-mods.mdx"},sidebar:"sidebar",previous:{title:"Mod Subscriptions & Management",permalink:"/unreal/getting-started/mod-subscriptions"},next:{title:"Edit an Existing Mod",permalink:"/unreal/getting-started/edit-mods"}},u={},c=[{value:"Submitting a new mod",id:"submitting-a-new-mod",level:2},{value:"Submitting a file for a mod",id:"submitting-a-file-for-a-mod",level:2}];function m(e){const t={a:"a",admonition:"admonition",code:"code",h2:"h2",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,i.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(t.p,{children:"Submitting a mod from inside your game and making it visible to other players involves two steps:"}),"\n",(0,o.jsxs)(t.ul,{children:["\n",(0,o.jsx)(t.li,{children:"Submission of the mod"}),"\n",(0,o.jsx)(t.li,{children:'Submission of the mod\u2019s data (aka "the mod file")'}),"\n"]}),"\n",(0,o.jsxs)(t.p,{children:["These steps are outlined below. Mods can also be edited after submission, as ",(0,o.jsx)(t.a,{href:"edit-mods",children:"detailed here"})]}),"\n",(0,o.jsx)(t.h2,{id:"submitting-a-new-mod",children:"Submitting a new mod"}),"\n",(0,o.jsxs)(t.p,{children:["To submit a mod, you must first create a mod handle using ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_GetModCreationHandle",children:(0,o.jsx)(t.code,{children:"GetModCreationHandle"})}),", and use that handle when calling ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_SubmitNewModAsync",children:(0,o.jsx)(t.code,{children:"SubmitNewModAsync"})}),". Note that the newly created mod will remain hidden until a mod file is added in the next step."]}),"\n",(0,o.jsxs)(a.A,{"group-id":"languages",children:[(0,o.jsx)(s.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(t.p,{children:(0,o.jsx)(t.img,{alt:"submit_new_mod",src:n(17792).A+"",width:"870",height:"519"})})}),(0,o.jsx)(s.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-cpp",children:'void UModioManagerSubsystem::SubmitNewMod()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tFModioModCreationHandle Handle = Subsystem->GetModCreationHandle();\n\n \tFModioCreateModParams Params;\n \tParams.Name = TEXT("My Amazing Mod");\n \tParams.Description = TEXT("This mod does amazing things");\n \tParams.PathToLogoFile = TEXT("C:\\\\path\\\\to\\\\image.png");\n\n \tSubsystem->SubmitNewModAsync(Handle, Params, FOnSubmitNewModDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnSubmitNewModComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnSubmitNewModComplete(FModioErrorCode ErrorCode, TOptional ModId)\n{\n if (!ErrorCode)\n {\n \t// Mod was successfully submitted \n \t// We can now call SubmitNewModFileForMod with this ModId\n }\n}\n'})})})]}),"\n",(0,o.jsx)(t.h2,{id:"submitting-a-file-for-a-mod",children:"Submitting a file for a mod"}),"\n",(0,o.jsxs)(t.p,{children:["Once you have successfully submitted a mod, you can submit a mod file for that mod using ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_SubmitNewModFileForMod",children:(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"})}),". When you submit a mod file, you pass a ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#_modiocreatemodfileparams",children:(0,o.jsx)(t.code,{children:"ModioCreateModFileParams"})})," containing the directory containing all the files that you want to submit. The plugin will compress this folder into a .zip file and upload it as the active version of the mod."]}),"\n",(0,o.jsxs)(t.p,{children:["In the future, if the mod is updated and requires a new mod file, ",(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"})," can be called again. The most recent mod file uploaded by ",(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"})," will be set as the active version."]}),"\n",(0,o.jsx)(t.admonition,{type:"note",children:(0,o.jsxs)(t.p,{children:["There is no callback for ",(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"}),"; you\u2019ll be notified of the completed upload by your ",(0,o.jsx)(t.a,{href:"mod-subscriptions#installation-management",children:(0,o.jsx)(t.strong,{children:"mod management callback"})}),"."]})}),"\n",(0,o.jsxs)(a.A,{"group-id":"languages",children:[(0,o.jsxs)(s.A,{value:"blueprint",label:"Blueprint",children:[(0,o.jsx)(t.p,{children:"After the callback for submitting a mod has completed, you can get the Mod Id to use for file submission."}),(0,o.jsx)(t.p,{children:(0,o.jsx)(t.img,{alt:"submit_new_mod_file",src:n(8759).A+"",width:"862",height:"483"})})]}),(0,o.jsx)(s.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-cpp",children:'void UModioManagerSubsystem::SubmitNewModFile(FModioModID ModId)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tFModioCreateModFileParams Params;\n \tParams.PathToModRootDirectory = TEXT("C:\\\\path\\\\to\\\\mod-folder");\n \tSubsystem->SubmitNewModFileForMod(ModId, Params);\n }\n}\n'})})})]})]})}function b(e={}){const{wrapper:t}={...(0,i.R)(),...e.components};return t?(0,o.jsx)(t,{...e,children:(0,o.jsx)(m,{...e})}):m(e)}},19365:(e,t,n)=>{n.d(t,{A:()=>s});n(96540);var o=n(18215);const i={tabItem:"tabItem_Ymn6"};var a=n(74848);function s(e){let{children:t,hidden:n,className:s}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,o.A)(i.tabItem,s),hidden:n,children:t})}},11470:(e,t,n)=>{n.d(t,{A:()=>y});var o=n(96540),i=n(18215),a=n(23104),s=n(56347),r=n(205),d=n(57485),l=n(31682),u=n(70679);function c(e){return o.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:n}=e;return(0,o.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:n,attributes:o,default:i}}=e;return{value:t,label:n,attributes:o,default:i}}))}(n);return function(e){const t=(0,l.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function b(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:n}=e;const i=(0,s.W6)(),a=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,d.aZ)(a),(0,o.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(i.location.search);t.set(a,e),i.replace({...i.location,search:t.toString()})}),[a,i])]}function p(e){const{defaultValue:t,queryString:n=!1,groupId:i}=e,a=m(e),[s,d]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!b({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const o=n.find((e=>e.default))??n[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:t,tabValues:a}))),[l,c]=h({queryString:n,groupId:i}),[p,f]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[i,a]=(0,u.Dv)(n);return[i,(0,o.useCallback)((e=>{n&&a.set(e)}),[n,a])]}({groupId:i}),g=(()=>{const e=l??p;return b({value:e,tabValues:a})?e:null})();(0,r.A)((()=>{g&&d(g)}),[g]);return{selectedValue:s,selectValue:(0,o.useCallback)((e=>{if(!b({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);d(e),c(e),f(e)}),[c,f,a]),tabValues:a}}var f=n(92303);const g={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var v=n(74848);function x(e){let{className:t,block:n,selectedValue:o,selectValue:s,tabValues:r}=e;const d=[],{blockElementScrollPositionUntilNextRender:l}=(0,a.a_)(),u=e=>{const t=e.currentTarget,n=d.indexOf(t),i=r[n].value;i!==o&&(l(t),s(i))},c=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const n=d.indexOf(e.currentTarget)+1;t=d[n]??d[0];break}case"ArrowLeft":{const n=d.indexOf(e.currentTarget)-1;t=d[n]??d[d.length-1];break}}t?.focus()};return(0,v.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,i.A)("tabs",{"tabs--block":n},t),children:r.map((e=>{let{value:t,label:n,attributes:a}=e;return(0,v.jsx)("li",{role:"tab",tabIndex:o===t?0:-1,"aria-selected":o===t,ref:e=>d.push(e),onKeyDown:c,onClick:u,...a,className:(0,i.A)("tabs__item",g.tabItem,a?.className,{"tabs__item--active":o===t}),children:n??t},t)}))})}function M(e){let{lazy:t,children:n,selectedValue:i}=e;const a=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===i));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return(0,v.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==i})))})}function w(e){const t=p(e);return(0,v.jsxs)("div",{className:(0,i.A)("tabs-container",g.tabList),children:[(0,v.jsx)(x,{...t,...e}),(0,v.jsx)(M,{...t,...e})]})}function y(e){const t=(0,f.A)();return(0,v.jsx)(w,{...e,children:c(e.children)},String(t))}},89236:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(19365),i=(n(96540),n(74848));function a(e){return(0,i.jsx)(i.Fragment,{children:(0,i.jsx)(o.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(11470),i=(n(96540),n(74848));function a(e){return(0,i.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,i.jsx)(o.A,{...e})})}},17792:(e,t,n)=>{n.d(t,{A:()=>o});const o=n.p+"assets/images/submit_new_mod-d860f68584f2f0b67dd74c6e95467fcb.png"},8759:(e,t,n)=>{n.d(t,{A:()=>o});const o=n.p+"assets/images/submit_new_mod_file-e1cde8dca73f1d2d5985433981e8c5a3.png"},28453:(e,t,n)=>{n.d(t,{R:()=>s,x:()=>r});var o=n(96540);const i={},a=o.createContext(i);function s(e){const t=o.useContext(a);return o.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function r(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:s(e.components),o.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/546be59f.aa883044.js b/Doc/assets/js/546be59f.aa883044.js new file mode 100644 index 00000000..d11310b9 --- /dev/null +++ b/Doc/assets/js/546be59f.aa883044.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[8335],{45912:(e,o,i)=>{i.r(o),i.d(o,{assets:()=>d,contentTitle:()=>r,default:()=>u,frontMatter:()=>s,metadata:()=>a,toc:()=>l});var n=i(74848),t=i(28453);const s={id:"ue-profiling",title:"Profiling",slug:"/unreal/profiling/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/profiling.mdx"},r=void 0,a={id:"ue-profiling",title:"Profiling",description:"The mod.io UE plugin includes some basic profiling and metrics that can be viewed and exposed via Unreal\u2019s stat commands and via Unreal Insights. Profiling via the MODIOENABLEPROFILING and MODIOUNREALPROFILING_SUPPORT macros are enabled by default in the mod.io plugin for non-shipping builds.",source:"@site/public/en-us/profiling.mdx",sourceDirName:".",slug:"/unreal/profiling/",permalink:"/unreal/profiling/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/profiling.mdx",tags:[],version:"current",frontMatter:{id:"ue-profiling",title:"Profiling",slug:"/unreal/profiling/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/profiling.mdx"},sidebar:"sidebar",previous:{title:"Asset Support for Mods",permalink:"/unreal/asset-support-for-mods/"},next:{title:"Mod Creation Tool",permalink:"/unreal/mod-creation-tool/"}},d={},l=[{value:"Showing Captured Data",id:"showing-captured-data",level:2}];function c(e){const o={code:"code",h2:"h2",li:"li",p:"p",ul:"ul",...(0,t.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsxs)(o.p,{children:["The mod.io UE plugin includes some basic profiling and metrics that can be viewed and exposed via Unreal\u2019s ",(0,n.jsx)(o.code,{children:"stat"})," commands and via Unreal Insights. Profiling via the ",(0,n.jsx)(o.code,{children:"MODIO_ENABLE_PROFILING"})," and ",(0,n.jsx)(o.code,{children:"MODIO_UNREAL_PROFILING_SUPPORT"})," macros are enabled by default in the mod.io plugin for non-shipping builds."]}),"\n",(0,n.jsx)(o.h2,{id:"showing-captured-data",children:"Showing Captured Data"}),"\n",(0,n.jsxs)(o.p,{children:["The plugin includes two stat groups that can be shown: ",(0,n.jsx)(o.code,{children:"Modio"})," (accessed via ",(0,n.jsx)(o.code,{children:"stat modio"}),") and ModioScoped (async operations profiled for access via Unreal Insights)."]}),"\n",(0,n.jsxs)(o.ul,{children:["\n",(0,n.jsxs)(o.li,{children:[(0,n.jsx)(o.code,{children:"Modio"})," includes some basic counters, such as the length of various background queues that process API requests and downloads."]}),"\n",(0,n.jsxs)(o.li,{children:[(0,n.jsx)(o.code,{children:"ModioScoped"})," captures performance insights for a majority of the async operations that the plugin performs. This data is not suitable for showing in realtime via a ",(0,n.jsx)(o.code,{children:"stat"})," command, however it is captured when creating stat profiles for use in Unreal Insights."]}),"\n"]})]})}function u(e={}){const{wrapper:o}={...(0,t.R)(),...e.components};return o?(0,n.jsx)(o,{...e,children:(0,n.jsx)(c,{...e})}):c(e)}},28453:(e,o,i)=>{i.d(o,{R:()=>r,x:()=>a});var n=i(96540);const t={},s=n.createContext(t);function r(e){const o=n.useContext(s);return n.useMemo((function(){return"function"==typeof e?e(o):{...o,...e}}),[o,e])}function a(e){let o;return o=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),n.createElement(s.Provider,{value:o},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/546be59f.aab0b80d.js b/Doc/assets/js/546be59f.aab0b80d.js deleted file mode 100644 index 4b175abe..00000000 --- a/Doc/assets/js/546be59f.aab0b80d.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[8335],{45912:(e,i,o)=>{o.r(i),o.d(i,{assets:()=>d,contentTitle:()=>r,default:()=>u,frontMatter:()=>s,metadata:()=>a,toc:()=>l});var n=o(74848),t=o(28453);const s={id:"ue-profiling",title:"Profiling",slug:"/unreal/profiling/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/profiling.mdx"},r=void 0,a={id:"ue-profiling",title:"Profiling",description:"The mod.io UE plugin includes some basic profiling and metrics that can be viewed and exposed via Unreal\u2019s stat commands and via Unreal Insights. Profiling via the MODIOENABLEPROFILING and MODIOUNREALPROFILING_SUPPORT macros are enabled by default in the mod.io plugin for non-shipping builds.",source:"@site/public/en-us/profiling.mdx",sourceDirName:".",slug:"/unreal/profiling/",permalink:"/unreal/profiling/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/profiling.mdx",tags:[],version:"current",frontMatter:{id:"ue-profiling",title:"Profiling",slug:"/unreal/profiling/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/profiling.mdx"},sidebar:"sidebar",previous:{title:"Asset Support for Mods",permalink:"/unreal/asset-support-for-mods/"},next:{title:"Mod Creation Tool",permalink:"/unreal/mod-creation-tool/"}},d={},l=[{value:"Showing Captured Data",id:"showing-captured-data",level:2}];function c(e){const i={code:"code",h2:"h2",li:"li",p:"p",ul:"ul",...(0,t.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsxs)(i.p,{children:["The mod.io UE plugin includes some basic profiling and metrics that can be viewed and exposed via Unreal\u2019s ",(0,n.jsx)(i.code,{children:"stat"})," commands and via Unreal Insights. Profiling via the ",(0,n.jsx)(i.code,{children:"MODIO_ENABLE_PROFILING"})," and ",(0,n.jsx)(i.code,{children:"MODIO_UNREAL_PROFILING_SUPPORT"})," macros are enabled by default in the mod.io plugin for non-shipping builds."]}),"\n",(0,n.jsx)(i.h2,{id:"showing-captured-data",children:"Showing Captured Data"}),"\n",(0,n.jsxs)(i.p,{children:["The plugin includes two stat groups that can be shown: ",(0,n.jsx)(i.code,{children:"Modio"})," (accessed via ",(0,n.jsx)(i.code,{children:"stat modio"}),") and ModioScoped (async operations profiled for access via Unreal Insights)."]}),"\n",(0,n.jsxs)(i.ul,{children:["\n",(0,n.jsxs)(i.li,{children:[(0,n.jsx)(i.code,{children:"Modio"})," includes some basic counters, such as the length of various background queues that process API requests and downloads."]}),"\n",(0,n.jsxs)(i.li,{children:[(0,n.jsx)(i.code,{children:"ModioScoped"})," captures performance insights for a majority of the async operations that the plugin performs. This data is not suitable for showing in realtime via a ",(0,n.jsx)(i.code,{children:"stat"})," command, however it is captured when creating stat profiles for use in Unreal Insights."]}),"\n"]})]})}function u(e={}){const{wrapper:i}={...(0,t.R)(),...e.components};return i?(0,n.jsx)(i,{...e,children:(0,n.jsx)(c,{...e})}):c(e)}},28453:(e,i,o)=>{o.d(i,{R:()=>r,x:()=>a});var n=o(96540);const t={},s=n.createContext(t);function r(e){const i=n.useContext(s);return n.useMemo((function(){return"function"==typeof e?e(i):{...i,...e}}),[i,e])}function a(e){let i;return i=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),n.createElement(s.Provider,{value:i},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/562207a5.a07c0f1f.js b/Doc/assets/js/562207a5.a07c0f1f.js new file mode 100644 index 00000000..4845440d --- /dev/null +++ b/Doc/assets/js/562207a5.a07c0f1f.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[1991],{6046:(e,t,r)=>{r.r(t),r.d(t,{assets:()=>d,contentTitle:()=>i,default:()=>h,frontMatter:()=>o,metadata:()=>l,toc:()=>c});var n=r(74848),s=r(28453),u=r(27064),a=r(89236);const o={id:"ue-mute-user",title:"Muting and Unmuting a User",slug:"/unreal/getting-started/mute-user",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/mute-user.mdx"},i=void 0,l={id:"getting-started/ue-mute-user",title:"Muting and Unmuting a User",description:"Users have the ability to disable updates from other user\u2019s mods. This will prevent mod.io from returning mods authored by the muted user. There are three actions available to take: mute a user, unmute a user, and list muted users",source:"@site/public/en-us/getting-started/mute-user.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/mute-user",permalink:"/unreal/getting-started/mute-user",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/mute-user.mdx",tags:[],version:"current",frontMatter:{id:"ue-mute-user",title:"Muting and Unmuting a User",slug:"/unreal/getting-started/mute-user",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/mute-user.mdx"},sidebar:"sidebar",previous:{title:"Temporary Mod Sets",permalink:"/unreal/getting-started/temporary-mods"},next:{title:"Metrics Play Sessions",permalink:"/unreal/getting-started/metrics-play-sessions"}},d={},c=[{value:"Mute a user",id:"mute-a-user",level:2},{value:"Unmute a user",id:"unmute-a-user",level:2},{value:"List muted users",id:"list-muted-users",level:2}];function m(e){const t={a:"a",admonition:"admonition",code:"code",h2:"h2",img:"img",p:"p",pre:"pre",...(0,s.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsx)(t.p,{children:"Users have the ability to disable updates from other user\u2019s mods. This will prevent mod.io from returning mods authored by the muted user. There are three actions available to take: mute a user, unmute a user, and list muted users"}),"\n",(0,n.jsx)(t.admonition,{type:"note",children:(0,n.jsx)(t.p,{children:"To perform any of these actions, the muting user must be authenticated."})}),"\n",(0,n.jsx)(t.h2,{id:"mute-a-user",children:"Mute a user"}),"\n",(0,n.jsxs)(t.p,{children:["To mute a user, call ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#muteuserasync",children:(0,n.jsx)(t.code,{children:"MuteUserAsync"})})," with the corresponding ",(0,n.jsx)(t.code,{children:"ModioUserID"})," and a callback."]}),"\n",(0,n.jsxs)(u.A,{"group-id":"languages",children:[(0,n.jsx)(a.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"mute_user",src:r(11446).A+"",width:"690",height:"320"})})}),(0,n.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"\nvoid UModioManagerSubsystem::MuteAUser(FModioUserID UserID)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->MuteUserAsync(UserID, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnMuteUserComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnMuteUserComplete(FModioErrorCode ErrorCode)\n{\n if (!ErrorCode)\n {\n \t// User successfully muted \n }\n}\n\n"})})})]}),"\n",(0,n.jsx)(t.h2,{id:"unmute-a-user",children:"Unmute a user"}),"\n",(0,n.jsxs)(t.p,{children:["To perform the inverse operation call ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#unmuteuserasync",children:(0,n.jsx)(t.code,{children:"UnmuteUserAsync"})})," with the corresponding ",(0,n.jsx)(t.code,{children:"ModioUserID"})," and a callback."]}),"\n",(0,n.jsxs)(u.A,{"group-id":"languages",children:[(0,n.jsx)(a.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"unmute_user",src:r(12557).A+"",width:"673",height:"316"})})}),(0,n.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"\nvoid UModioManagerSubsystem::UnmuteAUser(FModioUserID UserID)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->UnmuteUserAsync(UserID, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnUnmuteUserComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnUnmuteUserComplete(FModioErrorCode ErrorCode)\n{\n if (!ErrorCode)\n {\n \t// User successfully unmuted \n }\n}\n"})})})]}),"\n",(0,n.jsx)(t.h2,{id:"list-muted-users",children:"List muted users"}),"\n",(0,n.jsxs)(t.p,{children:[(0,n.jsx)(t.a,{href:"/unreal/refdocs/#getmutedusersasync",children:(0,n.jsx)(t.code,{children:"GetMutedUsersAsync"})})," returns a list of users previously muted by an authenticated user."]}),"\n",(0,n.jsxs)(u.A,{"group-id":"languages",children:[(0,n.jsx)(a.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"get_muted_users",src:r(29404).A+"",width:"621",height:"303"})})}),(0,n.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"\nvoid UModioManagerSubsystem::ListMutedUsers()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->GetMutedUsersAsync(FOnMuteUsersDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnListMutedUsersComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnListMutedUsersComplete(FModioErrorCode ErrorCode, FModioOptionalUserList MutedUsers)\n{\n if (!ErrorCode)\n {\n \t// List of muted users successfully retrieved \n }\n}\n\n"})})})]})]})}function h(e={}){const{wrapper:t}={...(0,s.R)(),...e.components};return t?(0,n.jsx)(t,{...e,children:(0,n.jsx)(m,{...e})}):m(e)}},19365:(e,t,r)=>{r.d(t,{A:()=>a});r(96540);var n=r(18215);const s={tabItem:"tabItem_Ymn6"};var u=r(74848);function a(e){let{children:t,hidden:r,className:a}=e;return(0,u.jsx)("div",{role:"tabpanel",className:(0,n.A)(s.tabItem,a),hidden:r,children:t})}},11470:(e,t,r)=>{r.d(t,{A:()=>j});var n=r(96540),s=r(18215),u=r(23104),a=r(56347),o=r(205),i=r(57485),l=r(31682),d=r(70679);function c(e){return n.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,n.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:r}=e;return(0,n.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:r,attributes:n,default:s}}=e;return{value:t,label:r,attributes:n,default:s}}))}(r);return function(e){const t=(0,l.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,r])}function h(e){let{value:t,tabValues:r}=e;return r.some((e=>e.value===t))}function b(e){let{queryString:t=!1,groupId:r}=e;const s=(0,a.W6)(),u=function(e){let{queryString:t=!1,groupId:r}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!r)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return r??null}({queryString:t,groupId:r});return[(0,i.aZ)(u),(0,n.useCallback)((e=>{if(!u)return;const t=new URLSearchParams(s.location.search);t.set(u,e),s.replace({...s.location,search:t.toString()})}),[u,s])]}function p(e){const{defaultValue:t,queryString:r=!1,groupId:s}=e,u=m(e),[a,i]=(0,n.useState)((()=>function(e){let{defaultValue:t,tabValues:r}=e;if(0===r.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!h({value:t,tabValues:r}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${r.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const n=r.find((e=>e.default))??r[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:t,tabValues:u}))),[l,c]=b({queryString:r,groupId:s}),[p,g]=function(e){let{groupId:t}=e;const r=function(e){return e?`docusaurus.tab.${e}`:null}(t),[s,u]=(0,d.Dv)(r);return[s,(0,n.useCallback)((e=>{r&&u.set(e)}),[r,u])]}({groupId:s}),f=(()=>{const e=l??p;return h({value:e,tabValues:u})?e:null})();(0,o.A)((()=>{f&&i(f)}),[f]);return{selectedValue:a,selectValue:(0,n.useCallback)((e=>{if(!h({value:e,tabValues:u}))throw new Error(`Can't select invalid tab value=${e}`);i(e),c(e),g(e)}),[c,g,u]),tabValues:u}}var g=r(92303);const f={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var v=r(74848);function y(e){let{className:t,block:r,selectedValue:n,selectValue:a,tabValues:o}=e;const i=[],{blockElementScrollPositionUntilNextRender:l}=(0,u.a_)(),d=e=>{const t=e.currentTarget,r=i.indexOf(t),s=o[r].value;s!==n&&(l(t),a(s))},c=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const r=i.indexOf(e.currentTarget)+1;t=i[r]??i[0];break}case"ArrowLeft":{const r=i.indexOf(e.currentTarget)-1;t=i[r]??i[i.length-1];break}}t?.focus()};return(0,v.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,s.A)("tabs",{"tabs--block":r},t),children:o.map((e=>{let{value:t,label:r,attributes:u}=e;return(0,v.jsx)("li",{role:"tab",tabIndex:n===t?0:-1,"aria-selected":n===t,ref:e=>i.push(e),onKeyDown:c,onClick:d,...u,className:(0,s.A)("tabs__item",f.tabItem,u?.className,{"tabs__item--active":n===t}),children:r??t},t)}))})}function x(e){let{lazy:t,children:r,selectedValue:s}=e;const u=(Array.isArray(r)?r:[r]).filter(Boolean);if(t){const e=u.find((e=>e.props.value===s));return e?(0,n.cloneElement)(e,{className:"margin-top--md"}):null}return(0,v.jsx)("div",{className:"margin-top--md",children:u.map(((e,t)=>(0,n.cloneElement)(e,{key:t,hidden:e.props.value!==s})))})}function U(e){const t=p(e);return(0,v.jsxs)("div",{className:(0,s.A)("tabs-container",f.tabList),children:[(0,v.jsx)(y,{...t,...e}),(0,v.jsx)(x,{...t,...e})]})}function j(e){const t=(0,g.A)();return(0,v.jsx)(U,{...e,children:c(e.children)},String(t))}},89236:(e,t,r)=>{r.d(t,{A:()=>u});var n=r(19365),s=(r(96540),r(74848));function u(e){return(0,s.jsx)(s.Fragment,{children:(0,s.jsx)(n.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,r)=>{r.d(t,{A:()=>u});var n=r(11470),s=(r(96540),r(74848));function u(e){return(0,s.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,s.jsx)(n.A,{...e})})}},29404:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/get_muted_users-776736c4f580fac9b441be863741f765.png"},11446:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/mute_user-5cd9224a06f9cdb5d139d5d4b5c457ac.png"},12557:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/unmute_user-3f752651297d3df49a2a2424b0ec0db0.png"},28453:(e,t,r)=>{r.d(t,{R:()=>a,x:()=>o});var n=r(96540);const s={},u=n.createContext(s);function a(e){const t=n.useContext(u);return n.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function o(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:a(e.components),n.createElement(u.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/562207a5.b60eadca.js b/Doc/assets/js/562207a5.b60eadca.js deleted file mode 100644 index 98232243..00000000 --- a/Doc/assets/js/562207a5.b60eadca.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[1991],{6046:(e,t,r)=>{r.r(t),r.d(t,{assets:()=>d,contentTitle:()=>i,default:()=>h,frontMatter:()=>o,metadata:()=>l,toc:()=>c});var n=r(74848),s=r(28453),a=r(27064),u=r(89236);const o={id:"ue-mute-user",title:"Muting and Unmuting a User",slug:"/unreal/getting-started/mute-user",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/mute-user.mdx"},i=void 0,l={id:"getting-started/ue-mute-user",title:"Muting and Unmuting a User",description:"Users have the ability to disable updates from other user\u2019s mods. This will prevent mod.io from returning mods authored by the muted user. There are three actions available to take: mute a user, unmute a user, and list muted users",source:"@site/public/en-us/getting-started/mute-user.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/mute-user",permalink:"/unreal/getting-started/mute-user",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/mute-user.mdx",tags:[],version:"current",frontMatter:{id:"ue-mute-user",title:"Muting and Unmuting a User",slug:"/unreal/getting-started/mute-user",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/mute-user.mdx"},sidebar:"sidebar",previous:{title:"Temporary Mod Sets",permalink:"/unreal/getting-started/temporary-mods"},next:{title:"Metrics Play Sessions",permalink:"/unreal/getting-started/metrics-play-sessions"}},d={},c=[{value:"Mute a user",id:"mute-a-user",level:2},{value:"Unmute a user",id:"unmute-a-user",level:2},{value:"List muted users",id:"list-muted-users",level:2}];function m(e){const t={a:"a",admonition:"admonition",code:"code",h2:"h2",img:"img",p:"p",pre:"pre",...(0,s.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsx)(t.p,{children:"Users have the ability to disable updates from other user\u2019s mods. This will prevent mod.io from returning mods authored by the muted user. There are three actions available to take: mute a user, unmute a user, and list muted users"}),"\n",(0,n.jsx)(t.admonition,{type:"note",children:(0,n.jsx)(t.p,{children:"To perform any of these actions, the muting user must be authenticated."})}),"\n",(0,n.jsx)(t.h2,{id:"mute-a-user",children:"Mute a user"}),"\n",(0,n.jsxs)(t.p,{children:["To mute a user, call ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_MuteUserAsync",children:(0,n.jsx)(t.code,{children:"MuteUserAsync"})})," with the corresponding ",(0,n.jsx)(t.code,{children:"ModioUserID"})," and a callback."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(u.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"mute_user",src:r(25531).A+"",width:"690",height:"320"})})}),(0,n.jsx)(u.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"\nvoid UModioManagerSubsystem::MuteAUser(FModioUserID UserID)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->MuteUserAsync(UserID, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnMuteUserComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnMuteUserComplete(FModioErrorCode ErrorCode)\n{\n if (!ErrorCode)\n {\n \t// User successfully muted \n }\n}\n\n"})})})]}),"\n",(0,n.jsx)(t.h2,{id:"unmute-a-user",children:"Unmute a user"}),"\n",(0,n.jsxs)(t.p,{children:["To perform the inverse operation call ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_UnmuteUserAsync",children:(0,n.jsx)(t.code,{children:"UnmuteUserAsync"})})," with the corresponding ",(0,n.jsx)(t.code,{children:"ModioUserID"})," and a callback."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(u.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"unmute_user",src:r(31492).A+"",width:"673",height:"316"})})}),(0,n.jsx)(u.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"\nvoid UModioManagerSubsystem::UnmuteAUser(FModioUserID UserID)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->UnmuteUserAsync(UserID, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnUnmuteUserComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnUnmuteUserComplete(FModioErrorCode ErrorCode)\n{\n if (!ErrorCode)\n {\n \t// User successfully unmuted \n }\n}\n"})})})]}),"\n",(0,n.jsx)(t.h2,{id:"list-muted-users",children:"List muted users"}),"\n",(0,n.jsxs)(t.p,{children:[(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_GetMutedUsersAsync",children:(0,n.jsx)(t.code,{children:"GetMutedUsersAsync"})})," returns a list of users previously muted by an authenticated user."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(u.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"get_muted_users",src:r(35305).A+"",width:"621",height:"303"})})}),(0,n.jsx)(u.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:"\nvoid UModioManagerSubsystem::ListMutedUsers()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->GetMutedUsersAsync(FOnMuteUsersDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnListMutedUsersComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnListMutedUsersComplete(FModioErrorCode ErrorCode, FModioOptionalUserList MutedUsers)\n{\n if (!ErrorCode)\n {\n \t// List of muted users successfully retrieved \n }\n}\n\n"})})})]})]})}function h(e={}){const{wrapper:t}={...(0,s.R)(),...e.components};return t?(0,n.jsx)(t,{...e,children:(0,n.jsx)(m,{...e})}):m(e)}},19365:(e,t,r)=>{r.d(t,{A:()=>u});r(96540);var n=r(18215);const s={tabItem:"tabItem_Ymn6"};var a=r(74848);function u(e){let{children:t,hidden:r,className:u}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,n.A)(s.tabItem,u),hidden:r,children:t})}},11470:(e,t,r)=>{r.d(t,{A:()=>M});var n=r(96540),s=r(18215),a=r(23104),u=r(56347),o=r(205),i=r(57485),l=r(31682),d=r(70679);function c(e){return n.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,n.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:r}=e;return(0,n.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:r,attributes:n,default:s}}=e;return{value:t,label:r,attributes:n,default:s}}))}(r);return function(e){const t=(0,l.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,r])}function h(e){let{value:t,tabValues:r}=e;return r.some((e=>e.value===t))}function b(e){let{queryString:t=!1,groupId:r}=e;const s=(0,u.W6)(),a=function(e){let{queryString:t=!1,groupId:r}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!r)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return r??null}({queryString:t,groupId:r});return[(0,i.aZ)(a),(0,n.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(s.location.search);t.set(a,e),s.replace({...s.location,search:t.toString()})}),[a,s])]}function p(e){const{defaultValue:t,queryString:r=!1,groupId:s}=e,a=m(e),[u,i]=(0,n.useState)((()=>function(e){let{defaultValue:t,tabValues:r}=e;if(0===r.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!h({value:t,tabValues:r}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${r.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const n=r.find((e=>e.default))??r[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:t,tabValues:a}))),[l,c]=b({queryString:r,groupId:s}),[p,g]=function(e){let{groupId:t}=e;const r=function(e){return e?`docusaurus.tab.${e}`:null}(t),[s,a]=(0,d.Dv)(r);return[s,(0,n.useCallback)((e=>{r&&a.set(e)}),[r,a])]}({groupId:s}),f=(()=>{const e=l??p;return h({value:e,tabValues:a})?e:null})();(0,o.A)((()=>{f&&i(f)}),[f]);return{selectedValue:u,selectValue:(0,n.useCallback)((e=>{if(!h({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);i(e),c(e),g(e)}),[c,g,a]),tabValues:a}}var g=r(92303);const f={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var v=r(74848);function y(e){let{className:t,block:r,selectedValue:n,selectValue:u,tabValues:o}=e;const i=[],{blockElementScrollPositionUntilNextRender:l}=(0,a.a_)(),d=e=>{const t=e.currentTarget,r=i.indexOf(t),s=o[r].value;s!==n&&(l(t),u(s))},c=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const r=i.indexOf(e.currentTarget)+1;t=i[r]??i[0];break}case"ArrowLeft":{const r=i.indexOf(e.currentTarget)-1;t=i[r]??i[i.length-1];break}}t?.focus()};return(0,v.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,s.A)("tabs",{"tabs--block":r},t),children:o.map((e=>{let{value:t,label:r,attributes:a}=e;return(0,v.jsx)("li",{role:"tab",tabIndex:n===t?0:-1,"aria-selected":n===t,ref:e=>i.push(e),onKeyDown:c,onClick:d,...a,className:(0,s.A)("tabs__item",f.tabItem,a?.className,{"tabs__item--active":n===t}),children:r??t},t)}))})}function x(e){let{lazy:t,children:r,selectedValue:s}=e;const a=(Array.isArray(r)?r:[r]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===s));return e?(0,n.cloneElement)(e,{className:"margin-top--md"}):null}return(0,v.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,n.cloneElement)(e,{key:t,hidden:e.props.value!==s})))})}function U(e){const t=p(e);return(0,v.jsxs)("div",{className:(0,s.A)("tabs-container",f.tabList),children:[(0,v.jsx)(y,{...t,...e}),(0,v.jsx)(x,{...t,...e})]})}function M(e){const t=(0,g.A)();return(0,v.jsx)(U,{...e,children:c(e.children)},String(t))}},89236:(e,t,r)=>{r.d(t,{A:()=>a});var n=r(19365),s=(r(96540),r(74848));function a(e){return(0,s.jsx)(s.Fragment,{children:(0,s.jsx)(n.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,r)=>{r.d(t,{A:()=>a});var n=r(11470),s=(r(96540),r(74848));function a(e){return(0,s.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,s.jsx)(n.A,{...e})})}},35305:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/get_muted_users-776736c4f580fac9b441be863741f765.png"},25531:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/mute_user-5cd9224a06f9cdb5d139d5d4b5c457ac.png"},31492:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/unmute_user-3f752651297d3df49a2a2424b0ec0db0.png"},28453:(e,t,r)=>{r.d(t,{R:()=>u,x:()=>o});var n=r(96540);const s={},a=n.createContext(s);function u(e){const t=n.useContext(a);return n.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function o(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:u(e.components),n.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/65ba71d8.3c23f95a.js b/Doc/assets/js/65ba71d8.3c23f95a.js new file mode 100644 index 00000000..a62c4a7a --- /dev/null +++ b/Doc/assets/js/65ba71d8.3c23f95a.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[6783],{46770:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>l,contentTitle:()=>s,default:()=>u,frontMatter:()=>r,metadata:()=>a,toc:()=>d});var i=t(74848),o=t(28453);t(27064),t(89236);const r={id:"ue-plugin-structure",title:"Plugin Structure & Concepts",slug:"/unreal/getting-started/plugin-structure/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/plugin-structure.mdx"},s=void 0,a={id:"getting-started/ue-plugin-structure",title:"Plugin Structure & Concepts",description:"ModioSubsystem",source:"@site/public/en-us/getting-started/plugin-structure.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/plugin-structure/",permalink:"/unreal/getting-started/plugin-structure/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/plugin-structure.mdx",tags:[],version:"current",frontMatter:{id:"ue-plugin-structure",title:"Plugin Structure & Concepts",slug:"/unreal/getting-started/plugin-structure/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/plugin-structure.mdx"},sidebar:"sidebar",previous:{title:"Getting Started",permalink:"/unreal/getting-started/"},next:{title:"Initialization & Teardown",permalink:"/unreal/getting-started/initialization"}},l={},d=[{value:"ModioSubsystem",id:"modiosubsystem",level:2},{value:"Value objects",id:"value-objects",level:2},{value:"UTF-8 guarantees",id:"utf-8-guarantees",level:2},{value:"Thread-safety guarantees",id:"thread-safety-guarantees",level:2},{value:"Non-blocking, asynchronous interface",id:"non-blocking-asynchronous-interface",level:2},{value:"Callback conventions",id:"callback-conventions",level:3},{value:"Maintaining the plugin event loop",id:"maintaining-the-plugin-event-loop",level:3},{value:"Error handling",id:"error-handling",level:2},{value:"User sessions",id:"user-sessions",level:2},{value:"Session IDs",id:"session-ids",level:3},{value:"Mod data directory",id:"mod-data-directory",level:2},{value:"Globally for a system account",id:"globally-for-a-system-account",level:3},{value:"Per-game for a system account",id:"per-game-for-a-system-account",level:3}];function c(e){const n={a:"a",admonition:"admonition",code:"code",h2:"h2",h3:"h3",li:"li",mdxAdmonitionTitle:"mdxAdmonitionTitle",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,o.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"modiosubsystem",children:(0,i.jsx)(n.code,{children:"ModioSubsystem"})}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine Plugin consumes our ",(0,i.jsx)(n.a,{href:"/cppsdk/",children:"Native SDK"})," internally, and its public API is similar. The Native SDK exposes a number of free functions within the ",(0,i.jsx)(n.code,{children:"Modio"})," namespace, and this plugin wraps those functions in a ",(0,i.jsx)(n.a,{href:"https://dev.epicgames.com/documentation/en-us/unreal-engine/programming-subsystems-in-unreal-engine",children:"Subsystem"}),"."]}),"\n",(0,i.jsxs)(n.p,{children:["The lifetime of the ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#modiosubsystem",children:(0,i.jsx)(n.code,{children:"ModioSubsystem"})})," is automatically managed by the engine and provides a convenient way to access the plugin's functionality. This is broadly grouped into the following categories:"]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Plugin management (initialization, teardown, event loop)"}),"\n",(0,i.jsx)(n.li,{children:"User authentication"}),"\n",(0,i.jsx)(n.li,{children:"Mod browsing and querying"}),"\n",(0,i.jsx)(n.li,{children:"Mod management (subscription, unsubscription, and installation)"}),"\n"]}),"\n",(0,i.jsx)(n.h2,{id:"value-objects",children:"Value objects"}),"\n",(0,i.jsxs)(n.p,{children:["All data returned by the plugin is presented using a small set of ",(0,i.jsx)(n.code,{children:"USTRUCTs"}),", containing information such as:"]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Details for mods that are available for installation"}),"\n",(0,i.jsx)(n.li,{children:"Progress information about mods being installed"}),"\n",(0,i.jsx)(n.li,{children:"Details and load paths for installed mods"}),"\n"]}),"\n",(0,i.jsxs)(n.p,{children:['As structs, these are value types, so if you want to hold onto them once you\u2019ve shut down the plugin you can do so. They do not expose any methods that "talk back" to the ',(0,i.jsx)(n.code,{children:"ModioSubsystem"}),", so their methods are safe to call. This allows you, for instance, to initialize the plugin, query the installed mods, and retain that list even after shutting down the plugin and its event loop."]}),"\n",(0,i.jsx)(n.h2,{id:"utf-8-guarantees",children:"UTF-8 guarantees"}),"\n",(0,i.jsx)(n.p,{children:"The plugin uses UTF8 for all strings, as does the mod.io REST API."}),"\n",(0,i.jsx)(n.h2,{id:"thread-safety-guarantees",children:"Thread-safety guarantees"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io plugin is thread-safe with the exception of ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#run-pending-handlers",children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})}),". If you are calling ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"})," manually, you must always call it on the same thread."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["The plugin event loop, any internal event handlers, and all callbacks you provide to the mod.io plugin will be run on the thread invoking ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"}),"."]})}),"\n",(0,i.jsx)(n.h2,{id:"non-blocking-asynchronous-interface",children:"Non-blocking, asynchronous interface"}),"\n",(0,i.jsxs)(n.p,{children:["The plugin communicates with the mod.io servers, the filesystem on the device you\u2019re using, and platform-provided services for authentication. These may not return results immediately, so many functions provided by the ",(0,i.jsx)(n.code,{children:"ModioSubsystem"})," are non-blocking and asynchronous."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["All async methods in the public API end with the suffix ",(0,i.jsx)(n.code,{children:"Async"}),"."]})}),"\n",(0,i.jsx)(n.h3,{id:"callback-conventions",children:"Callback conventions"}),"\n",(0,i.jsxs)(n.p,{children:["All of these asynchronous methods take a delegate as an argument, which will be invoked ",(0,i.jsx)(n.strong,{children:"exactly once"})," with the results of the requested operation. Every async callback takes a ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#modioerrorcode",children:(0,i.jsx)(n.code,{children:"ModioErrorCode"})})," as its first parameter, with any optional results wrapped in ",(0,i.jsx)(n.code,{children:"TOptional"})," so that you can easily check if a result is valid."]}),"\n",(0,i.jsxs)(n.p,{children:["Async functions should not be considered complete until their callback is invoked. For example, although ",(0,i.jsx)(n.code,{children:"InitializeAsync"})," returns immediately, you must wait for its callback (with a successful ",(0,i.jsx)(n.code,{children:"ModioErrorCode"}),") before you are able to call any other functions from the ",(0,i.jsx)(n.code,{children:"ModioSubsystem"}),"."]}),"\n",(0,i.jsxs)(n.p,{children:["Any return values provided to your callback are passed-by-value; the plugin doesn\u2019t expect you to have to call ",(0,i.jsx)(n.code,{children:"release"})," or otherwise free up resources given to you."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["Even if the plugin is shut down while asynchronous operations are in-flight, your delegates will still be invoked ",(0,i.jsx)(n.strong,{children:"exactly once"}),". In this instance, the ",(0,i.jsx)(n.code,{children:"ModioErrorCode"})," the delegate receives will indicate a cancellation state, and you should check for this as part of your error handling in your delegates."]})}),"\n",(0,i.jsx)(n.h3,{id:"maintaining-the-plugin-event-loop",children:"Maintaining the plugin event loop"}),"\n",(0,i.jsxs)(n.p,{children:["In order to provide a non-blocking implementation, the plugin operates an internal event loop. This event loop will only ever run on the thread which calls ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#run-pending-handlers",children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})}),". ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"})," is the function we provide to tick the internal event loop and process any pending work the plugin needs to perform."]}),"\n",(0,i.jsxs)(n.p,{children:["You should either enable the ",(0,i.jsx)(n.a,{href:"/unreal/installation-and-setup/#plugin-configuration",children:"configuration setting"})," to allow the plugin to manage a background thread which automatically calls ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"}),", or invoke it on tick in the game thread yourself."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["This means that if you stop calling ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"}),", any pending asynchronous API methods you have called will not complete and their associated callbacks will not be invoked, nor will the internal data allocated for those operations be freed."]})}),"\n",(0,i.jsx)(n.h2,{id:"error-handling",children:"Error handling"}),"\n",(0,i.jsxs)(n.p,{children:["Many of the plugin's functions utilize the ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#modioerrorcode",children:(0,i.jsx)(n.code,{children:"ModioErrorCode"})})," type. For example, all ",(0,i.jsx)(n.a,{href:"#non-blocking-asynchronous-interface",children:"async"})," functions take a ",(0,i.jsx)(n.code,{children:"ModioErrorCode"})," as the first parameter for the function's delegate. This is essentially an opaque wrapper around a numeric error code with a category and an associated string message."]}),"\n",(0,i.jsx)(n.p,{children:"The plugin doesn\u2019t attempt to predict what your error-handling logic or requirements are. Instead, we return the error code to you so you can decide what to do. For instance, if you call a function and receive an error code matching a network error condition, do you want to close down the plugin? Retry again according to custom back-off logic? That decision is left to the consuming application."}),"\n",(0,i.jsxs)(n.p,{children:["For more details on the error codes and how to inspect their values, please see our ",(0,i.jsx)(n.a,{href:"/unreal/getting-started/error-handling",children:"Error Handling quick-start guide"}),"."]}),"\n",(0,i.jsx)(n.h2,{id:"user-sessions",children:"User sessions"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io plugin runs on a per-platform-user basis. If you are using the plugin on a platform that requires user switching support, you must call ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#shutdownasync",children:(0,i.jsx)(n.code,{children:"ShutdownAsync"})})," and re-initialize the plugin by calling ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#initializeasync",children:(0,i.jsx)(n.code,{children:"InitializeAsync"})})," with a different ",(0,i.jsx)(n.a,{href:"#session-ids",children:"Session ID"}),"."]}),"\n",(0,i.jsx)(n.p,{children:"Generating a stable platform-specific Session ID for each user will ensure that an incoming user who has already authenticated with mod.io on the current device and game won\u2019t need to re-authenticate unless their authentication token has expired."}),"\n",(0,i.jsx)(n.h3,{id:"session-ids",children:"Session IDs"}),"\n",(0,i.jsx)(n.p,{children:"Session IDs are chosen by the developer. They can be whatever you like as long as they are deterministic and stable. Session IDs create a scope or a local profile for the current user to live in so that a single system can support multiple authenticated users side-by-side without requiring de-authentication of the previous user. Internally, the Session ID is used to create a folder containing the authentication information and cached profile of the authenticated user."}),"\n",(0,i.jsx)(n.p,{children:"On console platforms, we suggest that the Session ID should be a string representation of the platform-provided user ID. This gives the best experience when it comes to things like user switching. For example, a game on Xbox using a sanitized string representation of the Xbox Live ID as the Session ID would have a folder structure in persistent storage like the following:"}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{children:"/mod.io///\n/mod.io///\n"})}),"\n",(0,i.jsx)(n.p,{children:"With this example, you can detect the user associated with the current controller on game start and initialize the mod.io plugin with the Session ID set to the stable string representation of that user's Xbox Live ID. As a result, the user's authentication status will be maintained from the previous session on that device."}),"\n",(0,i.jsx)(n.p,{children:"In the case of a Windows title with user-provided profile names, the same folder structure would be more like the following:"}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{children:"%USERDATA%/mod.io//ProfileName1/\n%USERDATA%/mod.io//ProfileName2/\n%USERDATA%/mod.io//ProfileName3/\n"})}),"\n",(0,i.jsx)(n.p,{children:"This allows multiple players, for example siblings, to each have their own session living in the same Windows account."}),"\n",(0,i.jsx)(n.h2,{id:"mod-data-directory",children:"Mod data directory"}),"\n",(0,i.jsx)(n.p,{children:"The plugin stores mods in a game-specific directory in the following directory by default:"}),"\n",(0,i.jsxs)(n.table,{children:[(0,i.jsx)(n.thead,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.th,{children:"Windows"}),(0,i.jsx)(n.th,{children:"Linux"}),(0,i.jsx)(n.th,{children:"macOS"}),(0,i.jsx)(n.th,{children:"iOS"})]})}),(0,i.jsx)(n.tbody,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:(0,i.jsx)(n.code,{children:"${FolderID_Public}/mod.io"})}),(0,i.jsx)(n.td,{children:(0,i.jsx)(n.code,{children:"${USER_HOME}/mod.io"})}),(0,i.jsx)(n.td,{children:(0,i.jsx)(n.code,{children:"${USER_HOME}/Library/Application Support/mod.io"})}),(0,i.jsx)(n.td,{children:(0,i.jsx)(n.code,{children:"${APP-DIRECTORY}/Documents/mod.io"})})]})})]}),"\n",(0,i.jsx)(n.p,{children:"However, this value can be overridden in one of two ways:"}),"\n",(0,i.jsx)(n.h3,{id:"globally-for-a-system-account",children:"Globally for a system account"}),"\n",(0,i.jsxs)(n.p,{children:["On first run of a game using the plugin, ",(0,i.jsx)(n.code,{children:"${FOLDERID_LocalAppData}/mod.io/globalsettings.json"})," will be created."]}),"\n",(0,i.jsxs)(n.p,{children:["This JSON object contains a ",(0,i.jsx)(n.code,{children:"RootLocalStoragePath"})," element - changing the string here to a valid path on disk will globally redirect the mod installation directory for ",(0,i.jsx)(n.strong,{children:"ALL"})," games using the mod.io Unreal Engine Plugin or Native SDK for the current system account."]}),"\n",(0,i.jsx)(n.admonition,{type:"warning",children:(0,i.jsx)(n.p,{children:"Changing this value while the SDK is initialized is not supported and behaviour is undefined."})}),"\n",(0,i.jsx)(n.h3,{id:"per-game-for-a-system-account",children:"Per-game for a system account"}),"\n",(0,i.jsxs)(n.p,{children:["Per-game settings are stored in ",(0,i.jsx)(n.code,{children:"${FOLDERID_LocalAppData}/mod.io/${GameId}/${mod.io user string}/user.json"}),"."]}),"\n",(0,i.jsxs)(n.p,{children:["Adding a ",(0,i.jsx)(n.code,{children:"RootLocalStoragePath"})," element to this file will redirect the mod installation directory for this specific game only for the current system account. Removing this value will cause the game to revert back to the global value in ",(0,i.jsx)(n.code,{children:"globalsettings.json"}),"."]}),"\n",(0,i.jsxs)(n.admonition,{type:"note",children:[(0,i.jsx)(n.mdxAdmonitionTitle,{}),(0,i.jsx)(n.p,{children:"In Linux and macOS, mods and cached data binds to a single user. Other client have their own instance in their home directory."})]})]})}function u(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(c,{...e})}):c(e)}},19365:(e,n,t)=>{t.d(n,{A:()=>s});t(96540);var i=t(18215);const o={tabItem:"tabItem_Ymn6"};var r=t(74848);function s(e){let{children:n,hidden:t,className:s}=e;return(0,r.jsx)("div",{role:"tabpanel",className:(0,i.A)(o.tabItem,s),hidden:t,children:n})}},11470:(e,n,t)=>{t.d(n,{A:()=>w});var i=t(96540),o=t(18215),r=t(23104),s=t(56347),a=t(205),l=t(57485),d=t(31682),c=t(70679);function u(e){return i.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,i.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:n,children:t}=e;return(0,i.useMemo)((()=>{const e=n??function(e){return u(e).map((e=>{let{props:{value:n,label:t,attributes:i,default:o}}=e;return{value:n,label:t,attributes:i,default:o}}))}(t);return function(e){const n=(0,d.X)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function p(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function g(e){let{queryString:n=!1,groupId:t}=e;const o=(0,s.W6)(),r=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,l.aZ)(r),(0,i.useCallback)((e=>{if(!r)return;const n=new URLSearchParams(o.location.search);n.set(r,e),o.replace({...o.location,search:n.toString()})}),[r,o])]}function m(e){const{defaultValue:n,queryString:t=!1,groupId:o}=e,r=h(e),[s,l]=(0,i.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!p({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const i=t.find((e=>e.default))??t[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:n,tabValues:r}))),[d,u]=g({queryString:t,groupId:o}),[m,f]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[o,r]=(0,c.Dv)(t);return[o,(0,i.useCallback)((e=>{t&&r.set(e)}),[t,r])]}({groupId:o}),x=(()=>{const e=d??m;return p({value:e,tabValues:r})?e:null})();(0,a.A)((()=>{x&&l(x)}),[x]);return{selectedValue:s,selectValue:(0,i.useCallback)((e=>{if(!p({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);l(e),u(e),f(e)}),[u,f,r]),tabValues:r}}var f=t(92303);const x={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var b=t(74848);function y(e){let{className:n,block:t,selectedValue:i,selectValue:s,tabValues:a}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,r.a_)(),c=e=>{const n=e.currentTarget,t=l.indexOf(n),o=a[t].value;o!==i&&(d(n),s(o))},u=e=>{let n=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const t=l.indexOf(e.currentTarget)+1;n=l[t]??l[0];break}case"ArrowLeft":{const t=l.indexOf(e.currentTarget)-1;n=l[t]??l[l.length-1];break}}n?.focus()};return(0,b.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.A)("tabs",{"tabs--block":t},n),children:a.map((e=>{let{value:n,label:t,attributes:r}=e;return(0,b.jsx)("li",{role:"tab",tabIndex:i===n?0:-1,"aria-selected":i===n,ref:e=>l.push(e),onKeyDown:u,onClick:c,...r,className:(0,o.A)("tabs__item",x.tabItem,r?.className,{"tabs__item--active":i===n}),children:t??n},n)}))})}function v(e){let{lazy:n,children:t,selectedValue:o}=e;const r=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=r.find((e=>e.props.value===o));return e?(0,i.cloneElement)(e,{className:"margin-top--md"}):null}return(0,b.jsx)("div",{className:"margin-top--md",children:r.map(((e,n)=>(0,i.cloneElement)(e,{key:n,hidden:e.props.value!==o})))})}function j(e){const n=m(e);return(0,b.jsxs)("div",{className:(0,o.A)("tabs-container",x.tabList),children:[(0,b.jsx)(y,{...n,...e}),(0,b.jsx)(v,{...n,...e})]})}function w(e){const n=(0,f.A)();return(0,b.jsx)(j,{...e,children:u(e.children)},String(n))}},89236:(e,n,t)=>{t.d(n,{A:()=>r});var i=t(19365),o=(t(96540),t(74848));function r(e){return(0,o.jsx)(o.Fragment,{children:(0,o.jsx)(i.A,{className:"tw-rounded-md",...e})})}},27064:(e,n,t)=>{t.d(n,{A:()=>r});var i=t(11470),o=(t(96540),t(74848));function r(e){return(0,o.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,o.jsx)(i.A,{...e})})}},28453:(e,n,t)=>{t.d(n,{R:()=>s,x:()=>a});var i=t(96540);const o={},r=i.createContext(o);function s(e){const n=i.useContext(r);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:s(e.components),i.createElement(r.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/65ba71d8.9f5de872.js b/Doc/assets/js/65ba71d8.9f5de872.js deleted file mode 100644 index 4bcb8686..00000000 --- a/Doc/assets/js/65ba71d8.9f5de872.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[6783],{46770:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>l,contentTitle:()=>s,default:()=>u,frontMatter:()=>r,metadata:()=>a,toc:()=>d});var i=t(74848),o=t(28453);t(27064),t(89236);const r={id:"ue-plugin-structure",title:"Plugin Structure & Concepts",slug:"/unreal/getting-started/plugin-structure/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/plugin-structure.mdx"},s=void 0,a={id:"getting-started/ue-plugin-structure",title:"Plugin Structure & Concepts",description:"ModioSubsystem",source:"@site/public/en-us/getting-started/plugin-structure.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/plugin-structure/",permalink:"/unreal/getting-started/plugin-structure/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/plugin-structure.mdx",tags:[],version:"current",frontMatter:{id:"ue-plugin-structure",title:"Plugin Structure & Concepts",slug:"/unreal/getting-started/plugin-structure/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/plugin-structure.mdx"},sidebar:"sidebar",previous:{title:"Getting Started",permalink:"/unreal/getting-started/"},next:{title:"Initialization & Teardown",permalink:"/unreal/getting-started/initialization"}},l={},d=[{value:"ModioSubsystem",id:"modiosubsystem",level:2},{value:"Value objects",id:"value-objects",level:2},{value:"UTF-8 guarantees",id:"utf-8-guarantees",level:2},{value:"Thread-safety guarantees",id:"thread-safety-guarantees",level:2},{value:"Non-blocking, asynchronous interface",id:"non-blocking-asynchronous-interface",level:2},{value:"Callback conventions",id:"callback-conventions",level:3},{value:"Maintaining the plugin event loop",id:"maintaining-the-plugin-event-loop",level:3},{value:"Error handling",id:"error-handling",level:2},{value:"User sessions",id:"user-sessions",level:2},{value:"Session IDs",id:"session-ids",level:3},{value:"Mod data directory",id:"mod-data-directory",level:2},{value:"Globally for a system account",id:"globally-for-a-system-account",level:3},{value:"Per-game for a system account",id:"per-game-for-a-system-account",level:3}];function c(e){const n={a:"a",admonition:"admonition",code:"code",h2:"h2",h3:"h3",li:"li",mdxAdmonitionTitle:"mdxAdmonitionTitle",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,o.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"modiosubsystem",children:(0,i.jsx)(n.code,{children:"ModioSubsystem"})}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine Plugin consumes our ",(0,i.jsx)(n.a,{href:"/cppsdk/",children:"Native SDK"})," internally, and its public API is similar. The Native SDK exposes a number of free functions within the ",(0,i.jsx)(n.code,{children:"Modio"})," namespace, and this plugin wraps those functions in a ",(0,i.jsx)(n.a,{href:"https://dev.epicgames.com/documentation/en-us/unreal-engine/programming-subsystems-in-unreal-engine",children:"Subsystem"}),"."]}),"\n",(0,i.jsxs)(n.p,{children:["The lifetime of the ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modiosubsystem",children:(0,i.jsx)(n.code,{children:"ModioSubsystem"})})," is automatically managed by the engine and provides a convenient way to access the plugin's functionality. This is broadly grouped into the following categories:"]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Plugin management (initialization, teardown, event loop)"}),"\n",(0,i.jsx)(n.li,{children:"User authentication"}),"\n",(0,i.jsx)(n.li,{children:"Mod browsing and querying"}),"\n",(0,i.jsx)(n.li,{children:"Mod management (subscription, unsubscription, and installation)"}),"\n"]}),"\n",(0,i.jsx)(n.h2,{id:"value-objects",children:"Value objects"}),"\n",(0,i.jsxs)(n.p,{children:["All data returned by the plugin is presented using a small set of ",(0,i.jsx)(n.code,{children:"USTRUCTs"}),", containing information such as:"]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Details for mods that are available for installation"}),"\n",(0,i.jsx)(n.li,{children:"Progress information about mods being installed"}),"\n",(0,i.jsx)(n.li,{children:"Details and load paths for installed mods"}),"\n"]}),"\n",(0,i.jsxs)(n.p,{children:['As structs, these are value types, so if you want to hold onto them once you\u2019ve shut down the plugin you can do so. They do not expose any methods that "talk back" to the ',(0,i.jsx)(n.code,{children:"ModioSubsystem"}),", so their methods are safe to call. This allows you, for instance, to initialize the plugin, query the installed mods, and retain that list even after shutting down the plugin and its event loop."]}),"\n",(0,i.jsx)(n.h2,{id:"utf-8-guarantees",children:"UTF-8 guarantees"}),"\n",(0,i.jsx)(n.p,{children:"The plugin uses UTF8 for all strings, as does the mod.io REST API."}),"\n",(0,i.jsx)(n.h2,{id:"thread-safety-guarantees",children:"Thread-safety guarantees"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io plugin is thread-safe with the exception of ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#RunPendingHandlers",children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})}),". If you are calling ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"})," manually, you must always call it on the same thread."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["The plugin event loop, any internal event handlers, and all callbacks you provide to the mod.io plugin will be run on the thread invoking ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"}),"."]})}),"\n",(0,i.jsx)(n.h2,{id:"non-blocking-asynchronous-interface",children:"Non-blocking, asynchronous interface"}),"\n",(0,i.jsxs)(n.p,{children:["The plugin communicates with the mod.io servers, the filesystem on the device you\u2019re using, and platform-provided services for authentication. These may not return results immediately, so many functions provided by the ",(0,i.jsx)(n.code,{children:"ModioSubsystem"})," are non-blocking and asynchronous."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["All async methods in the public API end with the suffix ",(0,i.jsx)(n.code,{children:"Async"}),"."]})}),"\n",(0,i.jsx)(n.h3,{id:"callback-conventions",children:"Callback conventions"}),"\n",(0,i.jsxs)(n.p,{children:["All of these asynchronous methods take a delegate as an argument, which will be invoked ",(0,i.jsx)(n.strong,{children:"exactly once"})," with the results of the requested operation. Every async callback takes a ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modioerrorcode",children:(0,i.jsx)(n.code,{children:"ModioErrorCode"})})," as its first parameter, with any optional results wrapped in ",(0,i.jsx)(n.code,{children:"TOptional"})," so that you can easily check if a result is valid."]}),"\n",(0,i.jsxs)(n.p,{children:["Async functions should not be considered complete until their callback is invoked. For example, although ",(0,i.jsx)(n.code,{children:"InitializeAsync"})," returns immediately, you must wait for its callback (with a successful ",(0,i.jsx)(n.code,{children:"ModioErrorCode"}),") before you are able to call any other functions from the ",(0,i.jsx)(n.code,{children:"ModioSubsystem"}),"."]}),"\n",(0,i.jsxs)(n.p,{children:["Any return values provided to your callback are passed-by-value; the plugin doesn\u2019t expect you to have to call ",(0,i.jsx)(n.code,{children:"release"})," or otherwise free up resources given to you."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["Even if the plugin is shut down while asynchronous operations are in-flight, your delegates will still be invoked ",(0,i.jsx)(n.strong,{children:"exactly once"}),". In this instance, the ",(0,i.jsx)(n.code,{children:"ModioErrorCode"})," the delegate receives will indicate a cancellation state, and you should check for this as part of your error handling in your delegates."]})}),"\n",(0,i.jsx)(n.h3,{id:"maintaining-the-plugin-event-loop",children:"Maintaining the plugin event loop"}),"\n",(0,i.jsxs)(n.p,{children:["In order to provide a non-blocking implementation, the plugin operates an internal event loop. This event loop will only ever run on the thread which calls ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#RunPendingHandlers",children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})}),". ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"})," is the function we provide to tick the internal event loop and process any pending work the plugin needs to perform."]}),"\n",(0,i.jsxs)(n.p,{children:["You should either enable the ",(0,i.jsx)(n.a,{href:"/unreal/installation-and-setup/#plugin-configuration",children:"configuration setting"})," to allow the plugin to manage a background thread which automatically calls ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"}),", or invoke it on tick in the game thread yourself."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["This means that if you stop calling ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"}),", any pending asynchronous API methods you have called will not complete and their associated callbacks will not be invoked, nor will the internal data allocated for those operations be freed."]})}),"\n",(0,i.jsx)(n.h2,{id:"error-handling",children:"Error handling"}),"\n",(0,i.jsxs)(n.p,{children:["Many of the plugin's functions utilize the ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modioerrorcode",children:(0,i.jsx)(n.code,{children:"ModioErrorCode"})})," type. For example, all ",(0,i.jsx)(n.a,{href:"#non-blocking-asynchronous-interface",children:"async"})," functions take a ",(0,i.jsx)(n.code,{children:"ModioErrorCode"})," as the first parameter for the function's delegate. This is essentially an opaque wrapper around a numeric error code with a category and an associated string message."]}),"\n",(0,i.jsx)(n.p,{children:"The plugin doesn\u2019t attempt to predict what your error-handling logic or requirements are. Instead, we return the error code to you so you can decide what to do. For instance, if you call a function and receive an error code matching a network error condition, do you want to close down the plugin? Retry again according to custom back-off logic? That decision is left to the consuming application."}),"\n",(0,i.jsxs)(n.p,{children:["For more details on the error codes and how to inspect their values, please see our ",(0,i.jsx)(n.a,{href:"/unreal/getting-started/error-handling",children:"Error Handling quick-start guide"}),"."]}),"\n",(0,i.jsx)(n.h2,{id:"user-sessions",children:"User sessions"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io plugin runs on a per-platform-user basis. If you are using the plugin on a platform that requires user switching support, you must call ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_ShutdownAsync",children:(0,i.jsx)(n.code,{children:"ShutdownAsync"})})," and re-initialize the plugin by calling ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_InitializeAsync",children:(0,i.jsx)(n.code,{children:"InitializeAsync"})})," with a different ",(0,i.jsx)(n.a,{href:"#session-ids",children:"Session ID"}),"."]}),"\n",(0,i.jsx)(n.p,{children:"Generating a stable platform-specific Session ID for each user will ensure that an incoming user who has already authenticated with mod.io on the current device and game won\u2019t need to re-authenticate unless their authentication token has expired."}),"\n",(0,i.jsx)(n.h3,{id:"session-ids",children:"Session IDs"}),"\n",(0,i.jsx)(n.p,{children:"Session IDs are chosen by the developer. They can be whatever you like as long as they are deterministic and stable. Session IDs create a scope or a local profile for the current user to live in so that a single system can support multiple authenticated users side-by-side without requiring de-authentication of the previous user. Internally, the Session ID is used to create a folder containing the authentication information and cached profile of the authenticated user."}),"\n",(0,i.jsx)(n.p,{children:"On console platforms, we suggest that the Session ID should be a string representation of the platform-provided user ID. This gives the best experience when it comes to things like user switching. For example, a game on Xbox using a sanitized string representation of the Xbox Live ID as the Session ID would have a folder structure in persistent storage like the following:"}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{children:"/mod.io///\n/mod.io///\n"})}),"\n",(0,i.jsx)(n.p,{children:"With this example, you can detect the user associated with the current controller on game start and initialize the mod.io plugin with the Session ID set to the stable string representation of that user's Xbox Live ID. As a result, the user's authentication status will be maintained from the previous session on that device."}),"\n",(0,i.jsx)(n.p,{children:"In the case of a Windows title with user-provided profile names, the same folder structure would be more like the following:"}),"\n",(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{children:"%USERDATA%/mod.io//ProfileName1/\n%USERDATA%/mod.io//ProfileName2/\n%USERDATA%/mod.io//ProfileName3/\n"})}),"\n",(0,i.jsx)(n.p,{children:"This allows multiple players, for example siblings, to each have their own session living in the same Windows account."}),"\n",(0,i.jsx)(n.h2,{id:"mod-data-directory",children:"Mod data directory"}),"\n",(0,i.jsx)(n.p,{children:"The plugin stores mods in a game-specific directory in the following directory by default:"}),"\n",(0,i.jsxs)(n.table,{children:[(0,i.jsx)(n.thead,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.th,{children:"Windows"}),(0,i.jsx)(n.th,{children:"Linux"}),(0,i.jsx)(n.th,{children:"macOS"}),(0,i.jsx)(n.th,{children:"iOS"})]})}),(0,i.jsx)(n.tbody,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:(0,i.jsx)(n.code,{children:"${FolderID_Public}/mod.io"})}),(0,i.jsx)(n.td,{children:(0,i.jsx)(n.code,{children:"${USER_HOME}/mod.io"})}),(0,i.jsx)(n.td,{children:(0,i.jsx)(n.code,{children:"${USER_HOME}/Library/Application Support/mod.io"})}),(0,i.jsx)(n.td,{children:(0,i.jsx)(n.code,{children:"${APP-DIRECTORY}/Documents/mod.io"})})]})})]}),"\n",(0,i.jsx)(n.p,{children:"However, this value can be overridden in one of two ways:"}),"\n",(0,i.jsx)(n.h3,{id:"globally-for-a-system-account",children:"Globally for a system account"}),"\n",(0,i.jsxs)(n.p,{children:["On first run of a game using the plugin, ",(0,i.jsx)(n.code,{children:"${FOLDERID_LocalAppData}/mod.io/globalsettings.json"})," will be created."]}),"\n",(0,i.jsxs)(n.p,{children:["This JSON object contains a ",(0,i.jsx)(n.code,{children:"RootLocalStoragePath"})," element - changing the string here to a valid path on disk will globally redirect the mod installation directory for ",(0,i.jsx)(n.strong,{children:"ALL"})," games using the mod.io Unreal Engine Plugin or Native SDK for the current system account."]}),"\n",(0,i.jsx)(n.admonition,{type:"warning",children:(0,i.jsx)(n.p,{children:"Changing this value while the SDK is initialized is not supported and behaviour is undefined."})}),"\n",(0,i.jsx)(n.h3,{id:"per-game-for-a-system-account",children:"Per-game for a system account"}),"\n",(0,i.jsxs)(n.p,{children:["Per-game settings are stored in ",(0,i.jsx)(n.code,{children:"${FOLDERID_LocalAppData}/mod.io/${GameId}/${mod.io user string}/user.json"}),"."]}),"\n",(0,i.jsxs)(n.p,{children:["Adding a ",(0,i.jsx)(n.code,{children:"RootLocalStoragePath"})," element to this file will redirect the mod installation directory for this specific game only for the current system account. Removing this value will cause the game to revert back to the global value in ",(0,i.jsx)(n.code,{children:"globalsettings.json"}),"."]}),"\n",(0,i.jsxs)(n.admonition,{type:"note",children:[(0,i.jsx)(n.mdxAdmonitionTitle,{}),(0,i.jsx)(n.p,{children:"In Linux and macOS, mods and cached data binds to a single user. Other client have their own instance in their home directory."})]})]})}function u(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(c,{...e})}):c(e)}},19365:(e,n,t)=>{t.d(n,{A:()=>s});t(96540);var i=t(18215);const o={tabItem:"tabItem_Ymn6"};var r=t(74848);function s(e){let{children:n,hidden:t,className:s}=e;return(0,r.jsx)("div",{role:"tabpanel",className:(0,i.A)(o.tabItem,s),hidden:t,children:n})}},11470:(e,n,t)=>{t.d(n,{A:()=>w});var i=t(96540),o=t(18215),r=t(23104),s=t(56347),a=t(205),l=t(57485),d=t(31682),c=t(70679);function u(e){return i.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,i.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:n,children:t}=e;return(0,i.useMemo)((()=>{const e=n??function(e){return u(e).map((e=>{let{props:{value:n,label:t,attributes:i,default:o}}=e;return{value:n,label:t,attributes:i,default:o}}))}(t);return function(e){const n=(0,d.X)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function p(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function m(e){let{queryString:n=!1,groupId:t}=e;const o=(0,s.W6)(),r=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,l.aZ)(r),(0,i.useCallback)((e=>{if(!r)return;const n=new URLSearchParams(o.location.search);n.set(r,e),o.replace({...o.location,search:n.toString()})}),[r,o])]}function g(e){const{defaultValue:n,queryString:t=!1,groupId:o}=e,r=h(e),[s,l]=(0,i.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!p({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const i=t.find((e=>e.default))??t[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:n,tabValues:r}))),[d,u]=m({queryString:t,groupId:o}),[g,f]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[o,r]=(0,c.Dv)(t);return[o,(0,i.useCallback)((e=>{t&&r.set(e)}),[t,r])]}({groupId:o}),x=(()=>{const e=d??g;return p({value:e,tabValues:r})?e:null})();(0,a.A)((()=>{x&&l(x)}),[x]);return{selectedValue:s,selectValue:(0,i.useCallback)((e=>{if(!p({value:e,tabValues:r}))throw new Error(`Can't select invalid tab value=${e}`);l(e),u(e),f(e)}),[u,f,r]),tabValues:r}}var f=t(92303);const x={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var b=t(74848);function y(e){let{className:n,block:t,selectedValue:i,selectValue:s,tabValues:a}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,r.a_)(),c=e=>{const n=e.currentTarget,t=l.indexOf(n),o=a[t].value;o!==i&&(d(n),s(o))},u=e=>{let n=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const t=l.indexOf(e.currentTarget)+1;n=l[t]??l[0];break}case"ArrowLeft":{const t=l.indexOf(e.currentTarget)-1;n=l[t]??l[l.length-1];break}}n?.focus()};return(0,b.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.A)("tabs",{"tabs--block":t},n),children:a.map((e=>{let{value:n,label:t,attributes:r}=e;return(0,b.jsx)("li",{role:"tab",tabIndex:i===n?0:-1,"aria-selected":i===n,ref:e=>l.push(e),onKeyDown:u,onClick:c,...r,className:(0,o.A)("tabs__item",x.tabItem,r?.className,{"tabs__item--active":i===n}),children:t??n},n)}))})}function v(e){let{lazy:n,children:t,selectedValue:o}=e;const r=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=r.find((e=>e.props.value===o));return e?(0,i.cloneElement)(e,{className:"margin-top--md"}):null}return(0,b.jsx)("div",{className:"margin-top--md",children:r.map(((e,n)=>(0,i.cloneElement)(e,{key:n,hidden:e.props.value!==o})))})}function j(e){const n=g(e);return(0,b.jsxs)("div",{className:(0,o.A)("tabs-container",x.tabList),children:[(0,b.jsx)(y,{...n,...e}),(0,b.jsx)(v,{...n,...e})]})}function w(e){const n=(0,f.A)();return(0,b.jsx)(j,{...e,children:u(e.children)},String(n))}},89236:(e,n,t)=>{t.d(n,{A:()=>r});var i=t(19365),o=(t(96540),t(74848));function r(e){return(0,o.jsx)(o.Fragment,{children:(0,o.jsx)(i.A,{className:"tw-rounded-md",...e})})}},27064:(e,n,t)=>{t.d(n,{A:()=>r});var i=t(11470),o=(t(96540),t(74848));function r(e){return(0,o.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,o.jsx)(i.A,{...e})})}},28453:(e,n,t)=>{t.d(n,{R:()=>s,x:()=>a});var i=t(96540);const o={},r=i.createContext(o);function s(e){const n=i.useContext(r);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:s(e.components),i.createElement(r.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/732ed63c.5c782035.js b/Doc/assets/js/732ed63c.5c782035.js deleted file mode 100644 index 46e73c00..00000000 --- a/Doc/assets/js/732ed63c.5c782035.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[1530],{20524:(e,t,r)=>{r.r(t),r.d(t,{assets:()=>d,contentTitle:()=>l,default:()=>h,frontMatter:()=>i,metadata:()=>u,toc:()=>c});var n=r(74848),s=r(28453),o=r(27064),a=r(89236);const i={id:"ue-browsing-mods",title:"Browsing Mods",slug:"/unreal/getting-started/browsing-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/browsing-mods.mdx"},l=void 0,u={id:"getting-started/ue-browsing-mods",title:"Browsing Mods",description:"After initializing the plugin and authenticating a user, you can query the available mods using ListAllModsAsync.",source:"@site/public/en-us/getting-started/browsing-mods.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/browsing-mods",permalink:"/unreal/getting-started/browsing-mods",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/browsing-mods.mdx",tags:[],version:"current",frontMatter:{id:"ue-browsing-mods",title:"Browsing Mods",slug:"/unreal/getting-started/browsing-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/browsing-mods.mdx"},sidebar:"sidebar",previous:{title:"User Authentication",permalink:"/unreal/getting-started/user-authentication"},next:{title:"Mod Subscriptions & Management",permalink:"/unreal/getting-started/mod-subscriptions"}},d={},c=[];function m(e){const t={a:"a",code:"code",img:"img",p:"p",pre:"pre",...(0,s.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsxs)(t.p,{children:["After ",(0,n.jsx)(t.a,{href:"initialization",children:"initializing the plugin"})," and ",(0,n.jsx)(t.a,{href:"user-authentication",children:"authenticating a user"}),", you can query the available mods using ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_ListAllModsAsync",children:(0,n.jsx)(t.code,{children:"ListAllModsAsync"})}),"."]}),"\n",(0,n.jsxs)(t.p,{children:[(0,n.jsx)(t.code,{children:"ListAllModsAsync"})," supports filtering by name, tag, author, mature content, and more using ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#_modiofilterparams",children:(0,n.jsx)(t.code,{children:"ModioFilterParams"})}),". You can sort results as specified by ",(0,n.jsx)(t.code,{children:"ModioSortFieldType"}),", and request paginated or indexed results. By default, ",(0,n.jsx)(t.code,{children:"ModioFilterParams"})," asks for the first 100 results (the maximum number returnable in a query) sorted by ",(0,n.jsx)(t.code,{children:"ModioModID"}),"."]}),"\n",(0,n.jsxs)(o.A,{"group-id":"languages",children:[(0,n.jsxs)(a.A,{value:"blueprint",label:"Blueprint",children:[(0,n.jsxs)(t.p,{children:["The primary way this is done is through ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_ListAllModsAsync",children:"K2_ListAllModsAsync"}),"."]}),(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"list_all_mods",src:r(91144).A+"",width:"1376",height:"582"})}),(0,n.jsxs)(t.p,{children:["To search for a specific query string, sort in a different order, or combine different filters, you can use a ",(0,n.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#_modiofilterparams",children:"ModioFilterParams"})," object like this:"]}),(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"list_all_mods_filter",src:r(99999).A+"",width:"1387",height:"655"})})]}),(0,n.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:'void UModioManagerSubsystem::ListAllMods()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tFModioFilterParams Filter;\n \t// Build the filter by chaining together multiple calls\n \tFilter.PagedResults(1, 5).IndexedResults(3, 5).WithTags("Multiplayer").SortBy(EModioSortFieldType::ID, EModioSortDirection::Descending);\n\n \tSubsystem->ListAllModsAsync(Filter, FOnListAllModsDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnListAllModsComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnListAllModsComplete(FModioErrorCode ErrorCode, TOptional OptionalModList)\n{\n // Ensure ListAllModsAsync was successful\n if (!ErrorCode)\n {\n \t// ModList is guaranteed to be valid if there is no error\n \tTArray ModInfoArray = OptionalModList.GetValue().GetRawList();\n\n \t// Do something with ModInfoArray\n\n \t// You can use OptionalModList().GetValue().Paged related methods to make further paginated requests if required\n }\n}\n\n'})})})]})]})}function h(e={}){const{wrapper:t}={...(0,s.R)(),...e.components};return t?(0,n.jsx)(t,{...e,children:(0,n.jsx)(m,{...e})}):m(e)}},19365:(e,t,r)=>{r.d(t,{A:()=>a});r(96540);var n=r(18215);const s={tabItem:"tabItem_Ymn6"};var o=r(74848);function a(e){let{children:t,hidden:r,className:a}=e;return(0,o.jsx)("div",{role:"tabpanel",className:(0,n.A)(s.tabItem,a),hidden:r,children:t})}},11470:(e,t,r)=>{r.d(t,{A:()=>w});var n=r(96540),s=r(18215),o=r(23104),a=r(56347),i=r(205),l=r(57485),u=r(31682),d=r(70679);function c(e){return n.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,n.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:r}=e;return(0,n.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:r,attributes:n,default:s}}=e;return{value:t,label:r,attributes:n,default:s}}))}(r);return function(e){const t=(0,u.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,r])}function h(e){let{value:t,tabValues:r}=e;return r.some((e=>e.value===t))}function g(e){let{queryString:t=!1,groupId:r}=e;const s=(0,a.W6)(),o=function(e){let{queryString:t=!1,groupId:r}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!r)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return r??null}({queryString:t,groupId:r});return[(0,l.aZ)(o),(0,n.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(s.location.search);t.set(o,e),s.replace({...s.location,search:t.toString()})}),[o,s])]}function p(e){const{defaultValue:t,queryString:r=!1,groupId:s}=e,o=m(e),[a,l]=(0,n.useState)((()=>function(e){let{defaultValue:t,tabValues:r}=e;if(0===r.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!h({value:t,tabValues:r}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${r.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const n=r.find((e=>e.default))??r[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:t,tabValues:o}))),[u,c]=g({queryString:r,groupId:s}),[p,b]=function(e){let{groupId:t}=e;const r=function(e){return e?`docusaurus.tab.${e}`:null}(t),[s,o]=(0,d.Dv)(r);return[s,(0,n.useCallback)((e=>{r&&o.set(e)}),[r,o])]}({groupId:s}),f=(()=>{const e=u??p;return h({value:e,tabValues:o})?e:null})();(0,i.A)((()=>{f&&l(f)}),[f]);return{selectedValue:a,selectValue:(0,n.useCallback)((e=>{if(!h({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),b(e)}),[c,b,o]),tabValues:o}}var b=r(92303);const f={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var y=r(74848);function v(e){let{className:t,block:r,selectedValue:n,selectValue:a,tabValues:i}=e;const l=[],{blockElementScrollPositionUntilNextRender:u}=(0,o.a_)(),d=e=>{const t=e.currentTarget,r=l.indexOf(t),s=i[r].value;s!==n&&(u(t),a(s))},c=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const r=l.indexOf(e.currentTarget)+1;t=l[r]??l[0];break}case"ArrowLeft":{const r=l.indexOf(e.currentTarget)-1;t=l[r]??l[l.length-1];break}}t?.focus()};return(0,y.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,s.A)("tabs",{"tabs--block":r},t),children:i.map((e=>{let{value:t,label:r,attributes:o}=e;return(0,y.jsx)("li",{role:"tab",tabIndex:n===t?0:-1,"aria-selected":n===t,ref:e=>l.push(e),onKeyDown:c,onClick:d,...o,className:(0,s.A)("tabs__item",f.tabItem,o?.className,{"tabs__item--active":n===t}),children:r??t},t)}))})}function x(e){let{lazy:t,children:r,selectedValue:s}=e;const o=(Array.isArray(r)?r:[r]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===s));return e?(0,n.cloneElement)(e,{className:"margin-top--md"}):null}return(0,y.jsx)("div",{className:"margin-top--md",children:o.map(((e,t)=>(0,n.cloneElement)(e,{key:t,hidden:e.props.value!==s})))})}function M(e){const t=p(e);return(0,y.jsxs)("div",{className:(0,s.A)("tabs-container",f.tabList),children:[(0,y.jsx)(v,{...t,...e}),(0,y.jsx)(x,{...t,...e})]})}function w(e){const t=(0,b.A)();return(0,y.jsx)(M,{...e,children:c(e.children)},String(t))}},89236:(e,t,r)=>{r.d(t,{A:()=>o});var n=r(19365),s=(r(96540),r(74848));function o(e){return(0,s.jsx)(s.Fragment,{children:(0,s.jsx)(n.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,r)=>{r.d(t,{A:()=>o});var n=r(11470),s=(r(96540),r(74848));function o(e){return(0,s.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,s.jsx)(n.A,{...e})})}},91144:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/list_all_mods-0849d8a4fb5c0a31b6942f132824ea04.png"},99999:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/list_all_mods_filter-64856fbea375ef27ca0c47169e161dfe.png"},28453:(e,t,r)=>{r.d(t,{R:()=>a,x:()=>i});var n=r(96540);const s={},o=n.createContext(s);function a(e){const t=n.useContext(o);return n.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function i(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:a(e.components),n.createElement(o.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/732ed63c.d993f088.js b/Doc/assets/js/732ed63c.d993f088.js new file mode 100644 index 00000000..115eb603 --- /dev/null +++ b/Doc/assets/js/732ed63c.d993f088.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[1530],{20524:(e,t,r)=>{r.r(t),r.d(t,{assets:()=>d,contentTitle:()=>l,default:()=>h,frontMatter:()=>i,metadata:()=>u,toc:()=>c});var n=r(74848),s=r(28453),o=r(27064),a=r(89236);const i={id:"ue-browsing-mods",title:"Browsing Mods",slug:"/unreal/getting-started/browsing-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/browsing-mods.mdx"},l=void 0,u={id:"getting-started/ue-browsing-mods",title:"Browsing Mods",description:"After initializing the plugin and authenticating a user, you can query the available mods using ListAllModsAsync.",source:"@site/public/en-us/getting-started/browsing-mods.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/browsing-mods",permalink:"/unreal/getting-started/browsing-mods",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/browsing-mods.mdx",tags:[],version:"current",frontMatter:{id:"ue-browsing-mods",title:"Browsing Mods",slug:"/unreal/getting-started/browsing-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/browsing-mods.mdx"},sidebar:"sidebar",previous:{title:"User Authentication",permalink:"/unreal/getting-started/user-authentication"},next:{title:"Mod Subscriptions & Management",permalink:"/unreal/getting-started/mod-subscriptions"}},d={},c=[];function m(e){const t={a:"a",code:"code",img:"img",p:"p",pre:"pre",...(0,s.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsxs)(t.p,{children:["After ",(0,n.jsx)(t.a,{href:"initialization",children:"initializing the plugin"})," and ",(0,n.jsx)(t.a,{href:"user-authentication",children:"authenticating a user"}),", you can query the available mods using ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#listallmodsasync",children:(0,n.jsx)(t.code,{children:"ListAllModsAsync"})}),"."]}),"\n",(0,n.jsxs)(t.p,{children:[(0,n.jsx)(t.code,{children:"ListAllModsAsync"})," supports filtering by name, tag, author, mature content, and more using ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#modiofilterparams",children:(0,n.jsx)(t.code,{children:"ModioFilterParams"})}),". You can sort results as specified by ",(0,n.jsx)(t.code,{children:"ModioSortFieldType"}),", and request paginated or indexed results. By default, ",(0,n.jsx)(t.code,{children:"ModioFilterParams"})," asks for the first 100 results (the maximum number returnable in a query) sorted by ",(0,n.jsx)(t.code,{children:"ModioModID"}),"."]}),"\n",(0,n.jsxs)(o.A,{"group-id":"languages",children:[(0,n.jsxs)(a.A,{value:"blueprint",label:"Blueprint",children:[(0,n.jsxs)(t.p,{children:["The primary way this is done is through ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#listallmodsasync",children:"K2_ListAllModsAsync"}),"."]}),(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"list_all_mods",src:r(4305).A+"",width:"1376",height:"582"})}),(0,n.jsxs)(t.p,{children:["To search for a specific query string, sort in a different order, or combine different filters, you can use a ",(0,n.jsx)(t.a,{href:"/unreal/refdocs/#modiofilterparams",children:"ModioFilterParams"})," object like this:"]}),(0,n.jsx)(t.p,{children:(0,n.jsx)(t.img,{alt:"list_all_mods_filter",src:r(10740).A+"",width:"1387",height:"655"})})]}),(0,n.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(t.pre,{children:(0,n.jsx)(t.code,{className:"language-cpp",children:'void UModioManagerSubsystem::ListAllMods()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tFModioFilterParams Filter;\n \t// Build the filter by chaining together multiple calls\n \tFilter.PagedResults(1, 5).IndexedResults(3, 5).WithTags("Multiplayer").SortBy(EModioSortFieldType::ID, EModioSortDirection::Descending);\n\n \tSubsystem->ListAllModsAsync(Filter, FOnListAllModsDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnListAllModsComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnListAllModsComplete(FModioErrorCode ErrorCode, TOptional OptionalModList)\n{\n // Ensure ListAllModsAsync was successful\n if (!ErrorCode)\n {\n \t// ModList is guaranteed to be valid if there is no error\n \tTArray ModInfoArray = OptionalModList.GetValue().GetRawList();\n\n \t// Do something with ModInfoArray\n\n \t// You can use OptionalModList().GetValue().Paged related methods to make further paginated requests if required\n }\n}\n\n'})})})]})]})}function h(e={}){const{wrapper:t}={...(0,s.R)(),...e.components};return t?(0,n.jsx)(t,{...e,children:(0,n.jsx)(m,{...e})}):m(e)}},19365:(e,t,r)=>{r.d(t,{A:()=>a});r(96540);var n=r(18215);const s={tabItem:"tabItem_Ymn6"};var o=r(74848);function a(e){let{children:t,hidden:r,className:a}=e;return(0,o.jsx)("div",{role:"tabpanel",className:(0,n.A)(s.tabItem,a),hidden:r,children:t})}},11470:(e,t,r)=>{r.d(t,{A:()=>M});var n=r(96540),s=r(18215),o=r(23104),a=r(56347),i=r(205),l=r(57485),u=r(31682),d=r(70679);function c(e){return n.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,n.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:r}=e;return(0,n.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:r,attributes:n,default:s}}=e;return{value:t,label:r,attributes:n,default:s}}))}(r);return function(e){const t=(0,u.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,r])}function h(e){let{value:t,tabValues:r}=e;return r.some((e=>e.value===t))}function g(e){let{queryString:t=!1,groupId:r}=e;const s=(0,a.W6)(),o=function(e){let{queryString:t=!1,groupId:r}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!r)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return r??null}({queryString:t,groupId:r});return[(0,l.aZ)(o),(0,n.useCallback)((e=>{if(!o)return;const t=new URLSearchParams(s.location.search);t.set(o,e),s.replace({...s.location,search:t.toString()})}),[o,s])]}function b(e){const{defaultValue:t,queryString:r=!1,groupId:s}=e,o=m(e),[a,l]=(0,n.useState)((()=>function(e){let{defaultValue:t,tabValues:r}=e;if(0===r.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!h({value:t,tabValues:r}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${r.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const n=r.find((e=>e.default))??r[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:t,tabValues:o}))),[u,c]=g({queryString:r,groupId:s}),[b,f]=function(e){let{groupId:t}=e;const r=function(e){return e?`docusaurus.tab.${e}`:null}(t),[s,o]=(0,d.Dv)(r);return[s,(0,n.useCallback)((e=>{r&&o.set(e)}),[r,o])]}({groupId:s}),p=(()=>{const e=u??b;return h({value:e,tabValues:o})?e:null})();(0,i.A)((()=>{p&&l(p)}),[p]);return{selectedValue:a,selectValue:(0,n.useCallback)((e=>{if(!h({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),f(e)}),[c,f,o]),tabValues:o}}var f=r(92303);const p={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var y=r(74848);function v(e){let{className:t,block:r,selectedValue:n,selectValue:a,tabValues:i}=e;const l=[],{blockElementScrollPositionUntilNextRender:u}=(0,o.a_)(),d=e=>{const t=e.currentTarget,r=l.indexOf(t),s=i[r].value;s!==n&&(u(t),a(s))},c=e=>{let t=null;switch(e.key){case"Enter":d(e);break;case"ArrowRight":{const r=l.indexOf(e.currentTarget)+1;t=l[r]??l[0];break}case"ArrowLeft":{const r=l.indexOf(e.currentTarget)-1;t=l[r]??l[l.length-1];break}}t?.focus()};return(0,y.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,s.A)("tabs",{"tabs--block":r},t),children:i.map((e=>{let{value:t,label:r,attributes:o}=e;return(0,y.jsx)("li",{role:"tab",tabIndex:n===t?0:-1,"aria-selected":n===t,ref:e=>l.push(e),onKeyDown:c,onClick:d,...o,className:(0,s.A)("tabs__item",p.tabItem,o?.className,{"tabs__item--active":n===t}),children:r??t},t)}))})}function x(e){let{lazy:t,children:r,selectedValue:s}=e;const o=(Array.isArray(r)?r:[r]).filter(Boolean);if(t){const e=o.find((e=>e.props.value===s));return e?(0,n.cloneElement)(e,{className:"margin-top--md"}):null}return(0,y.jsx)("div",{className:"margin-top--md",children:o.map(((e,t)=>(0,n.cloneElement)(e,{key:t,hidden:e.props.value!==s})))})}function w(e){const t=b(e);return(0,y.jsxs)("div",{className:(0,s.A)("tabs-container",p.tabList),children:[(0,y.jsx)(v,{...t,...e}),(0,y.jsx)(x,{...t,...e})]})}function M(e){const t=(0,f.A)();return(0,y.jsx)(w,{...e,children:c(e.children)},String(t))}},89236:(e,t,r)=>{r.d(t,{A:()=>o});var n=r(19365),s=(r(96540),r(74848));function o(e){return(0,s.jsx)(s.Fragment,{children:(0,s.jsx)(n.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,r)=>{r.d(t,{A:()=>o});var n=r(11470),s=(r(96540),r(74848));function o(e){return(0,s.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,s.jsx)(n.A,{...e})})}},4305:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/list_all_mods-0849d8a4fb5c0a31b6942f132824ea04.png"},10740:(e,t,r)=>{r.d(t,{A:()=>n});const n=r.p+"assets/images/list_all_mods_filter-64856fbea375ef27ca0c47169e161dfe.png"},28453:(e,t,r)=>{r.d(t,{R:()=>a,x:()=>i});var n=r(96540);const s={},o=n.createContext(s);function a(e){const t=n.useContext(o);return n.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function i(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:a(e.components),n.createElement(o.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/8ac087ef.27a161e3.js b/Doc/assets/js/8ac087ef.27a161e3.js deleted file mode 100644 index 0d38f7d2..00000000 --- a/Doc/assets/js/8ac087ef.27a161e3.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[1493],{8822:(e,n,s)=>{s.r(n),s.d(n,{assets:()=>c,contentTitle:()=>d,default:()=>a,frontMatter:()=>o,metadata:()=>r,toc:()=>t});var i=s(74848),l=s(28453);const o={id:"ue-asset-support-for-mods",title:"Asset Support for Mods",slug:"/unreal/asset-support-for-mods/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/asset-support-for-mods"},d=void 0,r={id:"ue-asset-support-for-mods",title:"Asset Support for Mods",description:"Overview",source:"@site/public/en-us/asset-support-for-mods.mdx",sourceDirName:".",slug:"/unreal/asset-support-for-mods/",permalink:"/unreal/asset-support-for-mods/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/asset-support-for-mods",tags:[],version:"current",frontMatter:{id:"ue-asset-support-for-mods",title:"Asset Support for Mods",slug:"/unreal/asset-support-for-mods/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/asset-support-for-mods"},sidebar:"sidebar",previous:{title:"Marketplace",permalink:"/unreal/marketplace/"},next:{title:"Profiling",permalink:"/unreal/profiling/"}},c={},t=[{value:"Overview",id:"overview",level:2},{value:"Supported Asset Types",id:"supported-asset-types",level:2},{value:"Allowed Actors In Levels",id:"allowed-actors-in-levels",level:3},{value:"Allowed Components in Actors",id:"allowed-components-in-actors",level:3},{value:"Unsupported Asset Types",id:"unsupported-asset-types",level:2},{value:"Prohibited Actors in Levels",id:"prohibited-actors-in-levels",level:3},{value:"Prohibited Components in Actors",id:"prohibited-components-in-actors",level:3},{value:"Asset Support in Packages (UPackage)",id:"asset-support-in-packages-upackage",level:2}];function h(e){const n={code:"code",h2:"h2",h3:"h3",li:"li",ol:"ol",p:"p",ul:"ul",...(0,l.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"overview",children:"Overview"}),"\n",(0,i.jsx)(n.p,{children:"The main factor for supporting assets is dependency: only dependencies to the project's content are allowed, but not engine's dependency. When trying to package any asset that has engine dependency, it will fail."}),"\n",(0,i.jsxs)(n.p,{children:["Some assets have internal dependencies which can't be bypassed without modifying the engine's source code. For example, when trying to create a directional light or point light in the level, you'll have an ",(0,i.jsx)(n.code,{children:"EmptyActor"})," dependency to the engine - packaging of that asset and consequently the whole mod will fail. Therefore, you should generally avoid using any engine's assets dependency (except for very basic ones, such as dependency to ",(0,i.jsx)(n.code,{children:"AActor"})," class), and instead create your own classes in your project and implement the necessary functionality there, and allow the mod to use those classes instead of the engine's ones."]}),"\n",(0,i.jsx)(n.h2,{id:"supported-asset-types",children:"Supported Asset Types"}),"\n",(0,i.jsx)(n.h3,{id:"allowed-actors-in-levels",children:"Allowed Actors In Levels"}),"\n",(0,i.jsx)(n.p,{children:"These actors, including custom actors derived from them and saved in the project or mod, can be placed in levels without causing packaging issues in the mod:"}),"\n",(0,i.jsxs)(n.ol,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AStaticMeshActor"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"APlayerStart"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"APawn"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ACharacter"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ADefaultPawn"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived actors"}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ABrush"})," & ",(0,i.jsx)(n.code,{children:"AVolume"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AKillZVolume"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ABlockingVolume"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AAudioVolume"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ACullDistanceVolume"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"APostProcessVolume"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived actors"}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ADecalActor"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"AInfo"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AVolumetricCloud"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AExponentialHeightFog"})}),"\n",(0,i.jsx)(n.li,{children:"Other supported derived actors"}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ATriggerBase"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ATriggerBox"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ATriggerCapsule"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ATriggerSphere"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived actors"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.h3,{id:"allowed-components-in-actors",children:"Allowed Components in Actors"}),"\n",(0,i.jsx)(n.p,{children:"These components, including custom components derived from them and saved in the project or mod, can be placed in actors without causing packaging issues in the mod:"}),"\n",(0,i.jsxs)(n.ol,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"USceneComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"UPrimitiveComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UTextRenderComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UArrowComponent"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"UShapeComponent"})," (collisions)","\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UBoxComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UCapsuleComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"USphereComponent"})}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"UMeshComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UStaticMeshComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UWidgetComponent"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived components"}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UCameraComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UChildActorComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UText3DComponent"})}),"\n",(0,i.jsx)(n.li,{children:"Other supported derived components"}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UWidgetComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UCameraComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UDecalComponent"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"UMovementComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UPawnMovementComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UCharacterMovementComponent"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived components"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"All other derived components"}),"\n"]}),"\n",(0,i.jsx)(n.h2,{id:"unsupported-asset-types",children:"Unsupported Asset Types"}),"\n",(0,i.jsx)(n.h3,{id:"prohibited-actors-in-levels",children:"Prohibited Actors in Levels"}),"\n",(0,i.jsx)(n.p,{children:"These actors, including custom actors derived from them and saved in the project or mod, should not be placed in levels as they will likely cause packaging issues in the mod:"}),"\n",(0,i.jsxs)(n.ol,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ALight"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ADirectionalLight"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"APointLight"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ARectLight"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ASpotLight"})}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"AInfo"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ASkyLight"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ASkyAtmosphere"})}),"\n",(0,i.jsx)(n.li,{children:"Other unsupported derived actors"}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ASkyAtmosphere"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ATextRenderActor"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Other unsupported derived actors"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.h3,{id:"prohibited-components-in-actors",children:"Prohibited Components in Actors"}),"\n",(0,i.jsx)(n.p,{children:"These components, including custom components derived from them and saved in the project or mod, should not be placed in actors as they will likely cause packaging issues in the mod:"}),"\n",(0,i.jsxs)(n.ol,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"USceneComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ULightComponentBase"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UDirectionalLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ULightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ULocalLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UPointLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"URectLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"USkyLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"USpotLightComponent"})}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"USynthComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UAudioComponent"})}),"\n",(0,i.jsx)(n.li,{children:"Other unsupported derived components"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.h2,{id:"asset-support-in-packages-upackage",children:"Asset Support in Packages (UPackage)"}),"\n",(0,i.jsx)(n.p,{children:"Generally, all possible asset types are supported to be included in the plugin after the packaging. The tested ones are listed below, but any other assets without dependency on the engine are supported as well, such as manually created ones in the mod itself:"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Actor"}),"\n",(0,i.jsx)(n.li,{children:"Pawn"}),"\n",(0,i.jsx)(n.li,{children:"Character"}),"\n",(0,i.jsx)(n.li,{children:"Actor Component"}),"\n",(0,i.jsx)(n.li,{children:"Scene Component"}),"\n",(0,i.jsx)(n.li,{children:"Level"}),"\n",(0,i.jsx)(n.li,{children:"Material"}),"\n",(0,i.jsx)(n.li,{children:"Widget"}),"\n",(0,i.jsx)(n.li,{children:"Texture"}),"\n",(0,i.jsx)(n.li,{children:"Sound Wave"}),"\n",(0,i.jsx)(n.li,{children:"Static Mesh"}),"\n",(0,i.jsx)(n.li,{children:"Skeletal Mesh"}),"\n"]})]})}function a(e={}){const{wrapper:n}={...(0,l.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},28453:(e,n,s)=>{s.d(n,{R:()=>d,x:()=>r});var i=s(96540);const l={},o=i.createContext(l);function d(e){const n=i.useContext(o);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(l):e.components||l:d(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/8ac087ef.4e5b794d.js b/Doc/assets/js/8ac087ef.4e5b794d.js new file mode 100644 index 00000000..ad043293 --- /dev/null +++ b/Doc/assets/js/8ac087ef.4e5b794d.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[1493],{8822:(e,n,s)=>{s.r(n),s.d(n,{assets:()=>c,contentTitle:()=>d,default:()=>a,frontMatter:()=>l,metadata:()=>r,toc:()=>t});var i=s(74848),o=s(28453);const l={id:"ue-asset-support-for-mods",title:"Asset Support for Mods",slug:"/unreal/asset-support-for-mods/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/asset-support-for-mods"},d=void 0,r={id:"ue-asset-support-for-mods",title:"Asset Support for Mods",description:"Overview",source:"@site/public/en-us/asset-support-for-mods.mdx",sourceDirName:".",slug:"/unreal/asset-support-for-mods/",permalink:"/unreal/asset-support-for-mods/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/asset-support-for-mods",tags:[],version:"current",frontMatter:{id:"ue-asset-support-for-mods",title:"Asset Support for Mods",slug:"/unreal/asset-support-for-mods/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/asset-support-for-mods"},sidebar:"sidebar",previous:{title:"Marketplace",permalink:"/unreal/marketplace/"},next:{title:"Profiling",permalink:"/unreal/profiling/"}},c={},t=[{value:"Overview",id:"overview",level:2},{value:"Supported Asset Types",id:"supported-asset-types",level:2},{value:"Allowed Actors In Levels",id:"allowed-actors-in-levels",level:3},{value:"Allowed Components in Actors",id:"allowed-components-in-actors",level:3},{value:"Unsupported Asset Types",id:"unsupported-asset-types",level:2},{value:"Prohibited Actors in Levels",id:"prohibited-actors-in-levels",level:3},{value:"Prohibited Components in Actors",id:"prohibited-components-in-actors",level:3},{value:"Asset Support in Packages (UPackage)",id:"asset-support-in-packages-upackage",level:2}];function h(e){const n={code:"code",h2:"h2",h3:"h3",li:"li",ol:"ol",p:"p",ul:"ul",...(0,o.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"overview",children:"Overview"}),"\n",(0,i.jsx)(n.p,{children:"The main factor for supporting assets is dependency: only dependencies to the project's content are allowed, but not engine's dependency. When trying to package any asset that has engine dependency, it will fail."}),"\n",(0,i.jsxs)(n.p,{children:["Some assets have internal dependencies which can't be bypassed without modifying the engine's source code. For example, when trying to create a directional light or point light in the level, you'll have an ",(0,i.jsx)(n.code,{children:"EmptyActor"})," dependency to the engine - packaging of that asset and consequently the whole mod will fail. Therefore, you should generally avoid using any engine's assets dependency (except for very basic ones, such as dependency to ",(0,i.jsx)(n.code,{children:"AActor"})," class), and instead create your own classes in your project and implement the necessary functionality there, and allow the mod to use those classes instead of the engine's ones."]}),"\n",(0,i.jsx)(n.h2,{id:"supported-asset-types",children:"Supported Asset Types"}),"\n",(0,i.jsx)(n.h3,{id:"allowed-actors-in-levels",children:"Allowed Actors In Levels"}),"\n",(0,i.jsx)(n.p,{children:"These actors, including custom actors derived from them and saved in the project or mod, can be placed in levels without causing packaging issues in the mod:"}),"\n",(0,i.jsxs)(n.ol,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AStaticMeshActor"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"APlayerStart"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"APawn"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ACharacter"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ADefaultPawn"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived actors"}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ABrush"})," & ",(0,i.jsx)(n.code,{children:"AVolume"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AKillZVolume"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ABlockingVolume"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AAudioVolume"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ACullDistanceVolume"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"APostProcessVolume"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived actors"}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ADecalActor"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"AInfo"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AVolumetricCloud"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"AExponentialHeightFog"})}),"\n",(0,i.jsx)(n.li,{children:"Other supported derived actors"}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ATriggerBase"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ATriggerBox"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ATriggerCapsule"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ATriggerSphere"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived actors"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.h3,{id:"allowed-components-in-actors",children:"Allowed Components in Actors"}),"\n",(0,i.jsx)(n.p,{children:"These components, including custom components derived from them and saved in the project or mod, can be placed in actors without causing packaging issues in the mod:"}),"\n",(0,i.jsxs)(n.ol,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"USceneComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"UPrimitiveComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UTextRenderComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UArrowComponent"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"UShapeComponent"})," (collisions)","\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UBoxComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UCapsuleComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"USphereComponent"})}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"UMeshComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UStaticMeshComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UWidgetComponent"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived components"}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UCameraComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UChildActorComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UText3DComponent"})}),"\n",(0,i.jsx)(n.li,{children:"Other supported derived components"}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UWidgetComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UCameraComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UDecalComponent"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"UMovementComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UPawnMovementComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UCharacterMovementComponent"})}),"\n",(0,i.jsx)(n.li,{children:"All other derived components"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"All other derived components"}),"\n"]}),"\n",(0,i.jsx)(n.h2,{id:"unsupported-asset-types",children:"Unsupported Asset Types"}),"\n",(0,i.jsx)(n.h3,{id:"prohibited-actors-in-levels",children:"Prohibited Actors in Levels"}),"\n",(0,i.jsx)(n.p,{children:"These actors, including custom actors derived from them and saved in the project or mod, should not be placed in levels as they will likely cause packaging issues in the mod:"}),"\n",(0,i.jsxs)(n.ol,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ALight"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ADirectionalLight"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"APointLight"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ARectLight"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ASpotLight"})}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"AInfo"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ASkyLight"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ASkyAtmosphere"})}),"\n",(0,i.jsx)(n.li,{children:"Other unsupported derived actors"}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ASkyAtmosphere"})}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ATextRenderActor"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Other unsupported derived actors"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.h3,{id:"prohibited-components-in-actors",children:"Prohibited Components in Actors"}),"\n",(0,i.jsx)(n.p,{children:"These components, including custom components derived from them and saved in the project or mod, should not be placed in actors as they will likely cause packaging issues in the mod:"}),"\n",(0,i.jsxs)(n.ol,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"USceneComponent"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"ULightComponentBase"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UDirectionalLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ULightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"ULocalLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UPointLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"URectLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"USkyLightComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"USpotLightComponent"})}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"USynthComponent"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.code,{children:"UAudioComponent"})}),"\n",(0,i.jsx)(n.li,{children:"Other unsupported derived components"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.h2,{id:"asset-support-in-packages-upackage",children:"Asset Support in Packages (UPackage)"}),"\n",(0,i.jsx)(n.p,{children:"Generally, all possible asset types are supported to be included in the plugin after the packaging. The tested ones are listed below, but any other assets without dependency on the engine are supported as well, such as manually created ones in the mod itself:"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Actor"}),"\n",(0,i.jsx)(n.li,{children:"Pawn"}),"\n",(0,i.jsx)(n.li,{children:"Character"}),"\n",(0,i.jsx)(n.li,{children:"Actor Component"}),"\n",(0,i.jsx)(n.li,{children:"Scene Component"}),"\n",(0,i.jsx)(n.li,{children:"Level"}),"\n",(0,i.jsx)(n.li,{children:"Material"}),"\n",(0,i.jsx)(n.li,{children:"Widget"}),"\n",(0,i.jsx)(n.li,{children:"Texture"}),"\n",(0,i.jsx)(n.li,{children:"Sound Wave"}),"\n",(0,i.jsx)(n.li,{children:"Static Mesh"}),"\n",(0,i.jsx)(n.li,{children:"Skeletal Mesh"}),"\n"]})]})}function a(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},28453:(e,n,s)=>{s.d(n,{R:()=>d,x:()=>r});var i=s(96540);const o={},l=i.createContext(o);function d(e){const n=i.useContext(l);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:d(e.components),i.createElement(l.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/8f59627d.be8524c0.js b/Doc/assets/js/8f59627d.be8524c0.js new file mode 100644 index 00000000..6ddc5d96 --- /dev/null +++ b/Doc/assets/js/8f59627d.be8524c0.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[3586],{47723:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>u,contentTitle:()=>l,default:()=>g,frontMatter:()=>s,metadata:()=>d,toc:()=>c});var i=t(74848),r=t(28453),a=t(27064),o=t(89236);const s={id:"ue-initialization",title:"Initialization & Teardown",slug:"/unreal/getting-started/initialization",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/initialization.mdx"},l=void 0,d={id:"getting-started/ue-initialization",title:"Initialization & Teardown",description:"Best practices",source:"@site/public/en-us/getting-started/initialization.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/initialization",permalink:"/unreal/getting-started/initialization",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/initialization.mdx",tags:[],version:"current",frontMatter:{id:"ue-initialization",title:"Initialization & Teardown",slug:"/unreal/getting-started/initialization",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/initialization.mdx"},sidebar:"sidebar",previous:{title:"Plugin Structure & Concepts",permalink:"/unreal/getting-started/plugin-structure/"},next:{title:"User Authentication",permalink:"/unreal/getting-started/user-authentication"}},u={},c=[{value:"Best practices",id:"best-practices",level:2},{value:"Calling RunPendingHandlers",id:"calling-runpendinghandlers",level:2},{value:"Initialization",id:"initialization",level:2},{value:"Shutdown",id:"shutdown",level:2}];function h(e){const n={a:"a",admonition:"admonition",code:"code",em:"em",h2:"h2",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,r.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"best-practices",children:"Best practices"}),"\n",(0,i.jsxs)(n.p,{children:["This guide describes the mod.io Unreal Engine plugin's basic event loop, intialization, and shutdown functions. Ideally, you should wrap these functions in a separate subsystem (e.g. ",(0,i.jsx)(n.code,{children:"ModioManagerSubsystem"}),") which is responsible for managing the full initialization and authentication flow for your game."]}),"\n",(0,i.jsxs)(n.p,{children:["For more information, see our ",(0,i.jsx)(n.a,{href:"user-authentication",children:"User Authentication"})," quick-start guide."]}),"\n",(0,i.jsx)(n.h2,{id:"calling-runpendinghandlers",children:"Calling RunPendingHandlers"}),"\n",(0,i.jsxs)(n.p,{children:["Before initializing the plugin, ensure ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#run-pending-handlers",children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})})," is being continually called. You can do this yourself in your project\u2019s main loop, or by setting ",(0,i.jsx)(n.a,{href:"/unreal/installation-and-setup/#using-a-background-thread",children:(0,i.jsx)(n.code,{children:"Use Background Thread"})})," to true in the mod.io project settings. For best performance, ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"})," should be called at least once per frame. See ",(0,i.jsx)(n.a,{href:"/unreal/getting-started/plugin-structure/#maintaining-the-plugin-event-loop",children:"Maintaining the plugin event loop"})," for additional information."]}),"\n",(0,i.jsx)(n.h2,{id:"initialization",children:"Initialization"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine plugin is initialized by calling ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#initializeasync",children:(0,i.jsx)(n.code,{children:"InitializeAsync"})}),", passing in a ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#modioinitializeoptions",children:(0,i.jsx)(n.code,{children:"ModioInitializeOptions"})})," with the relevant information for the current session, and a callback containing a ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#modioerrorcode",children:(0,i.jsx)(n.code,{children:"ModioErrorCode"})})," that will contain the result of the initialization on completion."]}),"\n",(0,i.jsxs)(n.p,{children:["Use ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#get-project-initialize-options-for-session-id",children:(0,i.jsx)(n.code,{children:"GetProjectInitializeOptionsForSessionId"})})," to initialize the plugin using the information entered in the ",(0,i.jsx)(n.a,{href:"/unreal/installation-and-setup/#plugin-configuration",children:"mod.io project settings"}),". On Windows, ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#get-default-session-id-windows",children:(0,i.jsx)(n.code,{children:"GetDefaultSessionIdWindows"})})," can be used as the ",(0,i.jsx)(n.a,{href:"/unreal/getting-started/plugin-structure/#session-ids",children:"Session ID"}),". Some platforms also require additional parameters added to ",(0,i.jsx)(n.code,{children:"ModioInitializeOptions.ExtendedInitializationParameters"}),". Refer to the relevant ",(0,i.jsx)(n.a,{href:"/platforms/",children:"platform documentation"})," for more information."]}),"\n",(0,i.jsxs)(a.A,{groupId:"languages",children:[(0,i.jsx)(o.A,{value:"cpp",label:"C++",children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::Init()\n{\t\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->InitializeAsync(UModioSDKLibrary::GetProjectInitializeOptionsForSessionId(UModioSDKLibrary::GetDefaultSessionIdWindows()), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnInitCallback));\n }\n UE_LOG(LogModioGame, Log, TEXT("Initializing mod.io"));\n}\n\nvoid UModioManagerSubsystem::OnInitCallback(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("mod.io initialization complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n\nvoid UModioManagerSubsystem::Tick(float DeltaTime)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->RunPendingHandlers();\n }\n}\n'})})}),(0,i.jsxs)(o.A,{value:"blueprint",label:"Blueprint",children:[(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"run_pending_handlers",src:t(82236).A+"",width:"689",height:"312"})}),(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"initasync_getoptions",src:t(65910).A+"",width:"1327",height:"556"})})]})]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["The error-handling in this sample has been omitted. See our ",(0,i.jsx)(n.a,{href:"error-handling",children:(0,i.jsx)(n.strong,{children:"Error Handling quick-start guide"})})," for more information."]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"InitializeAsync"})," is a asynchronous function, therefore you ",(0,i.jsx)(n.em,{children:"must"})," wait for the callback for confirmation that the initialization is complete."]}),"\n"]})}),"\n",(0,i.jsx)(n.h2,{id:"shutdown",children:"Shutdown"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine plugin is shut down by calling ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#shutdownasync",children:(0,i.jsx)(n.code,{children:"ShutdownAsync"})})," in a similar fashion."]}),"\n",(0,i.jsxs)(a.A,{groupId:"languages",children:[(0,i.jsx)(o.A,{value:"cpp",label:"C++",children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::Shutdown()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->ShutdownAsync(FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnShutdownComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnShutdownComplete(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("Shutdown complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n'})})}),(0,i.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"shutdownasync_runhandlers",src:t(2762).A+"",width:"996",height:"662"})})})]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["You ",(0,i.jsx)(n.em,{children:"must"})," continue to call ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#run-pending-handlers",children:(0,i.jsx)(n.strong,{children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})})})," while shutdown is in progress to allow intermediate handlers to complete."]}),"\n",(0,i.jsx)(n.li,{children:"Any in-flight mod.io async calls will complete with an error code indicating cancellation."}),"\n",(0,i.jsxs)(n.li,{children:["You ",(0,i.jsx)(n.em,{children:"must not"})," call ",(0,i.jsx)(n.strong,{children:(0,i.jsx)(n.code,{children:"ShutdownAsync"})})," from within another completion handler/callback. This may cause the application to deadlock."]}),"\n"]})})]})}function g(e={}){const{wrapper:n}={...(0,r.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},19365:(e,n,t)=>{t.d(n,{A:()=>o});t(96540);var i=t(18215);const r={tabItem:"tabItem_Ymn6"};var a=t(74848);function o(e){let{children:n,hidden:t,className:o}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,i.A)(r.tabItem,o),hidden:t,children:n})}},11470:(e,n,t)=>{t.d(n,{A:()=>w});var i=t(96540),r=t(18215),a=t(23104),o=t(56347),s=t(205),l=t(57485),d=t(31682),u=t(70679);function c(e){return i.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,i.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:n,children:t}=e;return(0,i.useMemo)((()=>{const e=n??function(e){return c(e).map((e=>{let{props:{value:n,label:t,attributes:i,default:r}}=e;return{value:n,label:t,attributes:i,default:r}}))}(t);return function(e){const n=(0,d.X)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function g(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function m(e){let{queryString:n=!1,groupId:t}=e;const r=(0,o.W6)(),a=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,l.aZ)(a),(0,i.useCallback)((e=>{if(!a)return;const n=new URLSearchParams(r.location.search);n.set(a,e),r.replace({...r.location,search:n.toString()})}),[a,r])]}function p(e){const{defaultValue:n,queryString:t=!1,groupId:r}=e,a=h(e),[o,l]=(0,i.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!g({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const i=t.find((e=>e.default))??t[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:n,tabValues:a}))),[d,c]=m({queryString:t,groupId:r}),[p,f]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[r,a]=(0,u.Dv)(t);return[r,(0,i.useCallback)((e=>{t&&a.set(e)}),[t,a])]}({groupId:r}),b=(()=>{const e=d??p;return g({value:e,tabValues:a})?e:null})();(0,s.A)((()=>{b&&l(b)}),[b]);return{selectedValue:o,selectValue:(0,i.useCallback)((e=>{if(!g({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),f(e)}),[c,f,a]),tabValues:a}}var f=t(92303);const b={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var x=t(74848);function j(e){let{className:n,block:t,selectedValue:i,selectValue:o,tabValues:s}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,a.a_)(),u=e=>{const n=e.currentTarget,t=l.indexOf(n),r=s[t].value;r!==i&&(d(n),o(r))},c=e=>{let n=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const t=l.indexOf(e.currentTarget)+1;n=l[t]??l[0];break}case"ArrowLeft":{const t=l.indexOf(e.currentTarget)-1;n=l[t]??l[l.length-1];break}}n?.focus()};return(0,x.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.A)("tabs",{"tabs--block":t},n),children:s.map((e=>{let{value:n,label:t,attributes:a}=e;return(0,x.jsx)("li",{role:"tab",tabIndex:i===n?0:-1,"aria-selected":i===n,ref:e=>l.push(e),onKeyDown:c,onClick:u,...a,className:(0,r.A)("tabs__item",b.tabItem,a?.className,{"tabs__item--active":i===n}),children:t??n},n)}))})}function y(e){let{lazy:n,children:t,selectedValue:r}=e;const a=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=a.find((e=>e.props.value===r));return e?(0,i.cloneElement)(e,{className:"margin-top--md"}):null}return(0,x.jsx)("div",{className:"margin-top--md",children:a.map(((e,n)=>(0,i.cloneElement)(e,{key:n,hidden:e.props.value!==r})))})}function v(e){const n=p(e);return(0,x.jsxs)("div",{className:(0,r.A)("tabs-container",b.tabList),children:[(0,x.jsx)(j,{...n,...e}),(0,x.jsx)(y,{...n,...e})]})}function w(e){const n=(0,f.A)();return(0,x.jsx)(v,{...e,children:c(e.children)},String(n))}},89236:(e,n,t)=>{t.d(n,{A:()=>a});var i=t(19365),r=(t(96540),t(74848));function a(e){return(0,r.jsx)(r.Fragment,{children:(0,r.jsx)(i.A,{className:"tw-rounded-md",...e})})}},27064:(e,n,t)=>{t.d(n,{A:()=>a});var i=t(11470),r=(t(96540),t(74848));function a(e){return(0,r.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,r.jsx)(i.A,{...e})})}},65910:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/initasync_getoptions-297e9d77d325d1bc7ab215940cd91c36.png"},82236:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/run_pending_handlers-3e2c3e1b54ee420c6d94eaf542368454.png"},2762:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/shutdownasync_runhandlers-c2e93330aca93ab1f70644e5fe625841.png"},28453:(e,n,t)=>{t.d(n,{R:()=>o,x:()=>s});var i=t(96540);const r={},a=i.createContext(r);function o(e){const n=i.useContext(a);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function s(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:o(e.components),i.createElement(a.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/8f59627d.f96d3965.js b/Doc/assets/js/8f59627d.f96d3965.js deleted file mode 100644 index d7149cb5..00000000 --- a/Doc/assets/js/8f59627d.f96d3965.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[3586],{47723:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>u,contentTitle:()=>l,default:()=>m,frontMatter:()=>s,metadata:()=>d,toc:()=>c});var i=t(74848),r=t(28453),o=t(27064),a=t(89236);const s={id:"ue-initialization",title:"Initialization & Teardown",slug:"/unreal/getting-started/initialization",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/initialization.mdx"},l=void 0,d={id:"getting-started/ue-initialization",title:"Initialization & Teardown",description:"Best practices",source:"@site/public/en-us/getting-started/initialization.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/initialization",permalink:"/unreal/getting-started/initialization",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/initialization.mdx",tags:[],version:"current",frontMatter:{id:"ue-initialization",title:"Initialization & Teardown",slug:"/unreal/getting-started/initialization",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/initialization.mdx"},sidebar:"sidebar",previous:{title:"Plugin Structure & Concepts",permalink:"/unreal/getting-started/plugin-structure/"},next:{title:"User Authentication",permalink:"/unreal/getting-started/user-authentication"}},u={},c=[{value:"Best practices",id:"best-practices",level:2},{value:"Calling RunPendingHandlers",id:"calling-runpendinghandlers",level:2},{value:"Initialization",id:"initialization",level:2},{value:"Shutdown",id:"shutdown",level:2}];function h(e){const n={a:"a",admonition:"admonition",code:"code",em:"em",h2:"h2",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,r.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"best-practices",children:"Best practices"}),"\n",(0,i.jsxs)(n.p,{children:["This guide describes the mod.io Unreal Engine plugin's basic event loop, intialization, and shutdown functions. Ideally, you should wrap these functions in a separate subsystem (e.g. ",(0,i.jsx)(n.code,{children:"ModioManagerSubsystem"}),") which is responsible for managing the full initialization and authentication flow for your game."]}),"\n",(0,i.jsxs)(n.p,{children:["For more information, see our ",(0,i.jsx)(n.a,{href:"user-authentication",children:"User Authentication"})," quick-start guide."]}),"\n",(0,i.jsx)(n.h2,{id:"calling-runpendinghandlers",children:"Calling RunPendingHandlers"}),"\n",(0,i.jsxs)(n.p,{children:["Before initializing the plugin, ensure ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#RunPendingHandlers",children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})})," is being continually called. You can do this yourself in your project\u2019s main loop, or by setting ",(0,i.jsx)(n.a,{href:"/unreal/installation-and-setup/#using-a-background-thread",children:(0,i.jsx)(n.code,{children:"Use Background Thread"})})," to true in the mod.io project settings. For best performance, ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"})," should be called at least once per frame. See ",(0,i.jsx)(n.a,{href:"/unreal/getting-started/plugin-structure/#maintaining-the-plugin-event-loop",children:"Maintaining the plugin event loop"})," for additional information."]}),"\n",(0,i.jsx)(n.h2,{id:"initialization",children:"Initialization"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine plugin is initialized by calling ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_InitializeAsync",children:(0,i.jsx)(n.code,{children:"InitializeAsync"})}),", passing in a ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modioinitializeoptions",children:(0,i.jsx)(n.code,{children:"ModioInitializeOptions"})})," with the relevant information for the current session, and a callback containing a ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modioerrorcode",children:(0,i.jsx)(n.code,{children:"ModioErrorCode"})})," that will contain the result of the initialization on completion."]}),"\n",(0,i.jsxs)(n.p,{children:["Use ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#GetProjectInitializeOptionsForSessionId",children:(0,i.jsx)(n.code,{children:"GetProjectInitializeOptionsForSessionId"})})," to initialize the plugin using the information entered in the ",(0,i.jsx)(n.a,{href:"/unreal/installation-and-setup/#plugin-configuration",children:"mod.io project settings"}),". On Windows, ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#GetDefaultSessionIdWindows",children:(0,i.jsx)(n.code,{children:"GetDefaultSessionIdWindows"})})," can be used as the ",(0,i.jsx)(n.a,{href:"/unreal/getting-started/plugin-structure/#session-ids",children:"Session ID"}),". Some platforms also require additional parameters added to ",(0,i.jsx)(n.code,{children:"ModioInitializeOptions.ExtendedInitializationParameters"}),". Refer to the relevant ",(0,i.jsx)(n.a,{href:"/platforms/",children:"platform documentation"})," for more information."]}),"\n",(0,i.jsxs)(o.A,{groupId:"languages",children:[(0,i.jsx)(a.A,{value:"cpp",label:"C++",children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::Init()\n{\t\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->InitializeAsync(UModioSDKLibrary::GetProjectInitializeOptionsForSessionId(UModioSDKLibrary::GetDefaultSessionIdWindows()), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnInitCallback));\n }\n UE_LOG(LogModioGame, Log, TEXT("Initializing mod.io"));\n}\n\nvoid UModioManagerSubsystem::OnInitCallback(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("mod.io initialization complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n\nvoid UModioManagerSubsystem::Tick(float DeltaTime)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->RunPendingHandlers();\n }\n}\n'})})}),(0,i.jsxs)(a.A,{value:"blueprint",label:"Blueprint",children:[(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"run_pending_handlers",src:t(43711).A+"",width:"689",height:"312"})}),(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"initasync_getoptions",src:t(91169).A+"",width:"1327",height:"556"})})]})]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["The error-handling in this sample has been omitted. See our ",(0,i.jsx)(n.a,{href:"error-handling",children:(0,i.jsx)(n.strong,{children:"Error Handling quick-start guide"})})," for more information."]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.code,{children:"InitializeAsync"})," is a asynchronous function, therefore you ",(0,i.jsx)(n.em,{children:"must"})," wait for the callback for confirmation that the initialization is complete."]}),"\n"]})}),"\n",(0,i.jsx)(n.h2,{id:"shutdown",children:"Shutdown"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine plugin is shut down by calling ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_ShutdownAsync",children:(0,i.jsx)(n.code,{children:"ShutdownAsync"})})," in a similar fashion."]}),"\n",(0,i.jsxs)(o.A,{groupId:"languages",children:[(0,i.jsx)(a.A,{value:"cpp",label:"C++",children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::Shutdown()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->ShutdownAsync(FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnShutdownComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnShutdownComplete(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("Shutdown complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n'})})}),(0,i.jsx)(a.A,{value:"blueprint",label:"Blueprint",children:(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"shutdownasync_runhandlers",src:t(47771).A+"",width:"996",height:"662"})})})]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["You ",(0,i.jsx)(n.em,{children:"must"})," continue to call ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#RunPendingHandlers",children:(0,i.jsx)(n.strong,{children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})})})," while shutdown is in progress to allow intermediate handlers to complete."]}),"\n",(0,i.jsx)(n.li,{children:"Any in-flight mod.io async calls will complete with an error code indicating cancellation."}),"\n",(0,i.jsxs)(n.li,{children:["You ",(0,i.jsx)(n.em,{children:"must not"})," call ",(0,i.jsx)(n.strong,{children:(0,i.jsx)(n.code,{children:"ShutdownAsync"})})," from within another completion handler/callback. This may cause the application to deadlock."]}),"\n"]})})]})}function m(e={}){const{wrapper:n}={...(0,r.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},19365:(e,n,t)=>{t.d(n,{A:()=>a});t(96540);var i=t(18215);const r={tabItem:"tabItem_Ymn6"};var o=t(74848);function a(e){let{children:n,hidden:t,className:a}=e;return(0,o.jsx)("div",{role:"tabpanel",className:(0,i.A)(r.tabItem,a),hidden:t,children:n})}},11470:(e,n,t)=>{t.d(n,{A:()=>w});var i=t(96540),r=t(18215),o=t(23104),a=t(56347),s=t(205),l=t(57485),d=t(31682),u=t(70679);function c(e){return i.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,i.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:n,children:t}=e;return(0,i.useMemo)((()=>{const e=n??function(e){return c(e).map((e=>{let{props:{value:n,label:t,attributes:i,default:r}}=e;return{value:n,label:t,attributes:i,default:r}}))}(t);return function(e){const n=(0,d.X)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function m(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function g(e){let{queryString:n=!1,groupId:t}=e;const r=(0,a.W6)(),o=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,l.aZ)(o),(0,i.useCallback)((e=>{if(!o)return;const n=new URLSearchParams(r.location.search);n.set(o,e),r.replace({...r.location,search:n.toString()})}),[o,r])]}function p(e){const{defaultValue:n,queryString:t=!1,groupId:r}=e,o=h(e),[a,l]=(0,i.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!m({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const i=t.find((e=>e.default))??t[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:n,tabValues:o}))),[d,c]=g({queryString:t,groupId:r}),[p,f]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[r,o]=(0,u.Dv)(t);return[r,(0,i.useCallback)((e=>{t&&o.set(e)}),[t,o])]}({groupId:r}),b=(()=>{const e=d??p;return m({value:e,tabValues:o})?e:null})();(0,s.A)((()=>{b&&l(b)}),[b]);return{selectedValue:a,selectValue:(0,i.useCallback)((e=>{if(!m({value:e,tabValues:o}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),f(e)}),[c,f,o]),tabValues:o}}var f=t(92303);const b={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var x=t(74848);function j(e){let{className:n,block:t,selectedValue:i,selectValue:a,tabValues:s}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,o.a_)(),u=e=>{const n=e.currentTarget,t=l.indexOf(n),r=s[t].value;r!==i&&(d(n),a(r))},c=e=>{let n=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const t=l.indexOf(e.currentTarget)+1;n=l[t]??l[0];break}case"ArrowLeft":{const t=l.indexOf(e.currentTarget)-1;n=l[t]??l[l.length-1];break}}n?.focus()};return(0,x.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.A)("tabs",{"tabs--block":t},n),children:s.map((e=>{let{value:n,label:t,attributes:o}=e;return(0,x.jsx)("li",{role:"tab",tabIndex:i===n?0:-1,"aria-selected":i===n,ref:e=>l.push(e),onKeyDown:c,onClick:u,...o,className:(0,r.A)("tabs__item",b.tabItem,o?.className,{"tabs__item--active":i===n}),children:t??n},n)}))})}function y(e){let{lazy:n,children:t,selectedValue:r}=e;const o=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=o.find((e=>e.props.value===r));return e?(0,i.cloneElement)(e,{className:"margin-top--md"}):null}return(0,x.jsx)("div",{className:"margin-top--md",children:o.map(((e,n)=>(0,i.cloneElement)(e,{key:n,hidden:e.props.value!==r})))})}function v(e){const n=p(e);return(0,x.jsxs)("div",{className:(0,r.A)("tabs-container",b.tabList),children:[(0,x.jsx)(j,{...n,...e}),(0,x.jsx)(y,{...n,...e})]})}function w(e){const n=(0,f.A)();return(0,x.jsx)(v,{...e,children:c(e.children)},String(n))}},89236:(e,n,t)=>{t.d(n,{A:()=>o});var i=t(19365),r=(t(96540),t(74848));function o(e){return(0,r.jsx)(r.Fragment,{children:(0,r.jsx)(i.A,{className:"tw-rounded-md",...e})})}},27064:(e,n,t)=>{t.d(n,{A:()=>o});var i=t(11470),r=(t(96540),t(74848));function o(e){return(0,r.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,r.jsx)(i.A,{...e})})}},91169:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/initasync_getoptions-297e9d77d325d1bc7ab215940cd91c36.png"},43711:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/run_pending_handlers-3e2c3e1b54ee420c6d94eaf542368454.png"},47771:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/shutdownasync_runhandlers-c2e93330aca93ab1f70644e5fe625841.png"},28453:(e,n,t)=>{t.d(n,{R:()=>a,x:()=>s});var i=t(96540);const r={},o=i.createContext(r);function a(e){const n=i.useContext(o);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function s(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:a(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/9e4dfacd.0af8d679.js b/Doc/assets/js/9e4dfacd.0af8d679.js deleted file mode 100644 index 98d9bc05..00000000 --- a/Doc/assets/js/9e4dfacd.0af8d679.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[6647],{38163:(t,e,n)=>{n.r(e),n.d(e,{assets:()=>l,contentTitle:()=>o,default:()=>c,frontMatter:()=>s,metadata:()=>d,toc:()=>a});var i=n(74848),r=n(28453);const s={id:"ue-getting-started",title:"Getting Started",slug:"/unreal/getting-started/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/getting-started.mdx"},o=void 0,d={id:"getting-started/ue-getting-started",title:"Getting Started",description:"Below are a series of quick-start guides with code samples demonstrating the mod.io Unreal Engine plugin's core functionality.",source:"@site/public/en-us/getting-started/getting-started.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/",permalink:"/unreal/getting-started/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/getting-started.mdx",tags:[],version:"current",frontMatter:{id:"ue-getting-started",title:"Getting Started",slug:"/unreal/getting-started/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/getting-started.mdx"},sidebar:"sidebar",previous:{title:"Installation & Setup",permalink:"/unreal/installation-and-setup/"},next:{title:"Plugin Structure & Concepts",permalink:"/unreal/getting-started/plugin-structure/"}},l={},a=[{value:"Quick-Start Guides",id:"quick-start-guides",level:3}];function u(t){const e={a:"a",h3:"h3",li:"li",p:"p",ul:"ul",...(0,r.R)(),...t.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(e.p,{children:"Below are a series of quick-start guides with code samples demonstrating the mod.io Unreal Engine plugin's core functionality."}),"\n",(0,i.jsx)(e.h3,{id:"quick-start-guides",children:"Quick-Start Guides"}),"\n",(0,i.jsxs)(e.ul,{children:["\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"plugin-structure",children:"Plugin Structure & Concepts"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"initialization",children:"Initialization & Teardown"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"user-authentication",children:"User Authentication"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"browsing-mods",children:"Browsing Mods"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"mod-subscriptions",children:"Mod Subscriptions & Management"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"error-handling",children:"Error Handling"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"submit-mods",children:"In-Game Mod Submission"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"edit-mods",children:"Edit an Existing Mod"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"monetization",children:"Monetization"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"temporary-mods",children:"Temporary Mod Sets"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"mute-user",children:"Muting & Unmuting a User"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"metrics-play-sessions",children:"Metrics Play Sessions"})}),"\n"]})]})}function c(t={}){const{wrapper:e}={...(0,r.R)(),...t.components};return e?(0,i.jsx)(e,{...t,children:(0,i.jsx)(u,{...t})}):u(t)}},28453:(t,e,n)=>{n.d(e,{R:()=>o,x:()=>d});var i=n(96540);const r={},s=i.createContext(r);function o(t){const e=i.useContext(s);return i.useMemo((function(){return"function"==typeof t?t(e):{...e,...t}}),[e,t])}function d(t){let e;return e=t.disableParentContext?"function"==typeof t.components?t.components(r):t.components||r:o(t.components),i.createElement(s.Provider,{value:e},t.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/9e4dfacd.74a35171.js b/Doc/assets/js/9e4dfacd.74a35171.js new file mode 100644 index 00000000..4d70272a --- /dev/null +++ b/Doc/assets/js/9e4dfacd.74a35171.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[6647],{38163:(t,e,n)=>{n.r(e),n.d(e,{assets:()=>l,contentTitle:()=>o,default:()=>c,frontMatter:()=>s,metadata:()=>d,toc:()=>a});var i=n(74848),r=n(28453);const s={id:"ue-getting-started",title:"Getting Started",slug:"/unreal/getting-started/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/getting-started.mdx"},o=void 0,d={id:"getting-started/ue-getting-started",title:"Getting Started",description:"Below are a series of quick-start guides with code samples demonstrating the mod.io Unreal Engine plugin's core functionality.",source:"@site/public/en-us/getting-started/getting-started.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/",permalink:"/unreal/getting-started/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/getting-started.mdx",tags:[],version:"current",frontMatter:{id:"ue-getting-started",title:"Getting Started",slug:"/unreal/getting-started/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/getting-started.mdx"},sidebar:"sidebar",previous:{title:"Installation & Setup",permalink:"/unreal/installation-and-setup/"},next:{title:"Plugin Structure & Concepts",permalink:"/unreal/getting-started/plugin-structure/"}},l={},a=[{value:"Quick-Start Guides",id:"quick-start-guides",level:3}];function u(t){const e={a:"a",h3:"h3",li:"li",p:"p",ul:"ul",...(0,r.R)(),...t.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(e.p,{children:"Below are a series of quick-start guides with code samples demonstrating the mod.io Unreal Engine plugin's core functionality."}),"\n",(0,i.jsx)(e.h3,{id:"quick-start-guides",children:"Quick-Start Guides"}),"\n",(0,i.jsxs)(e.ul,{children:["\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"plugin-structure",children:"Plugin Structure & Concepts"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"initialization",children:"Initialization & Teardown"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"user-authentication",children:"User Authentication"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"browsing-mods",children:"Browsing Mods"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"mod-subscriptions",children:"Mod Subscriptions & Management"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"error-handling",children:"Error Handling"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"submit-mods",children:"In-Game Mod Submission"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"edit-mods",children:"Edit an Existing Mod"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"monetization",children:"Monetization"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"temporary-mods",children:"Temporary Mod Sets"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"mute-user",children:"Muting & Unmuting a User"})}),"\n",(0,i.jsx)(e.li,{children:(0,i.jsx)(e.a,{href:"metrics-play-sessions",children:"Metrics Play Sessions"})}),"\n"]})]})}function c(t={}){const{wrapper:e}={...(0,r.R)(),...t.components};return e?(0,i.jsx)(e,{...t,children:(0,i.jsx)(u,{...t})}):u(t)}},28453:(t,e,n)=>{n.d(e,{R:()=>o,x:()=>d});var i=n(96540);const r={},s=i.createContext(r);function o(t){const e=i.useContext(s);return i.useMemo((function(){return"function"==typeof t?t(e):{...e,...t}}),[e,t])}function d(t){let e;return e=t.disableParentContext?"function"==typeof t.components?t.components(r):t.components||r:o(t.components),i.createElement(s.Provider,{value:e},t.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/b2348a3c.03af1dd0.js b/Doc/assets/js/b2348a3c.03af1dd0.js new file mode 100644 index 00000000..a827822f --- /dev/null +++ b/Doc/assets/js/b2348a3c.03af1dd0.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[5633],{54082:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>l,default:()=>p,frontMatter:()=>i,metadata:()=>d,toc:()=>c});var o=n(74848),r=n(28453),a=n(27064),s=n(89236);const i={id:"ue-temporary-mods",title:"Temporary Mod Sets",slug:"/unreal/getting-started/temporary-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/temporary-mods.mdx"},l=void 0,d={id:"getting-started/ue-temporary-mods",title:"Temporary Mod Sets",description:"In some situations, you may want mods to only exist on a temporary basis. For instance, in multiplayer environments where you don't want to subscribe a user to a piece of content. Temporary mod sets allow management of these transient pieces of content separately from subscriptions. Temp mod sets do not require authentication, however mod management must still be enabled.",source:"@site/public/en-us/getting-started/temporary-mods.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/temporary-mods",permalink:"/unreal/getting-started/temporary-mods",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/temporary-mods.mdx",tags:[],version:"current",frontMatter:{id:"ue-temporary-mods",title:"Temporary Mod Sets",slug:"/unreal/getting-started/temporary-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/temporary-mods.mdx"},sidebar:"sidebar",previous:{title:"Monetization",permalink:"/unreal/getting-started/monetization"},next:{title:"Muting and Unmuting a User",permalink:"/unreal/getting-started/mute-user"}},u={},c=[{value:"Installing Temporary Mods",id:"installing-temporary-mods",level:3}];function m(e){const t={a:"a",admonition:"admonition",code:"code",h3:"h3",img:"img",mdxAdmonitionTitle:"mdxAdmonitionTitle",p:"p",pre:"pre",...(0,r.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(t.p,{children:"In some situations, you may want mods to only exist on a temporary basis. For instance, in multiplayer environments where you don't want to subscribe a user to a piece of content. Temporary mod sets allow management of these transient pieces of content separately from subscriptions. Temp mod sets do not require authentication, however mod management must still be enabled."}),"\n",(0,o.jsxs)(t.p,{children:["Temp mods are downloaded to a separate folder from subscriptions, and are not updated or handled when you call ",(0,o.jsx)(t.code,{children:"FetchExternalUpdatesAsync"}),". That means you can prioritize download and installation of temp mods outside of the regular subscription flow."]}),"\n",(0,o.jsxs)(t.p,{children:["To use temp mods, you can start a ",(0,o.jsx)(t.code,{children:"TempModSet"})," by calling ",(0,o.jsx)(t.code,{children:"InitTempModSet"})," and passing a list of ",(0,o.jsx)(t.code,{children:"ModioModIDs"})," to be downloaded and extracted. At any time while a ",(0,o.jsx)(t.code,{children:"TempModSet"})," is open, you can call ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#addtotempmodset",children:(0,o.jsx)(t.code,{children:"AddToTempModSet"})})," to add mods to the set. These will be instantly downloaded and extracted. If you no longer need a mod, you can call ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#removefromtempmodset",children:(0,o.jsx)(t.code,{children:"RemoveFromTempModSet"})})," which will remove the file. Once you have finished with a ",(0,o.jsx)(t.code,{children:"TempModSet"}),", call ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#closetempmodset",children:(0,o.jsx)(t.code,{children:"CloseTempModSet"})})," which will delete all temporary mods. Temporary mods are also deleted the next time you re-initialize the mod.io plugin."]}),"\n",(0,o.jsxs)(t.p,{children:["Like regular mods, temp mods can be queried using ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#querytempmodset",children:(0,o.jsx)(t.code,{children:"QueryTempModSet"})})," to get a ",(0,o.jsx)(t.code,{children:"ModioModCollectionEntry"})," with an installation path."]}),"\n",(0,o.jsx)(t.h3,{id:"installing-temporary-mods",children:"Installing Temporary Mods"}),"\n",(0,o.jsxs)(a.A,{"group-id":"languages",children:[(0,o.jsx)(s.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(t.p,{children:(0,o.jsx)(t.img,{alt:"temp_mod_init",src:n(22177).A+"",width:"998",height:"240"})})}),(0,o.jsx)(s.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-cpp",children:'Modio::EnableModManagement([](Modio::ModManagementEvent ModEvent)\n{\n if (ModEvent.Status && ModEvent.Event == Modio::ModManagementEvent::EventType::Installed)\n {\n std::cout << "Mod with ID: " << ModEvent.ID << " is installed" << std::endl;\n }\n else \n {\n std::cout << "Mod with ID: " << ModEvent.ID << " failed to install: " << ModEvent.Status.message() << std::endl;\n }\n});\n\nstd::vector ModIds = {8, 4, 5};\n\nModio::InitTempModSet(ModIds);\n\nwhile(Modio::IsModManagementBusy())\n{\n Modio::RunPendingHandlers();\n}\n'})})})]}),"\n",(0,o.jsxs)(t.p,{children:["This call will start a ",(0,o.jsx)(t.code,{children:"TempModSet"})," and install mods with IDs 8, 4 and 5."]}),"\n",(0,o.jsxs)(t.admonition,{type:"note",children:[(0,o.jsx)(t.mdxAdmonitionTitle,{}),(0,o.jsxs)(t.p,{children:["If you add a mod to ",(0,o.jsx)(t.code,{children:"TempModSet"})," that is already subscribed, it will not be downloaded; the player will already have that content. If you try to unsubscribe from it while it's in ",(0,o.jsx)(t.code,{children:"TempModSet"}),", the plugin will wait for it to be removed from ",(0,o.jsx)(t.code,{children:"TempModSet"})," before processing the unsubscribe."]})]})]})}function p(e={}){const{wrapper:t}={...(0,r.R)(),...e.components};return t?(0,o.jsx)(t,{...e,children:(0,o.jsx)(m,{...e})}):m(e)}},19365:(e,t,n)=>{n.d(t,{A:()=>s});n(96540);var o=n(18215);const r={tabItem:"tabItem_Ymn6"};var a=n(74848);function s(e){let{children:t,hidden:n,className:s}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,o.A)(r.tabItem,s),hidden:n,children:t})}},11470:(e,t,n)=>{n.d(t,{A:()=>M});var o=n(96540),r=n(18215),a=n(23104),s=n(56347),i=n(205),l=n(57485),d=n(31682),u=n(70679);function c(e){return o.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:n}=e;return(0,o.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:n,attributes:o,default:r}}=e;return{value:t,label:n,attributes:o,default:r}}))}(n);return function(e){const t=(0,d.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function p(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:n}=e;const r=(0,s.W6)(),a=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l.aZ)(a),(0,o.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(r.location.search);t.set(a,e),r.replace({...r.location,search:t.toString()})}),[a,r])]}function b(e){const{defaultValue:t,queryString:n=!1,groupId:r}=e,a=m(e),[s,l]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!p({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const o=n.find((e=>e.default))??n[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:t,tabValues:a}))),[d,c]=h({queryString:n,groupId:r}),[b,f]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[r,a]=(0,u.Dv)(n);return[r,(0,o.useCallback)((e=>{n&&a.set(e)}),[n,a])]}({groupId:r}),g=(()=>{const e=d??b;return p({value:e,tabValues:a})?e:null})();(0,i.A)((()=>{g&&l(g)}),[g]);return{selectedValue:s,selectValue:(0,o.useCallback)((e=>{if(!p({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),f(e)}),[c,f,a]),tabValues:a}}var f=n(92303);const g={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var y=n(74848);function v(e){let{className:t,block:n,selectedValue:o,selectValue:s,tabValues:i}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,a.a_)(),u=e=>{const t=e.currentTarget,n=l.indexOf(t),r=i[n].value;r!==o&&(d(t),s(r))},c=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const n=l.indexOf(e.currentTarget)+1;t=l[n]??l[0];break}case"ArrowLeft":{const n=l.indexOf(e.currentTarget)-1;t=l[n]??l[l.length-1];break}}t?.focus()};return(0,y.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.A)("tabs",{"tabs--block":n},t),children:i.map((e=>{let{value:t,label:n,attributes:a}=e;return(0,y.jsx)("li",{role:"tab",tabIndex:o===t?0:-1,"aria-selected":o===t,ref:e=>l.push(e),onKeyDown:c,onClick:u,...a,className:(0,r.A)("tabs__item",g.tabItem,a?.className,{"tabs__item--active":o===t}),children:n??t},t)}))})}function x(e){let{lazy:t,children:n,selectedValue:r}=e;const a=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===r));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return(0,y.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==r})))})}function w(e){const t=b(e);return(0,y.jsxs)("div",{className:(0,r.A)("tabs-container",g.tabList),children:[(0,y.jsx)(v,{...t,...e}),(0,y.jsx)(x,{...t,...e})]})}function M(e){const t=(0,f.A)();return(0,y.jsx)(w,{...e,children:c(e.children)},String(t))}},89236:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(19365),r=(n(96540),n(74848));function a(e){return(0,r.jsx)(r.Fragment,{children:(0,r.jsx)(o.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(11470),r=(n(96540),n(74848));function a(e){return(0,r.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,r.jsx)(o.A,{...e})})}},22177:(e,t,n)=>{n.d(t,{A:()=>o});const o=n.p+"assets/images/temp_mod_init-d5d5b4131ec5e8ed84266b2aedf97c03.png"},28453:(e,t,n)=>{n.d(t,{R:()=>s,x:()=>i});var o=n(96540);const r={},a=o.createContext(r);function s(e){const t=o.useContext(a);return o.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function i(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:s(e.components),o.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/b2348a3c.612bac9c.js b/Doc/assets/js/b2348a3c.612bac9c.js deleted file mode 100644 index 6e73fa31..00000000 --- a/Doc/assets/js/b2348a3c.612bac9c.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[5633],{54082:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>l,default:()=>p,frontMatter:()=>i,metadata:()=>d,toc:()=>c});var o=n(74848),r=n(28453),a=n(27064),s=n(89236);const i={id:"ue-temporary-mods",title:"Temporary Mod Sets",slug:"/unreal/getting-started/temporary-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/temporary-mods.mdx"},l=void 0,d={id:"getting-started/ue-temporary-mods",title:"Temporary Mod Sets",description:"In some situations, you may want mods to only exist on a temporary basis. For instance, in multiplayer environments where you don't want to subscribe a user to a piece of content. Temporary mod sets allow management of these transient pieces of content separately from subscriptions. Temp mod sets do not require authentication, however mod management must still be enabled.",source:"@site/public/en-us/getting-started/temporary-mods.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/temporary-mods",permalink:"/unreal/getting-started/temporary-mods",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/temporary-mods.mdx",tags:[],version:"current",frontMatter:{id:"ue-temporary-mods",title:"Temporary Mod Sets",slug:"/unreal/getting-started/temporary-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/temporary-mods.mdx"},sidebar:"sidebar",previous:{title:"Monetization",permalink:"/unreal/getting-started/monetization"},next:{title:"Muting and Unmuting a User",permalink:"/unreal/getting-started/mute-user"}},u={},c=[{value:"Installing Temporary Mods",id:"installing-temporary-mods",level:3}];function m(e){const t={a:"a",admonition:"admonition",code:"code",h3:"h3",img:"img",mdxAdmonitionTitle:"mdxAdmonitionTitle",p:"p",pre:"pre",...(0,r.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(t.p,{children:"In some situations, you may want mods to only exist on a temporary basis. For instance, in multiplayer environments where you don't want to subscribe a user to a piece of content. Temporary mod sets allow management of these transient pieces of content separately from subscriptions. Temp mod sets do not require authentication, however mod management must still be enabled."}),"\n",(0,o.jsxs)(t.p,{children:["Temp mods are downloaded to a separate folder from subscriptions, and are not updated or handled when you call ",(0,o.jsx)(t.code,{children:"FetchExternalUpdatesAsync"}),". That means you can prioritize download and installation of temp mods outside of the regular subscription flow."]}),"\n",(0,o.jsxs)(t.p,{children:["To use temp mods, you can start a ",(0,o.jsx)(t.code,{children:"TempModSet"})," by calling ",(0,o.jsx)(t.code,{children:"InitTempModSet"})," and passing a list of ",(0,o.jsx)(t.code,{children:"ModioModIDs"})," to be downloaded and extracted. At any time while a ",(0,o.jsx)(t.code,{children:"TempModSet"})," is open, you can call ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_AddToTempModSet",children:(0,o.jsx)(t.code,{children:"AddToTempModSet"})})," to add mods to the set. These will be instantly downloaded and extracted. If you no longer need a mod, you can call ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_RemoveFromTempModSet",children:(0,o.jsx)(t.code,{children:"RemoveFromTempModSet"})})," which will remove the file. Once you have finished with a ",(0,o.jsx)(t.code,{children:"TempModSet"}),", call ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_CloseTempModSet",children:(0,o.jsx)(t.code,{children:"CloseTempModSet"})})," which will delete all temporary mods. Temporary mods are also deleted the next time you re-initialize the mod.io plugin."]}),"\n",(0,o.jsxs)(t.p,{children:["Like regular mods, temp mods can be queried using ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_QueryTempModSet",children:(0,o.jsx)(t.code,{children:"QueryTempModSet"})})," to get a ",(0,o.jsx)(t.code,{children:"ModioModCollectionEntry"})," with an installation path."]}),"\n",(0,o.jsx)(t.h3,{id:"installing-temporary-mods",children:"Installing Temporary Mods"}),"\n",(0,o.jsxs)(a.A,{"group-id":"languages",children:[(0,o.jsx)(s.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(t.p,{children:(0,o.jsx)(t.img,{alt:"temp_mod_init",src:n(39660).A+"",width:"998",height:"240"})})}),(0,o.jsx)(s.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-cpp",children:'Modio::EnableModManagement([](Modio::ModManagementEvent ModEvent)\n{\n if (ModEvent.Status && ModEvent.Event == Modio::ModManagementEvent::EventType::Installed)\n {\n std::cout << "Mod with ID: " << ModEvent.ID << " is installed" << std::endl;\n }\n else \n {\n std::cout << "Mod with ID: " << ModEvent.ID << " failed to install: " << ModEvent.Status.message() << std::endl;\n }\n});\n\nstd::vector ModIds = {8, 4, 5};\n\nModio::InitTempModSet(ModIds);\n\nwhile(Modio::IsModManagementBusy())\n{\n Modio::RunPendingHandlers();\n}\n'})})})]}),"\n",(0,o.jsxs)(t.p,{children:["This call will start a ",(0,o.jsx)(t.code,{children:"TempModSet"})," and install mods with IDs 8, 4 and 5."]}),"\n",(0,o.jsxs)(t.admonition,{type:"note",children:[(0,o.jsx)(t.mdxAdmonitionTitle,{}),(0,o.jsxs)(t.p,{children:["If you add a mod to ",(0,o.jsx)(t.code,{children:"TempModSet"})," that is already subscribed, it will not be downloaded; the player will already have that content. If you try to unsubscribe from it while it's in ",(0,o.jsx)(t.code,{children:"TempModSet"}),", the plugin will wait for it to be removed from ",(0,o.jsx)(t.code,{children:"TempModSet"})," before processing the unsubscribe."]})]})]})}function p(e={}){const{wrapper:t}={...(0,r.R)(),...e.components};return t?(0,o.jsx)(t,{...e,children:(0,o.jsx)(m,{...e})}):m(e)}},19365:(e,t,n)=>{n.d(t,{A:()=>s});n(96540);var o=n(18215);const r={tabItem:"tabItem_Ymn6"};var a=n(74848);function s(e){let{children:t,hidden:n,className:s}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,o.A)(r.tabItem,s),hidden:n,children:t})}},11470:(e,t,n)=>{n.d(t,{A:()=>M});var o=n(96540),r=n(18215),a=n(23104),s=n(56347),i=n(205),l=n(57485),d=n(31682),u=n(70679);function c(e){return o.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:n}=e;return(0,o.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:n,attributes:o,default:r}}=e;return{value:t,label:n,attributes:o,default:r}}))}(n);return function(e){const t=(0,d.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function p(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function h(e){let{queryString:t=!1,groupId:n}=e;const r=(0,s.W6)(),a=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l.aZ)(a),(0,o.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(r.location.search);t.set(a,e),r.replace({...r.location,search:t.toString()})}),[a,r])]}function b(e){const{defaultValue:t,queryString:n=!1,groupId:r}=e,a=m(e),[s,l]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!p({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const o=n.find((e=>e.default))??n[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:t,tabValues:a}))),[d,c]=h({queryString:n,groupId:r}),[b,f]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[r,a]=(0,u.Dv)(n);return[r,(0,o.useCallback)((e=>{n&&a.set(e)}),[n,a])]}({groupId:r}),g=(()=>{const e=d??b;return p({value:e,tabValues:a})?e:null})();(0,i.A)((()=>{g&&l(g)}),[g]);return{selectedValue:s,selectValue:(0,o.useCallback)((e=>{if(!p({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),c(e),f(e)}),[c,f,a]),tabValues:a}}var f=n(92303);const g={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var y=n(74848);function v(e){let{className:t,block:n,selectedValue:o,selectValue:s,tabValues:i}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,a.a_)(),u=e=>{const t=e.currentTarget,n=l.indexOf(t),r=i[n].value;r!==o&&(d(t),s(r))},c=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const n=l.indexOf(e.currentTarget)+1;t=l[n]??l[0];break}case"ArrowLeft":{const n=l.indexOf(e.currentTarget)-1;t=l[n]??l[l.length-1];break}}t?.focus()};return(0,y.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.A)("tabs",{"tabs--block":n},t),children:i.map((e=>{let{value:t,label:n,attributes:a}=e;return(0,y.jsx)("li",{role:"tab",tabIndex:o===t?0:-1,"aria-selected":o===t,ref:e=>l.push(e),onKeyDown:c,onClick:u,...a,className:(0,r.A)("tabs__item",g.tabItem,a?.className,{"tabs__item--active":o===t}),children:n??t},t)}))})}function x(e){let{lazy:t,children:n,selectedValue:r}=e;const a=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===r));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return(0,y.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==r})))})}function w(e){const t=b(e);return(0,y.jsxs)("div",{className:(0,r.A)("tabs-container",g.tabList),children:[(0,y.jsx)(v,{...t,...e}),(0,y.jsx)(x,{...t,...e})]})}function M(e){const t=(0,f.A)();return(0,y.jsx)(w,{...e,children:c(e.children)},String(t))}},89236:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(19365),r=(n(96540),n(74848));function a(e){return(0,r.jsx)(r.Fragment,{children:(0,r.jsx)(o.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(11470),r=(n(96540),n(74848));function a(e){return(0,r.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,r.jsx)(o.A,{...e})})}},39660:(e,t,n)=>{n.d(t,{A:()=>o});const o=n.p+"assets/images/temp_mod_init-d5d5b4131ec5e8ed84266b2aedf97c03.png"},28453:(e,t,n)=>{n.d(t,{R:()=>s,x:()=>i});var o=n(96540);const r={},a=o.createContext(r);function s(e){const t=o.useContext(a);return o.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function i(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:s(e.components),o.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/b69370a5.93b89783.js b/Doc/assets/js/b69370a5.93b89783.js deleted file mode 100644 index 8597f58c..00000000 --- a/Doc/assets/js/b69370a5.93b89783.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[7288],{36967:(e,r,t)=>{t.r(r),t.d(r,{assets:()=>c,contentTitle:()=>l,default:()=>p,frontMatter:()=>s,metadata:()=>d,toc:()=>u});var n=t(74848),o=t(28453),a=t(27064),i=t(89236);const s={id:"ue-error-handling",title:"Error Handling",slug:"/unreal/getting-started/error-handling",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/error-handling.mdx"},l=void 0,d={id:"getting-started/ue-error-handling",title:"Error Handling",description:"Error handling in Blueprint is still undergoing improvement. Please let us know if you have any feedback or suggestions for improving it!",source:"@site/public/en-us/getting-started/error-handling.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/error-handling",permalink:"/unreal/getting-started/error-handling",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/error-handling.mdx",tags:[],version:"current",frontMatter:{id:"ue-error-handling",title:"Error Handling",slug:"/unreal/getting-started/error-handling",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/error-handling.mdx"},sidebar:"sidebar",previous:{title:"Edit an Existing Mod",permalink:"/unreal/getting-started/edit-mods"},next:{title:"Monetization",permalink:"/unreal/getting-started/monetization"}},c={},u=[{value:"Checking for errors",id:"checking-for-errors",level:3},{value:"Inspecting ModioErrorCode more deeply",id:"inspecting-modioerrorcode-more-deeply",level:3},{value:"Semantic queries",id:"semantic-queries",level:4},{value:"Parameter validation errors",id:"parameter-validation-errors",level:3}];function h(e){const r={a:"a",admonition:"admonition",code:"code",h3:"h3",h4:"h4",img:"img",p:"p",pre:"pre",...(0,o.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsx)(r.admonition,{type:"note",children:(0,n.jsx)(r.p,{children:"Error handling in Blueprint is still undergoing improvement. Please let us know if you have any feedback or suggestions for improving it!"})}),"\n",(0,n.jsxs)(r.p,{children:["Many of the plugin's functions utilize the ",(0,n.jsx)(r.a,{href:"https://docs.mod.io/unrealref/#_modioerrorcode",children:(0,n.jsx)(r.code,{children:"ModioErrorCode"})})," type. For example, all ",(0,n.jsx)(r.a,{href:"/unreal/getting-started/plugin-structure/#non-blocking-asynchronous-interface",children:"async"})," functions take a ",(0,n.jsx)(r.code,{children:"ModioErrorCode"})," as the first parameter for the function's delegate. This is essentially an opaque wrapper around a numeric error code with a category and an associated string message."]}),"\n",(0,n.jsx)(r.h3,{id:"checking-for-errors",children:"Checking for errors"}),"\n",(0,n.jsxs)(r.p,{children:['A "truthy" error code represents an error. In Blueprint, you can check if a ',(0,n.jsx)(r.code,{children:"ModioErrorCode"})," represents a success or failure by using the ",(0,n.jsx)(r.code,{children:"IsError"})," node."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(i.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(r.p,{children:(0,n.jsx)(r.img,{alt:"is_error",src:t(26709).A+"",width:"470",height:"118"})})}),(0,n.jsx)(i.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(r.pre,{children:(0,n.jsx)(r.code,{className:"language-cpp",children:"void UModioManagerSubsystem::MyCallback(FModioErrorCode ErrorCode)\n{\n if (ErrorCode)\n {\n // Failure\n }\n else\n {\n // Success\n }\n}\n"})})})]}),"\n",(0,n.jsxs)(r.h3,{id:"inspecting-modioerrorcode-more-deeply",children:["Inspecting ",(0,n.jsx)(r.code,{children:"ModioErrorCode"})," more deeply"]}),"\n",(0,n.jsx)(r.p,{children:"Sometimes this information will be all that is required; just a simple 'success/fail' that you can handle."}),"\n",(0,n.jsxs)(r.p,{children:["In many cases, however, you will want to perform some degree of inspection on an ",(0,n.jsx)(r.code,{children:"ModioErrorCode"})," to determine specific information about that error - if nothing else, so that you can display a reason for the failure to the end user."]}),"\n",(0,n.jsx)(r.h4,{id:"semantic-queries",children:"Semantic queries"}),"\n",(0,n.jsx)(r.p,{children:'In your application, there are many errors that will likely be handled the same way. For example, you probably don\u2019t need to handle different network errors in different ways. The semantics of networking errors are largely "try the function again later".'}),"\n",(0,n.jsxs)(r.p,{children:["This is where ",(0,n.jsx)(r.a,{href:"https://docs.mod.io/unrealref/#ErrorCodeMatches",children:(0,n.jsx)(r.code,{children:"ErrorCodeMatches"})}),' comes in. It allows you to query if the error satisfies a particular condition, such as "does this code represent some kind of networking error", without needing to explicitly check the code against all the individual errors in the category.']}),"\n",(0,n.jsx)(r.p,{children:(0,n.jsx)(r.img,{alt:"error_code_matches_network_error",src:t(96338).A+"",width:"633",height:"228"})}),"\n",(0,n.jsx)(r.p,{children:"By querying if an error meets a specific condition, you can focus on handling a family of errors (in this case, network transmission errors) without needing to deal with individual errors within that group. No more manually checking against individual HttpError values, just a single query."}),"\n",(0,n.jsx)(r.p,{children:"These semantic checks help you to consolidate your error handling into limited set of generic error handlers rather than dealing with each potential outcome individually."}),"\n",(0,n.jsx)(r.p,{children:(0,n.jsx)(r.img,{alt:"error_code_matches_chaining",src:t(80710).A+"",width:"1231",height:"507"})}),"\n",(0,n.jsx)(r.h3,{id:"parameter-validation-errors",children:"Parameter validation errors"}),"\n",(0,n.jsxs)(r.p,{children:["Some of the plugin functions may return errors that indicate a parameter or data validation failure. For these cases, the plugin parses the error response from the mod.io API and stores the information about which parameters failed validation until the next network request is performed. If a plugin function returns an error which matches ",(0,n.jsx)(r.code,{children:"EModioErrorCondition::InvalidArgsError"}),", you can call ",(0,n.jsx)(r.code,{children:"GetLastValidationError"})," in your callback to retrieve those errors and display appropriate feedback to the end user."]}),"\n",(0,n.jsx)(r.p,{children:(0,n.jsx)(r.img,{alt:"get_last_validation_error",src:t(97284).A+"",width:"1340",height:"247"})})]})}function p(e={}){const{wrapper:r}={...(0,o.R)(),...e.components};return r?(0,n.jsx)(r,{...e,children:(0,n.jsx)(h,{...e})}):h(e)}},19365:(e,r,t)=>{t.d(r,{A:()=>i});t(96540);var n=t(18215);const o={tabItem:"tabItem_Ymn6"};var a=t(74848);function i(e){let{children:r,hidden:t,className:i}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,n.A)(o.tabItem,i),hidden:t,children:r})}},11470:(e,r,t)=>{t.d(r,{A:()=>j});var n=t(96540),o=t(18215),a=t(23104),i=t(56347),s=t(205),l=t(57485),d=t(31682),c=t(70679);function u(e){return n.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,n.isValidElement)(e)&&function(e){const{props:r}=e;return!!r&&"object"==typeof r&&"value"in r}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:r,children:t}=e;return(0,n.useMemo)((()=>{const e=r??function(e){return u(e).map((e=>{let{props:{value:r,label:t,attributes:n,default:o}}=e;return{value:r,label:t,attributes:n,default:o}}))}(t);return function(e){const r=(0,d.X)(e,((e,r)=>e.value===r.value));if(r.length>0)throw new Error(`Docusaurus error: Duplicate values "${r.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[r,t])}function p(e){let{value:r,tabValues:t}=e;return t.some((e=>e.value===r))}function m(e){let{queryString:r=!1,groupId:t}=e;const o=(0,i.W6)(),a=function(e){let{queryString:r=!1,groupId:t}=e;if("string"==typeof r)return r;if(!1===r)return null;if(!0===r&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:r,groupId:t});return[(0,l.aZ)(a),(0,n.useCallback)((e=>{if(!a)return;const r=new URLSearchParams(o.location.search);r.set(a,e),o.replace({...o.location,search:r.toString()})}),[a,o])]}function g(e){const{defaultValue:r,queryString:t=!1,groupId:o}=e,a=h(e),[i,l]=(0,n.useState)((()=>function(e){let{defaultValue:r,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(r){if(!p({value:r,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${r}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return r}const n=t.find((e=>e.default))??t[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:r,tabValues:a}))),[d,u]=m({queryString:t,groupId:o}),[g,f]=function(e){let{groupId:r}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(r),[o,a]=(0,c.Dv)(t);return[o,(0,n.useCallback)((e=>{t&&a.set(e)}),[t,a])]}({groupId:o}),b=(()=>{const e=d??g;return p({value:e,tabValues:a})?e:null})();(0,s.A)((()=>{b&&l(b)}),[b]);return{selectedValue:i,selectValue:(0,n.useCallback)((e=>{if(!p({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),u(e),f(e)}),[u,f,a]),tabValues:a}}var f=t(92303);const b={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var v=t(74848);function y(e){let{className:r,block:t,selectedValue:n,selectValue:i,tabValues:s}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,a.a_)(),c=e=>{const r=e.currentTarget,t=l.indexOf(r),o=s[t].value;o!==n&&(d(r),i(o))},u=e=>{let r=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const t=l.indexOf(e.currentTarget)+1;r=l[t]??l[0];break}case"ArrowLeft":{const t=l.indexOf(e.currentTarget)-1;r=l[t]??l[l.length-1];break}}r?.focus()};return(0,v.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.A)("tabs",{"tabs--block":t},r),children:s.map((e=>{let{value:r,label:t,attributes:a}=e;return(0,v.jsx)("li",{role:"tab",tabIndex:n===r?0:-1,"aria-selected":n===r,ref:e=>l.push(e),onKeyDown:u,onClick:c,...a,className:(0,o.A)("tabs__item",b.tabItem,a?.className,{"tabs__item--active":n===r}),children:t??r},r)}))})}function x(e){let{lazy:r,children:t,selectedValue:o}=e;const a=(Array.isArray(t)?t:[t]).filter(Boolean);if(r){const e=a.find((e=>e.props.value===o));return e?(0,n.cloneElement)(e,{className:"margin-top--md"}):null}return(0,v.jsx)("div",{className:"margin-top--md",children:a.map(((e,r)=>(0,n.cloneElement)(e,{key:r,hidden:e.props.value!==o})))})}function w(e){const r=g(e);return(0,v.jsxs)("div",{className:(0,o.A)("tabs-container",b.tabList),children:[(0,v.jsx)(y,{...r,...e}),(0,v.jsx)(x,{...r,...e})]})}function j(e){const r=(0,f.A)();return(0,v.jsx)(w,{...e,children:u(e.children)},String(r))}},89236:(e,r,t)=>{t.d(r,{A:()=>a});var n=t(19365),o=(t(96540),t(74848));function a(e){return(0,o.jsx)(o.Fragment,{children:(0,o.jsx)(n.A,{className:"tw-rounded-md",...e})})}},27064:(e,r,t)=>{t.d(r,{A:()=>a});var n=t(11470),o=(t(96540),t(74848));function a(e){return(0,o.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,o.jsx)(n.A,{...e})})}},80710:(e,r,t)=>{t.d(r,{A:()=>n});const n=t.p+"assets/images/error_code_matches_chaining-86eec7eaebbbe82bf284e055ebd192d1.png"},96338:(e,r,t)=>{t.d(r,{A:()=>n});const n=t.p+"assets/images/error_code_matches_network_error-7ff80a95b75b1cf050da273595c027c5.png"},97284:(e,r,t)=>{t.d(r,{A:()=>n});const n=t.p+"assets/images/get_last_validation_error-bc856beda3feb292b5056037c771719a.png"},26709:(e,r,t)=>{t.d(r,{A:()=>n});const n=t.p+"assets/images/is_error-d9d2d4f7512b0e2c1bc6de68fbb1c255.png"},28453:(e,r,t)=>{t.d(r,{R:()=>i,x:()=>s});var n=t(96540);const o={},a=n.createContext(o);function i(e){const r=n.useContext(a);return n.useMemo((function(){return"function"==typeof e?e(r):{...r,...e}}),[r,e])}function s(e){let r;return r=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:i(e.components),n.createElement(a.Provider,{value:r},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/b69370a5.cd361d24.js b/Doc/assets/js/b69370a5.cd361d24.js new file mode 100644 index 00000000..d11d5e7f --- /dev/null +++ b/Doc/assets/js/b69370a5.cd361d24.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[7288],{36967:(e,r,t)=>{t.r(r),t.d(r,{assets:()=>c,contentTitle:()=>l,default:()=>g,frontMatter:()=>s,metadata:()=>d,toc:()=>u});var n=t(74848),o=t(28453),a=t(27064),i=t(89236);const s={id:"ue-error-handling",title:"Error Handling",slug:"/unreal/getting-started/error-handling",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/error-handling.mdx"},l=void 0,d={id:"getting-started/ue-error-handling",title:"Error Handling",description:"Error handling in Blueprint is still undergoing improvement. Please let us know if you have any feedback or suggestions for improving it!",source:"@site/public/en-us/getting-started/error-handling.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/error-handling",permalink:"/unreal/getting-started/error-handling",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/error-handling.mdx",tags:[],version:"current",frontMatter:{id:"ue-error-handling",title:"Error Handling",slug:"/unreal/getting-started/error-handling",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/error-handling.mdx"},sidebar:"sidebar",previous:{title:"Edit an Existing Mod",permalink:"/unreal/getting-started/edit-mods"},next:{title:"Monetization",permalink:"/unreal/getting-started/monetization"}},c={},u=[{value:"Checking for errors",id:"checking-for-errors",level:3},{value:"Inspecting ModioErrorCode more deeply",id:"inspecting-modioerrorcode-more-deeply",level:3},{value:"Semantic queries",id:"semantic-queries",level:4},{value:"Parameter validation errors",id:"parameter-validation-errors",level:3}];function h(e){const r={a:"a",admonition:"admonition",code:"code",h3:"h3",h4:"h4",img:"img",p:"p",pre:"pre",...(0,o.R)(),...e.components};return(0,n.jsxs)(n.Fragment,{children:[(0,n.jsx)(r.admonition,{type:"note",children:(0,n.jsx)(r.p,{children:"Error handling in Blueprint is still undergoing improvement. Please let us know if you have any feedback or suggestions for improving it!"})}),"\n",(0,n.jsxs)(r.p,{children:["Many of the plugin's functions utilize the ",(0,n.jsx)(r.a,{href:"/unreal/refdocs/#modioerrorcode",children:(0,n.jsx)(r.code,{children:"ModioErrorCode"})})," type. For example, all ",(0,n.jsx)(r.a,{href:"/unreal/getting-started/plugin-structure/#non-blocking-asynchronous-interface",children:"async"})," functions take a ",(0,n.jsx)(r.code,{children:"ModioErrorCode"})," as the first parameter for the function's delegate. This is essentially an opaque wrapper around a numeric error code with a category and an associated string message."]}),"\n",(0,n.jsx)(r.h3,{id:"checking-for-errors",children:"Checking for errors"}),"\n",(0,n.jsxs)(r.p,{children:['A "truthy" error code represents an error. In Blueprint, you can check if a ',(0,n.jsx)(r.code,{children:"ModioErrorCode"})," represents a success or failure by using the ",(0,n.jsx)(r.code,{children:"IsError"})," node."]}),"\n",(0,n.jsxs)(a.A,{"group-id":"languages",children:[(0,n.jsx)(i.A,{value:"blueprint",label:"Blueprint",children:(0,n.jsx)(r.p,{children:(0,n.jsx)(r.img,{alt:"is_error",src:t(45530).A+"",width:"470",height:"118"})})}),(0,n.jsx)(i.A,{value:"c++",label:"C++",default:!0,children:(0,n.jsx)(r.pre,{children:(0,n.jsx)(r.code,{className:"language-cpp",children:"void UModioManagerSubsystem::MyCallback(FModioErrorCode ErrorCode)\n{\n if (ErrorCode)\n {\n // Failure\n }\n else\n {\n // Success\n }\n}\n"})})})]}),"\n",(0,n.jsxs)(r.h3,{id:"inspecting-modioerrorcode-more-deeply",children:["Inspecting ",(0,n.jsx)(r.code,{children:"ModioErrorCode"})," more deeply"]}),"\n",(0,n.jsx)(r.p,{children:"Sometimes this information will be all that is required; just a simple 'success/fail' that you can handle."}),"\n",(0,n.jsxs)(r.p,{children:["In many cases, however, you will want to perform some degree of inspection on an ",(0,n.jsx)(r.code,{children:"ModioErrorCode"})," to determine specific information about that error - if nothing else, so that you can display a reason for the failure to the end user."]}),"\n",(0,n.jsx)(r.h4,{id:"semantic-queries",children:"Semantic queries"}),"\n",(0,n.jsx)(r.p,{children:'In your application, there are many errors that will likely be handled the same way. For example, you probably don\u2019t need to handle different network errors in different ways. The semantics of networking errors are largely "try the function again later".'}),"\n",(0,n.jsxs)(r.p,{children:["This is where ",(0,n.jsx)(r.a,{href:"/unreal/refdocs/#error-code-matches",children:(0,n.jsx)(r.code,{children:"ErrorCodeMatches"})}),' comes in. It allows you to query if the error satisfies a particular condition, such as "does this code represent some kind of networking error", without needing to explicitly check the code against all the individual errors in the category.']}),"\n",(0,n.jsx)(r.p,{children:(0,n.jsx)(r.img,{alt:"error_code_matches_network_error",src:t(25701).A+"",width:"633",height:"228"})}),"\n",(0,n.jsx)(r.p,{children:"By querying if an error meets a specific condition, you can focus on handling a family of errors (in this case, network transmission errors) without needing to deal with individual errors within that group. No more manually checking against individual HttpError values, just a single query."}),"\n",(0,n.jsx)(r.p,{children:"These semantic checks help you to consolidate your error handling into limited set of generic error handlers rather than dealing with each potential outcome individually."}),"\n",(0,n.jsx)(r.p,{children:(0,n.jsx)(r.img,{alt:"error_code_matches_chaining",src:t(34803).A+"",width:"1231",height:"507"})}),"\n",(0,n.jsx)(r.h3,{id:"parameter-validation-errors",children:"Parameter validation errors"}),"\n",(0,n.jsxs)(r.p,{children:["Some of the plugin functions may return errors that indicate a parameter or data validation failure. For these cases, the plugin parses the error response from the mod.io API and stores the information about which parameters failed validation until the next network request is performed. If a plugin function returns an error which matches ",(0,n.jsx)(r.code,{children:"EModioErrorCondition::InvalidArgsError"}),", you can call ",(0,n.jsx)(r.code,{children:"GetLastValidationError"})," in your callback to retrieve those errors and display appropriate feedback to the end user."]}),"\n",(0,n.jsx)(r.p,{children:(0,n.jsx)(r.img,{alt:"get_last_validation_error",src:t(84905).A+"",width:"1340",height:"247"})})]})}function g(e={}){const{wrapper:r}={...(0,o.R)(),...e.components};return r?(0,n.jsx)(r,{...e,children:(0,n.jsx)(h,{...e})}):h(e)}},19365:(e,r,t)=>{t.d(r,{A:()=>i});t(96540);var n=t(18215);const o={tabItem:"tabItem_Ymn6"};var a=t(74848);function i(e){let{children:r,hidden:t,className:i}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,n.A)(o.tabItem,i),hidden:t,children:r})}},11470:(e,r,t)=>{t.d(r,{A:()=>j});var n=t(96540),o=t(18215),a=t(23104),i=t(56347),s=t(205),l=t(57485),d=t(31682),c=t(70679);function u(e){return n.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,n.isValidElement)(e)&&function(e){const{props:r}=e;return!!r&&"object"==typeof r&&"value"in r}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:r,children:t}=e;return(0,n.useMemo)((()=>{const e=r??function(e){return u(e).map((e=>{let{props:{value:r,label:t,attributes:n,default:o}}=e;return{value:r,label:t,attributes:n,default:o}}))}(t);return function(e){const r=(0,d.X)(e,((e,r)=>e.value===r.value));if(r.length>0)throw new Error(`Docusaurus error: Duplicate values "${r.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[r,t])}function g(e){let{value:r,tabValues:t}=e;return t.some((e=>e.value===r))}function m(e){let{queryString:r=!1,groupId:t}=e;const o=(0,i.W6)(),a=function(e){let{queryString:r=!1,groupId:t}=e;if("string"==typeof r)return r;if(!1===r)return null;if(!0===r&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:r,groupId:t});return[(0,l.aZ)(a),(0,n.useCallback)((e=>{if(!a)return;const r=new URLSearchParams(o.location.search);r.set(a,e),o.replace({...o.location,search:r.toString()})}),[a,o])]}function p(e){const{defaultValue:r,queryString:t=!1,groupId:o}=e,a=h(e),[i,l]=(0,n.useState)((()=>function(e){let{defaultValue:r,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(r){if(!g({value:r,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${r}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return r}const n=t.find((e=>e.default))??t[0];if(!n)throw new Error("Unexpected error: 0 tabValues");return n.value}({defaultValue:r,tabValues:a}))),[d,u]=m({queryString:t,groupId:o}),[p,f]=function(e){let{groupId:r}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(r),[o,a]=(0,c.Dv)(t);return[o,(0,n.useCallback)((e=>{t&&a.set(e)}),[t,a])]}({groupId:o}),b=(()=>{const e=d??p;return g({value:e,tabValues:a})?e:null})();(0,s.A)((()=>{b&&l(b)}),[b]);return{selectedValue:i,selectValue:(0,n.useCallback)((e=>{if(!g({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),u(e),f(e)}),[u,f,a]),tabValues:a}}var f=t(92303);const b={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var v=t(74848);function y(e){let{className:r,block:t,selectedValue:n,selectValue:i,tabValues:s}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,a.a_)(),c=e=>{const r=e.currentTarget,t=l.indexOf(r),o=s[t].value;o!==n&&(d(r),i(o))},u=e=>{let r=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const t=l.indexOf(e.currentTarget)+1;r=l[t]??l[0];break}case"ArrowLeft":{const t=l.indexOf(e.currentTarget)-1;r=l[t]??l[l.length-1];break}}r?.focus()};return(0,v.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.A)("tabs",{"tabs--block":t},r),children:s.map((e=>{let{value:r,label:t,attributes:a}=e;return(0,v.jsx)("li",{role:"tab",tabIndex:n===r?0:-1,"aria-selected":n===r,ref:e=>l.push(e),onKeyDown:u,onClick:c,...a,className:(0,o.A)("tabs__item",b.tabItem,a?.className,{"tabs__item--active":n===r}),children:t??r},r)}))})}function x(e){let{lazy:r,children:t,selectedValue:o}=e;const a=(Array.isArray(t)?t:[t]).filter(Boolean);if(r){const e=a.find((e=>e.props.value===o));return e?(0,n.cloneElement)(e,{className:"margin-top--md"}):null}return(0,v.jsx)("div",{className:"margin-top--md",children:a.map(((e,r)=>(0,n.cloneElement)(e,{key:r,hidden:e.props.value!==o})))})}function w(e){const r=p(e);return(0,v.jsxs)("div",{className:(0,o.A)("tabs-container",b.tabList),children:[(0,v.jsx)(y,{...r,...e}),(0,v.jsx)(x,{...r,...e})]})}function j(e){const r=(0,f.A)();return(0,v.jsx)(w,{...e,children:u(e.children)},String(r))}},89236:(e,r,t)=>{t.d(r,{A:()=>a});var n=t(19365),o=(t(96540),t(74848));function a(e){return(0,o.jsx)(o.Fragment,{children:(0,o.jsx)(n.A,{className:"tw-rounded-md",...e})})}},27064:(e,r,t)=>{t.d(r,{A:()=>a});var n=t(11470),o=(t(96540),t(74848));function a(e){return(0,o.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,o.jsx)(n.A,{...e})})}},34803:(e,r,t)=>{t.d(r,{A:()=>n});const n=t.p+"assets/images/error_code_matches_chaining-86eec7eaebbbe82bf284e055ebd192d1.png"},25701:(e,r,t)=>{t.d(r,{A:()=>n});const n=t.p+"assets/images/error_code_matches_network_error-7ff80a95b75b1cf050da273595c027c5.png"},84905:(e,r,t)=>{t.d(r,{A:()=>n});const n=t.p+"assets/images/get_last_validation_error-bc856beda3feb292b5056037c771719a.png"},45530:(e,r,t)=>{t.d(r,{A:()=>n});const n=t.p+"assets/images/is_error-d9d2d4f7512b0e2c1bc6de68fbb1c255.png"},28453:(e,r,t)=>{t.d(r,{R:()=>i,x:()=>s});var n=t(96540);const o={},a=n.createContext(o);function i(e){const r=n.useContext(a);return n.useMemo((function(){return"function"==typeof e?e(r):{...r,...e}}),[r,e])}function s(e){let r;return r=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:i(e.components),n.createElement(a.Provider,{value:r},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/b84ad1c9.2d53dc4b.js b/Doc/assets/js/b84ad1c9.2d53dc4b.js deleted file mode 100644 index 2247bab8..00000000 --- a/Doc/assets/js/b84ad1c9.2d53dc4b.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[4101],{97949:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>c,contentTitle:()=>d,default:()=>m,frontMatter:()=>r,metadata:()=>l,toc:()=>u});var o=t(74848),s=t(28453),i=t(27064),a=t(89236);const r={id:"ue-mod-subscriptions",title:"Mod Subscriptions & Management",slug:"/unreal/getting-started/mod-subscriptions",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/mod-subscriptions.mdx"},d=void 0,l={id:"getting-started/ue-mod-subscriptions",title:"Mod Subscriptions & Management",description:"So you\u2019ve shown the user some mods, and they've picked one they\u2019d like to install. How do you begin the installation process? Once the mod is installed, how do you know what files to load into memory?",source:"@site/public/en-us/getting-started/subscriptions.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/mod-subscriptions",permalink:"/unreal/getting-started/mod-subscriptions",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/mod-subscriptions.mdx",tags:[],version:"current",frontMatter:{id:"ue-mod-subscriptions",title:"Mod Subscriptions & Management",slug:"/unreal/getting-started/mod-subscriptions",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/mod-subscriptions.mdx"},sidebar:"sidebar",previous:{title:"Browsing Mods",permalink:"/unreal/getting-started/browsing-mods"},next:{title:"In-Game Mod Submission",permalink:"/unreal/getting-started/submit-mods"}},c={},u=[{value:"Mod subscriptions",id:"mod-subscriptions",level:2},{value:"External subscription changes",id:"external-subscription-changes",level:2},{value:"Checking the user subscription list",id:"checking-the-user-subscription-list",level:2},{value:"Installation management",id:"installation-management",level:2},{value:"Retrieving mod directory paths for loading",id:"retrieving-mod-directory-paths-for-loading",level:3}];function h(e){const n={a:"a",admonition:"admonition",code:"code",em:"em",h2:"h2",h3:"h3",img:"img",p:"p",pre:"pre",strong:"strong",...(0,s.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsxs)(n.p,{children:["So you\u2019ve ",(0,o.jsx)(n.a,{href:"browsing-mods",children:"shown the user some mods"}),", and they've picked one they\u2019d like to install. How do you begin the installation process? Once the mod is installed, how do you know what files to load into memory?"]}),"\n",(0,o.jsx)(n.h2,{id:"mod-subscriptions",children:"Mod subscriptions"}),"\n",(0,o.jsxs)(n.p,{children:["Users indicate they want to install a mod by ",(0,o.jsx)(n.strong,{children:"subscribing"})," to it. Subscriptions are stored on the mod.io servers and are associated with a user\u2019s mod.io account, so subscribed mods for a given game will be installed to all devices where a user logs into mod.io for that game. Similarly, when a user ",(0,o.jsx)(n.strong,{children:"unsubscribes"})," from a mod, that mod will be uninstalled from every device they\u2019re logged into mod.io with for that game."]}),"\n",(0,o.jsxs)(n.p,{children:["Subscriptions are managed with calls to either ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_SubscribeToModAsync",children:(0,o.jsx)(n.code,{children:"SubscribeToModAsync"})})," or ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_UnsubscribeFromModAsync",children:(0,o.jsx)(n.code,{children:"UnsubscribeFromModAsync"})})," containing the desired ",(0,o.jsx)(n.code,{children:"ModioModID"})," and a delegate to receive the status of the request."]}),"\n",(0,o.jsx)(n.admonition,{type:"note",children:(0,o.jsxs)(n.p,{children:["To subscribe or unsubscribe from a mod, ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_EnableModManagement",children:(0,o.jsx)(n.strong,{children:(0,o.jsx)(n.code,{children:"EnableModManagement"})})})," ",(0,o.jsx)(n.em,{children:"must"})," be called beforehand."]})}),"\n",(0,o.jsxs)(n.p,{children:[(0,o.jsx)(n.code,{children:"SubscribeToModAsync"})," also takes a bool indicating whether or not to to subscribe to any and all dependencies for the given ",(0,o.jsx)(n.code,{children:"ModioModID"}),"."]}),"\n",(0,o.jsx)(n.admonition,{type:"note",children:(0,o.jsxs)(n.p,{children:["When dependencies are included in the ",(0,o.jsx)(n.code,{children:"SubscribeToModAsync"})," call, they will ",(0,o.jsx)(n.em,{children:"not"})," download automatically. ",(0,o.jsxs)(n.strong,{children:["Only the primary mod specified by ",(0,o.jsx)(n.code,{children:"ModioModID"})," will download automatically."]})," To download all subscribed content including dependencies, call ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_FetchExternalUpdatesAsync",children:(0,o.jsx)(n.code,{children:"FetchExternalUpdatesAsync"})})," after ",(0,o.jsx)(n.code,{children:"SubscribeToModAsync"})," successfully completes."]})}),"\n",(0,o.jsxs)(i.A,{"group-id":"languages",children:[(0,o.jsxs)(a.A,{value:"blueprint",label:"Blueprint",children:[(0,o.jsx)(n.p,{children:(0,o.jsx)(n.img,{alt:"subscribe_to_mod",src:t(22095).A+"",width:"785",height:"393"})}),(0,o.jsx)(n.p,{children:(0,o.jsx)(n.img,{alt:"unsubscribe_from_mod",src:t(28269).A+"",width:"785",height:"393"})})]}),(0,o.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::SubscribeToMod(FModioModID ModToSubscribeTo, bool IncludeDependencies)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->SubscribeToModAsync(ModId, IncludeDependencies, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnSubscribeToModComplete, ModId));\n }\n}\n\nvoid UModioManagerSubsystem::OnSubscribeToModComplete(FModioErrorCode ErrorCode, FModioModID ModId)\n{\n if (!ErrorCode)\n {\n \t// Indicate success to your user. This ModId's files will begin installing.\n \t// Call FetchExternalUpdatesAsync if dependencies were included \n }\n}\n\nvoid UModioManagerSubsystem::UnsubscribeFromMod(FModioModID ModId)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->UnsubscribeFromModAsync(ModId, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnUnsubscribeFromModComplete, ModId));\n }\n}\n\nvoid UModioManagerSubsystem::OnUnsubscribeFromModComplete(FModioErrorCode ErrorCode, FModioModId ModId)\n{\n if (!ErrorCode)\n {\n \t// Indicate success to your user. This ModID's files will begin uninstalling.\n }\n}\n"})})})]}),"\n",(0,o.jsx)(n.h2,{id:"external-subscription-changes",children:"External subscription changes"}),"\n",(0,o.jsx)(n.p,{children:"Because mod.io's services are available via our website, users can manage their subscriptions outside of your application. This means that we need to be able to query the server for any external subscription changes."}),"\n",(0,o.jsxs)(n.p,{children:["Call ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_FetchExternalUpdatesAsync",children:(0,o.jsx)(n.code,{children:"FetchExternalUpdatesAsync"})})," to synchronise the server state with the plugin\u2019s local subscriptions. Any required installations or uninstallations based on the updated user subscriptions will be processed automatically."]}),"\n",(0,o.jsxs)(i.A,{"group-id":"languages",children:[(0,o.jsx)(a.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(n.p,{children:(0,o.jsx)(n.img,{alt:"fetch_external_updates",src:t(55371).A+"",width:"525",height:"315"})})}),(0,o.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-cpp",children:"\nvoid UModioManagerSubsystem::FetchExternalUpdates()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->FetchExternalUpdatesAsync(FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnFetchExternalUpdatesComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnFetchExternalUpdatesComplete(FModioErrorCode ErrorCode)\n{\n // error handling etc.\n}\n\n"})})})]}),"\n",(0,o.jsx)(n.admonition,{type:"warning",children:(0,o.jsxs)(n.p,{children:["Call ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_FetchExternalUpdatesAsync",children:(0,o.jsx)(n.code,{children:"FetchExternalUpdatesAsync"})})," sparingly, only when required to ensure that the local state is up-to-date such as on game start-up or based on user input (i.e. as a button on your UI). It should ",(0,o.jsx)(n.strong,{children:"NOT"})," be called on tick."]})}),"\n",(0,o.jsx)(n.h2,{id:"checking-the-user-subscription-list",children:"Checking the user subscription list"}),"\n",(0,o.jsxs)(n.p,{children:["To see which mods the user has subscribed to, call ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#QueryUserSubscriptions",children:(0,o.jsx)(n.code,{children:"QueryUserSubscriptions"})}),". This retrieves a ",(0,o.jsx)(n.code,{children:"TMap"})," of ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modiomodcollectionentry",children:(0,o.jsx)(n.code,{children:"ModioModCollectionEntry"})})," objects, one for each subscribed mod. Each ",(0,o.jsx)(n.code,{children:"ModioModCollectionEntry"})," object contains the mod\u2019s state, profile information, ID and more."]}),"\n",(0,o.jsx)(n.admonition,{type:"note",children:(0,o.jsxs)(n.p,{children:["This collection includes mods that are still in the process of being installed! Make sure to check the result of ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#GetModState",children:(0,o.jsx)(n.code,{children:"GetModState"})})," before attempting to load files from the mods in this collection. Alternatively, use ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#QueryUserInstallations",children:(0,o.jsx)(n.code,{children:"QueryUserInstallations"})})," as described in ",(0,o.jsx)(n.a,{href:"#retrieving-mod-directory-paths-for-loading",children:(0,o.jsx)(n.strong,{children:"Retrieving mod directory paths for loading"})}),"."]})}),"\n",(0,o.jsxs)(i.A,{"group-id":"languages",children:[(0,o.jsx)(a.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(n.p,{children:(0,o.jsx)(n.img,{alt:"query_user_subscriptions",src:t(567).A+"",width:"754",height:"338"})})}),(0,o.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::QueryUserSubscriptions()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tTMap SubscribedMods = Subsystem->QueryUserSubscriptions();\n \t// Do something with SubscribedMods e.g. display on a UI\n }\n}\n"})})})]}),"\n",(0,o.jsx)(n.h2,{id:"installation-management",children:"Installation management"}),"\n",(0,o.jsxs)(n.p,{children:["So a subscription marks a mod as requiring installation, and an unsubscription indicates uninstallation, but how do you actually control when the plugin ",(0,o.jsx)(n.strong,{children:"does"})," those things? After all, you don\u2019t want a mod to be uninstalled after your main program has loaded those files into memory, locking them from deletion. Likewise, you probably don\u2019t want to be using networking or processor resources during gameplay for downloading mods. To give you control over when these processes occur without forcing you to shut down the plugin, you can call ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_EnableModManagement",children:(0,o.jsx)(n.code,{children:"EnableModManagement"})})," and ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#DisableModManagement",children:(0,o.jsx)(n.code,{children:"DisableModManagement"})}),"."]}),"\n",(0,o.jsxs)(n.p,{children:["To notify your users when a mod is finished installing or updating, ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_EnableModManagement",children:(0,o.jsx)(n.code,{children:"EnableModManagement"})})," asks you to provide it with a callback. This callback will be invoked ",(0,o.jsx)(n.strong,{children:"every time"})," a ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modiomodmanagementevent",children:(0,o.jsx)(n.code,{children:"ModioModManagementEvent"})})," occurs i.e. whenever a mod is installed, updated, uninstalled, or uploaded by the plugin\u2019s internal event loop. This behavior persists until a corresponding call to ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#DisableModManagement",children:(0,o.jsx)(n.code,{children:"DisableModManagement"})})," or ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_ShutdownAsync",children:(0,o.jsx)(n.code,{children:"ShutdownAsync"})})," is made."]}),"\n",(0,o.jsx)(n.admonition,{type:"note",children:(0,o.jsxs)(n.p,{children:[(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_EnableModManagement",children:(0,o.jsx)(n.code,{children:"EnableModManagement"})})," is not an async function. It does not end with the ",(0,o.jsx)(n.code,{children:"*Async"})," suffix. Its callback operates differently to asynchronous result callbacks used elsewhere in the mod.io plugin."]})}),"\n",(0,o.jsxs)(i.A,{"group-id":"languages",children:[(0,o.jsx)(a.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(n.p,{children:(0,o.jsx)(n.img,{alt:"enable_mod_management",src:t(3462).A+"",width:"784",height:"539"})})}),(0,o.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::EnableModManagement()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->EnableModManagement(FOnModManagementDelegateFast::CreateUObject(this, &UModioManagerSubsystem::ModManagementCallback));\n }\n}\n\nvoid UModioManagerSubsystem::ModManagementCallback(FModioModManagementEvent ModManagementEvent)\n{\n if (ModManagementEvent.Status)\n {\n \t// error handling\n }\n switch(ModManagementEvent.Event)\n {\n \tcase EModioModManagementEventType::BeginInstall:\n \tcase EModioModManagementEventType::BeginUninstall:\n \tcase EModioModManagementEventType::BeginUpdate:\n \tcase EModioModManagementEventType::BeginUpload:\n \t\tUE_LOG(LogModioGame, Log, TEXT("Begin processing Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tcase EModioModManagementEventType::Installed:\n \t\tUE_LOG(LogModioGame, Log, TEXT("Received an Installed event for Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tcase EModioModManagementEventType::Uninstalled: \n \t\tUE_LOG(LogModioGame, Log, TEXT("Received an Uninstalled event for Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tcase EModioModManagementEventType::Updated:\n \t\tUE_LOG(LogModioGame, Log, TEXT("Received an Updated event for Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tcase EModioModManagementEventType::Uploaded:\n \t\tUE_LOG(LogModioGame, Log, TEXT("Received an Uploaded event for Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tdefault:;\n }\n}\n'})})})]}),"\n",(0,o.jsxs)(n.p,{children:["While mod management is enabled, the plugin assumes that it has the ability to make changes to the filesystem, including deleting mods that the user has no longer subscribed to. As a result you should make sure that you don\u2019t have any open handles to files inside the mod directories when you call ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_EnableModManagement",children:(0,o.jsx)(n.code,{children:"EnableModManagement"})}),"."]}),"\n",(0,o.jsxs)(n.p,{children:["When you want to be able to freely open files in the mod directories, call ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#DisableModManagement",children:(0,o.jsx)(n.code,{children:"DisableModManagement"})}),". The plugin will finish the current operation but will not continue any others."]}),"\n",(0,o.jsxs)(n.p,{children:["Call ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#IsModManagementBusy",children:(0,o.jsx)(n.code,{children:"IsModManagementBusy"})})," to see if mod management is currently processing a mod."]}),"\n",(0,o.jsx)(i.A,{"group-id":"languages",children:(0,o.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::DisableModManagement()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tif (!Subsystem->IsModManagementBusy())\n \t{\n \t\tSubsystem->DisableModManagement();\n \t}\n }\n}\n"})})})}),"\n",(0,o.jsx)(n.h3,{id:"retrieving-mod-directory-paths-for-loading",children:"Retrieving mod directory paths for loading"}),"\n",(0,o.jsx)(n.p,{children:"So now we have the user picking mods and marking them for installation, we\u2019re enabling mod management at a point where we don\u2019t mind the plugin changing the filesystem, and mods are being installed by the plugin. We now need to know where they are on disk, otherwise you can\u2019t load them into your game!"}),"\n",(0,o.jsxs)(n.p,{children:["The easiest way to do this is by using ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#QueryUserInstallations",children:(0,o.jsx)(n.code,{children:"QueryUserInstallations"})}),". This function returns a ",(0,o.jsx)(n.code,{children:"TMap"})," of ",(0,o.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modiomodcollectionentry",children:(0,o.jsx)(n.code,{children:"ModioModCollectionEntry"})})," objects that can be queried for folder paths to use for loading a mod's files into your game. ",(0,o.jsx)(n.code,{children:"QueryUserInstallations"})," also allows you to specify whether or not to include outdated mods."]}),"\n",(0,o.jsxs)(i.A,{"group-id":"languages",children:[(0,o.jsx)(a.A,{value:"c++",label:"C++",default:!0,children:(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::GetInstalledMods()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tTMap InstalledMods = Subsystem->QueryUserInstallations(false);\n \t\n \t// Do something with each installed mod, ie adding paths/loading the content appropriately\n }\n}\n"})})}),(0,o.jsx)(a.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(n.p,{children:(0,o.jsx)(n.img,{alt:"query_user_installations",src:t(51314).A+"",width:"1071",height:"248"})})})]})]})}function m(e={}){const{wrapper:n}={...(0,s.R)(),...e.components};return n?(0,o.jsx)(n,{...e,children:(0,o.jsx)(h,{...e})}):h(e)}},19365:(e,n,t)=>{t.d(n,{A:()=>a});t(96540);var o=t(18215);const s={tabItem:"tabItem_Ymn6"};var i=t(74848);function a(e){let{children:n,hidden:t,className:a}=e;return(0,i.jsx)("div",{role:"tabpanel",className:(0,o.A)(s.tabItem,a),hidden:t,children:n})}},11470:(e,n,t)=>{t.d(n,{A:()=>v});var o=t(96540),s=t(18215),i=t(23104),a=t(56347),r=t(205),d=t(57485),l=t(31682),c=t(70679);function u(e){return o.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:n,children:t}=e;return(0,o.useMemo)((()=>{const e=n??function(e){return u(e).map((e=>{let{props:{value:n,label:t,attributes:o,default:s}}=e;return{value:n,label:t,attributes:o,default:s}}))}(t);return function(e){const n=(0,l.X)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function m(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function b(e){let{queryString:n=!1,groupId:t}=e;const s=(0,a.W6)(),i=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,d.aZ)(i),(0,o.useCallback)((e=>{if(!i)return;const n=new URLSearchParams(s.location.search);n.set(i,e),s.replace({...s.location,search:n.toString()})}),[i,s])]}function g(e){const{defaultValue:n,queryString:t=!1,groupId:s}=e,i=h(e),[a,d]=(0,o.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!m({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const o=t.find((e=>e.default))??t[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:n,tabValues:i}))),[l,u]=b({queryString:t,groupId:s}),[g,p]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[s,i]=(0,c.Dv)(t);return[s,(0,o.useCallback)((e=>{t&&i.set(e)}),[t,i])]}({groupId:s}),M=(()=>{const e=l??g;return m({value:e,tabValues:i})?e:null})();(0,r.A)((()=>{M&&d(M)}),[M]);return{selectedValue:a,selectValue:(0,o.useCallback)((e=>{if(!m({value:e,tabValues:i}))throw new Error(`Can't select invalid tab value=${e}`);d(e),u(e),p(e)}),[u,p,i]),tabValues:i}}var p=t(92303);const M={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var f=t(74848);function y(e){let{className:n,block:t,selectedValue:o,selectValue:a,tabValues:r}=e;const d=[],{blockElementScrollPositionUntilNextRender:l}=(0,i.a_)(),c=e=>{const n=e.currentTarget,t=d.indexOf(n),s=r[t].value;s!==o&&(l(n),a(s))},u=e=>{let n=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const t=d.indexOf(e.currentTarget)+1;n=d[t]??d[0];break}case"ArrowLeft":{const t=d.indexOf(e.currentTarget)-1;n=d[t]??d[d.length-1];break}}n?.focus()};return(0,f.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,s.A)("tabs",{"tabs--block":t},n),children:r.map((e=>{let{value:n,label:t,attributes:i}=e;return(0,f.jsx)("li",{role:"tab",tabIndex:o===n?0:-1,"aria-selected":o===n,ref:e=>d.push(e),onKeyDown:u,onClick:c,...i,className:(0,s.A)("tabs__item",M.tabItem,i?.className,{"tabs__item--active":o===n}),children:t??n},n)}))})}function x(e){let{lazy:n,children:t,selectedValue:s}=e;const i=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=i.find((e=>e.props.value===s));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return(0,f.jsx)("div",{className:"margin-top--md",children:i.map(((e,n)=>(0,o.cloneElement)(e,{key:n,hidden:e.props.value!==s})))})}function j(e){const n=g(e);return(0,f.jsxs)("div",{className:(0,s.A)("tabs-container",M.tabList),children:[(0,f.jsx)(y,{...n,...e}),(0,f.jsx)(x,{...n,...e})]})}function v(e){const n=(0,p.A)();return(0,f.jsx)(j,{...e,children:u(e.children)},String(n))}},89236:(e,n,t)=>{t.d(n,{A:()=>i});var o=t(19365),s=(t(96540),t(74848));function i(e){return(0,s.jsx)(s.Fragment,{children:(0,s.jsx)(o.A,{className:"tw-rounded-md",...e})})}},27064:(e,n,t)=>{t.d(n,{A:()=>i});var o=t(11470),s=(t(96540),t(74848));function i(e){return(0,s.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,s.jsx)(o.A,{...e})})}},3462:(e,n,t)=>{t.d(n,{A:()=>o});const o=t.p+"assets/images/enable_mod_management-d7351b002e16e9781646be8381e57e12.png"},55371:(e,n,t)=>{t.d(n,{A:()=>o});const o=t.p+"assets/images/fetch_external_updates-815fb53cb41b2a7529c4ab0f6484f5dc.png"},51314:(e,n,t)=>{t.d(n,{A:()=>o});const o=t.p+"assets/images/query_user_installations-719f403a60b8a2a20541dd4d547fefd9.png"},567:(e,n,t)=>{t.d(n,{A:()=>o});const o=t.p+"assets/images/query_user_subscriptions-c0a465f3e522e5bbb9f8d0a28ddd47f5.png"},22095:(e,n,t)=>{t.d(n,{A:()=>o});const o=t.p+"assets/images/subscribe_to_mod-9f9f05d9c9ffd8d4ca2d2f6d383f25a3.png"},28269:(e,n,t)=>{t.d(n,{A:()=>o});const o=t.p+"assets/images/unsubscribe_from_mod-ad13b0ee52b1ada4ab5713683a5e42f2.png"},28453:(e,n,t)=>{t.d(n,{R:()=>a,x:()=>r});var o=t(96540);const s={},i=o.createContext(s);function a(e){const n=o.useContext(i);return o.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:a(e.components),o.createElement(i.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/b84ad1c9.ddb378a0.js b/Doc/assets/js/b84ad1c9.ddb378a0.js new file mode 100644 index 00000000..1f37e023 --- /dev/null +++ b/Doc/assets/js/b84ad1c9.ddb378a0.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[4101],{97949:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>c,contentTitle:()=>d,default:()=>m,frontMatter:()=>r,metadata:()=>l,toc:()=>u});var s=t(74848),o=t(28453),a=t(27064),i=t(89236);const r={id:"ue-mod-subscriptions",title:"Mod Subscriptions & Management",slug:"/unreal/getting-started/mod-subscriptions",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/mod-subscriptions.mdx"},d=void 0,l={id:"getting-started/ue-mod-subscriptions",title:"Mod Subscriptions & Management",description:"So you\u2019ve shown the user some mods, and they've picked one they\u2019d like to install. How do you begin the installation process? Once the mod is installed, how do you know what files to load into memory?",source:"@site/public/en-us/getting-started/subscriptions.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/mod-subscriptions",permalink:"/unreal/getting-started/mod-subscriptions",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/mod-subscriptions.mdx",tags:[],version:"current",frontMatter:{id:"ue-mod-subscriptions",title:"Mod Subscriptions & Management",slug:"/unreal/getting-started/mod-subscriptions",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/mod-subscriptions.mdx"},sidebar:"sidebar",previous:{title:"Browsing Mods",permalink:"/unreal/getting-started/browsing-mods"},next:{title:"In-Game Mod Submission",permalink:"/unreal/getting-started/submit-mods"}},c={},u=[{value:"Mod subscriptions",id:"mod-subscriptions",level:2},{value:"External subscription changes",id:"external-subscription-changes",level:2},{value:"Checking the user subscription list",id:"checking-the-user-subscription-list",level:2},{value:"Installation management",id:"installation-management",level:2},{value:"Retrieving mod directory paths for loading",id:"retrieving-mod-directory-paths-for-loading",level:3}];function h(e){const n={a:"a",admonition:"admonition",code:"code",em:"em",h2:"h2",h3:"h3",img:"img",p:"p",pre:"pre",strong:"strong",...(0,o.R)(),...e.components};return(0,s.jsxs)(s.Fragment,{children:[(0,s.jsxs)(n.p,{children:["So you\u2019ve ",(0,s.jsx)(n.a,{href:"browsing-mods",children:"shown the user some mods"}),", and they've picked one they\u2019d like to install. How do you begin the installation process? Once the mod is installed, how do you know what files to load into memory?"]}),"\n",(0,s.jsx)(n.h2,{id:"mod-subscriptions",children:"Mod subscriptions"}),"\n",(0,s.jsxs)(n.p,{children:["Users indicate they want to install a mod by ",(0,s.jsx)(n.strong,{children:"subscribing"})," to it. Subscriptions are stored on the mod.io servers and are associated with a user\u2019s mod.io account, so subscribed mods for a given game will be installed to all devices where a user logs into mod.io for that game. Similarly, when a user ",(0,s.jsx)(n.strong,{children:"unsubscribes"})," from a mod, that mod will be uninstalled from every device they\u2019re logged into mod.io with for that game."]}),"\n",(0,s.jsxs)(n.p,{children:["Subscriptions are managed with calls to either ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#subscribetomodasync",children:(0,s.jsx)(n.code,{children:"SubscribeToModAsync"})})," or ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#unsubscribefrommodasync",children:(0,s.jsx)(n.code,{children:"UnsubscribeFromModAsync"})})," containing the desired ",(0,s.jsx)(n.code,{children:"ModioModID"})," and a delegate to receive the status of the request."]}),"\n",(0,s.jsx)(n.admonition,{type:"note",children:(0,s.jsxs)(n.p,{children:["To subscribe or unsubscribe from a mod, ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#enablemodmanagement",children:(0,s.jsx)(n.strong,{children:(0,s.jsx)(n.code,{children:"EnableModManagement"})})})," ",(0,s.jsx)(n.em,{children:"must"})," be called beforehand."]})}),"\n",(0,s.jsxs)(n.p,{children:[(0,s.jsx)(n.code,{children:"SubscribeToModAsync"})," also takes a bool indicating whether or not to to subscribe to any and all dependencies for the given ",(0,s.jsx)(n.code,{children:"ModioModID"}),"."]}),"\n",(0,s.jsx)(n.admonition,{type:"note",children:(0,s.jsxs)(n.p,{children:["When dependencies are included in the ",(0,s.jsx)(n.code,{children:"SubscribeToModAsync"})," call, they will ",(0,s.jsx)(n.em,{children:"not"})," download automatically. ",(0,s.jsxs)(n.strong,{children:["Only the primary mod specified by ",(0,s.jsx)(n.code,{children:"ModioModID"})," will download automatically."]})," To download all subscribed content including dependencies, call ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#fetchexternalupdatesasync",children:(0,s.jsx)(n.code,{children:"FetchExternalUpdatesAsync"})})," after ",(0,s.jsx)(n.code,{children:"SubscribeToModAsync"})," successfully completes."]})}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsxs)(i.A,{value:"blueprint",label:"Blueprint",children:[(0,s.jsx)(n.p,{children:(0,s.jsx)(n.img,{alt:"subscribe_to_mod",src:t(84272).A+"",width:"785",height:"393"})}),(0,s.jsx)(n.p,{children:(0,s.jsx)(n.img,{alt:"unsubscribe_from_mod",src:t(66738).A+"",width:"785",height:"393"})})]}),(0,s.jsx)(i.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(n.pre,{children:(0,s.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::SubscribeToMod(FModioModID ModToSubscribeTo, bool IncludeDependencies)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->SubscribeToModAsync(ModId, IncludeDependencies, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnSubscribeToModComplete, ModId));\n }\n}\n\nvoid UModioManagerSubsystem::OnSubscribeToModComplete(FModioErrorCode ErrorCode, FModioModID ModId)\n{\n if (!ErrorCode)\n {\n \t// Indicate success to your user. This ModId's files will begin installing.\n \t// Call FetchExternalUpdatesAsync if dependencies were included \n }\n}\n\nvoid UModioManagerSubsystem::UnsubscribeFromMod(FModioModID ModId)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->UnsubscribeFromModAsync(ModId, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnUnsubscribeFromModComplete, ModId));\n }\n}\n\nvoid UModioManagerSubsystem::OnUnsubscribeFromModComplete(FModioErrorCode ErrorCode, FModioModId ModId)\n{\n if (!ErrorCode)\n {\n \t// Indicate success to your user. This ModID's files will begin uninstalling.\n }\n}\n"})})})]}),"\n",(0,s.jsx)(n.h2,{id:"external-subscription-changes",children:"External subscription changes"}),"\n",(0,s.jsx)(n.p,{children:"Because mod.io's services are available via our website, users can manage their subscriptions outside of your application. This means that we need to be able to query the server for any external subscription changes."}),"\n",(0,s.jsxs)(n.p,{children:["Call ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#fetchexternalupdatesasync",children:(0,s.jsx)(n.code,{children:"FetchExternalUpdatesAsync"})})," to synchronise the server state with the plugin\u2019s local subscriptions. Any required installations or uninstallations based on the updated user subscriptions will be processed automatically."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(i.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(n.p,{children:(0,s.jsx)(n.img,{alt:"fetch_external_updates",src:t(34796).A+"",width:"525",height:"315"})})}),(0,s.jsx)(i.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(n.pre,{children:(0,s.jsx)(n.code,{className:"language-cpp",children:"\nvoid UModioManagerSubsystem::FetchExternalUpdates()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->FetchExternalUpdatesAsync(FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnFetchExternalUpdatesComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnFetchExternalUpdatesComplete(FModioErrorCode ErrorCode)\n{\n // error handling etc.\n}\n\n"})})})]}),"\n",(0,s.jsx)(n.admonition,{type:"warning",children:(0,s.jsxs)(n.p,{children:["Call ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#fetchexternalupdatesasync",children:(0,s.jsx)(n.code,{children:"FetchExternalUpdatesAsync"})})," sparingly, only when required to ensure that the local state is up-to-date such as on game start-up or based on user input (i.e. as a button on your UI). It should ",(0,s.jsx)(n.strong,{children:"NOT"})," be called on tick."]})}),"\n",(0,s.jsx)(n.h2,{id:"checking-the-user-subscription-list",children:"Checking the user subscription list"}),"\n",(0,s.jsxs)(n.p,{children:["To see which mods the user has subscribed to, call ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#query-user-subscriptions",children:(0,s.jsx)(n.code,{children:"QueryUserSubscriptions"})}),". This retrieves a ",(0,s.jsx)(n.code,{children:"TMap"})," of ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#modiomodcollectionentry",children:(0,s.jsx)(n.code,{children:"ModioModCollectionEntry"})})," objects, one for each subscribed mod. Each ",(0,s.jsx)(n.code,{children:"ModioModCollectionEntry"})," object contains the mod\u2019s state, profile information, ID and more."]}),"\n",(0,s.jsx)(n.admonition,{type:"note",children:(0,s.jsxs)(n.p,{children:["This collection includes mods that are still in the process of being installed! Make sure to check the result of ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#get-mod-state",children:(0,s.jsx)(n.code,{children:"GetModState"})})," before attempting to load files from the mods in this collection. Alternatively, use ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#query-user-installations",children:(0,s.jsx)(n.code,{children:"QueryUserInstallations"})})," as described in ",(0,s.jsx)(n.a,{href:"#retrieving-mod-directory-paths-for-loading",children:(0,s.jsx)(n.strong,{children:"Retrieving mod directory paths for loading"})}),"."]})}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(i.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(n.p,{children:(0,s.jsx)(n.img,{alt:"query_user_subscriptions",src:t(68260).A+"",width:"754",height:"338"})})}),(0,s.jsx)(i.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(n.pre,{children:(0,s.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::QueryUserSubscriptions()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tTMap SubscribedMods = Subsystem->QueryUserSubscriptions();\n \t// Do something with SubscribedMods e.g. display on a UI\n }\n}\n"})})})]}),"\n",(0,s.jsx)(n.h2,{id:"installation-management",children:"Installation management"}),"\n",(0,s.jsxs)(n.p,{children:["So a subscription marks a mod as requiring installation, and an unsubscription indicates uninstallation, but how do you actually control when the plugin ",(0,s.jsx)(n.strong,{children:"does"})," those things? After all, you don\u2019t want a mod to be uninstalled after your main program has loaded those files into memory, locking them from deletion. Likewise, you probably don\u2019t want to be using networking or processor resources during gameplay for downloading mods. To give you control over when these processes occur without forcing you to shut down the plugin, you can call ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#enablemodmanagement",children:(0,s.jsx)(n.code,{children:"EnableModManagement"})})," and ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#disable-mod-management",children:(0,s.jsx)(n.code,{children:"DisableModManagement"})}),"."]}),"\n",(0,s.jsxs)(n.p,{children:["To notify your users when a mod is finished installing or updating, ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#enablemodmanagement",children:(0,s.jsx)(n.code,{children:"EnableModManagement"})})," asks you to provide it with a callback. This callback will be invoked ",(0,s.jsx)(n.strong,{children:"every time"})," a ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#modiomodmanagementevent",children:(0,s.jsx)(n.code,{children:"ModioModManagementEvent"})})," occurs i.e. whenever a mod is installed, updated, uninstalled, or uploaded by the plugin\u2019s internal event loop. This behavior persists until a corresponding call to ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#disable-mod-management",children:(0,s.jsx)(n.code,{children:"DisableModManagement"})})," or ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#shutdownasync",children:(0,s.jsx)(n.code,{children:"ShutdownAsync"})})," is made."]}),"\n",(0,s.jsx)(n.admonition,{type:"note",children:(0,s.jsxs)(n.p,{children:[(0,s.jsx)(n.a,{href:"/unreal/refdocs/#enablemodmanagement",children:(0,s.jsx)(n.code,{children:"EnableModManagement"})})," is not an async function. It does not end with the ",(0,s.jsx)(n.code,{children:"*Async"})," suffix. Its callback operates differently to asynchronous result callbacks used elsewhere in the mod.io plugin."]})}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(i.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(n.p,{children:(0,s.jsx)(n.img,{alt:"enable_mod_management",src:t(94787).A+"",width:"784",height:"539"})})}),(0,s.jsx)(i.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(n.pre,{children:(0,s.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::EnableModManagement()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->EnableModManagement(FOnModManagementDelegateFast::CreateUObject(this, &UModioManagerSubsystem::ModManagementCallback));\n }\n}\n\nvoid UModioManagerSubsystem::ModManagementCallback(FModioModManagementEvent ModManagementEvent)\n{\n if (ModManagementEvent.Status)\n {\n \t// error handling\n }\n switch(ModManagementEvent.Event)\n {\n \tcase EModioModManagementEventType::BeginInstall:\n \tcase EModioModManagementEventType::BeginUninstall:\n \tcase EModioModManagementEventType::BeginUpdate:\n \tcase EModioModManagementEventType::BeginUpload:\n \t\tUE_LOG(LogModioGame, Log, TEXT("Begin processing Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tcase EModioModManagementEventType::Installed:\n \t\tUE_LOG(LogModioGame, Log, TEXT("Received an Installed event for Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tcase EModioModManagementEventType::Uninstalled: \n \t\tUE_LOG(LogModioGame, Log, TEXT("Received an Uninstalled event for Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tcase EModioModManagementEventType::Updated:\n \t\tUE_LOG(LogModioGame, Log, TEXT("Received an Updated event for Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tcase EModioModManagementEventType::Uploaded:\n \t\tUE_LOG(LogModioGame, Log, TEXT("Received an Uploaded event for Mod %s"), *ModManagementEvent.ID.ToString());\n \t\tbreak;\n \tdefault:;\n }\n}\n'})})})]}),"\n",(0,s.jsxs)(n.p,{children:["While mod management is enabled, the plugin assumes that it has the ability to make changes to the filesystem, including deleting mods that the user has no longer subscribed to. As a result you should make sure that you don\u2019t have any open handles to files inside the mod directories when you call ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#enablemodmanagement",children:(0,s.jsx)(n.code,{children:"EnableModManagement"})}),"."]}),"\n",(0,s.jsxs)(n.p,{children:["When you want to be able to freely open files in the mod directories, call ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#disable-mod-management",children:(0,s.jsx)(n.code,{children:"DisableModManagement"})}),". The plugin will finish the current operation but will not continue any others."]}),"\n",(0,s.jsxs)(n.p,{children:["Call ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#is-mod-management-busy",children:(0,s.jsx)(n.code,{children:"IsModManagementBusy"})})," to see if mod management is currently processing a mod."]}),"\n",(0,s.jsx)(a.A,{"group-id":"languages",children:(0,s.jsx)(i.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(n.pre,{children:(0,s.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::DisableModManagement()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tif (!Subsystem->IsModManagementBusy())\n \t{\n \t\tSubsystem->DisableModManagement();\n \t}\n }\n}\n"})})})}),"\n",(0,s.jsx)(n.h3,{id:"retrieving-mod-directory-paths-for-loading",children:"Retrieving mod directory paths for loading"}),"\n",(0,s.jsx)(n.p,{children:"So now we have the user picking mods and marking them for installation, we\u2019re enabling mod management at a point where we don\u2019t mind the plugin changing the filesystem, and mods are being installed by the plugin. We now need to know where they are on disk, otherwise you can\u2019t load them into your game!"}),"\n",(0,s.jsxs)(n.p,{children:["The easiest way to do this is by using ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#query-user-installations",children:(0,s.jsx)(n.code,{children:"QueryUserInstallations"})}),". This function returns a ",(0,s.jsx)(n.code,{children:"TMap"})," of ",(0,s.jsx)(n.a,{href:"/unreal/refdocs/#modiomodcollectionentry",children:(0,s.jsx)(n.code,{children:"ModioModCollectionEntry"})})," objects that can be queried for folder paths to use for loading a mod's files into your game. ",(0,s.jsx)(n.code,{children:"QueryUserInstallations"})," also allows you to specify whether or not to include outdated mods."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(i.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(n.pre,{children:(0,s.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::GetInstalledMods()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tTMap InstalledMods = Subsystem->QueryUserInstallations(false);\n \t\n \t// Do something with each installed mod, ie adding paths/loading the content appropriately\n }\n}\n"})})}),(0,s.jsx)(i.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(n.p,{children:(0,s.jsx)(n.img,{alt:"query_user_installations",src:t(68413).A+"",width:"1071",height:"248"})})})]})]})}function m(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,s.jsx)(n,{...e,children:(0,s.jsx)(h,{...e})}):h(e)}},19365:(e,n,t)=>{t.d(n,{A:()=>i});t(96540);var s=t(18215);const o={tabItem:"tabItem_Ymn6"};var a=t(74848);function i(e){let{children:n,hidden:t,className:i}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,s.A)(o.tabItem,i),hidden:t,children:n})}},11470:(e,n,t)=>{t.d(n,{A:()=>v});var s=t(96540),o=t(18215),a=t(23104),i=t(56347),r=t(205),d=t(57485),l=t(31682),c=t(70679);function u(e){return s.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,s.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:n,children:t}=e;return(0,s.useMemo)((()=>{const e=n??function(e){return u(e).map((e=>{let{props:{value:n,label:t,attributes:s,default:o}}=e;return{value:n,label:t,attributes:s,default:o}}))}(t);return function(e){const n=(0,l.X)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function m(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function b(e){let{queryString:n=!1,groupId:t}=e;const o=(0,i.W6)(),a=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,d.aZ)(a),(0,s.useCallback)((e=>{if(!a)return;const n=new URLSearchParams(o.location.search);n.set(a,e),o.replace({...o.location,search:n.toString()})}),[a,o])]}function g(e){const{defaultValue:n,queryString:t=!1,groupId:o}=e,a=h(e),[i,d]=(0,s.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!m({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const s=t.find((e=>e.default))??t[0];if(!s)throw new Error("Unexpected error: 0 tabValues");return s.value}({defaultValue:n,tabValues:a}))),[l,u]=b({queryString:t,groupId:o}),[g,p]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[o,a]=(0,c.Dv)(t);return[o,(0,s.useCallback)((e=>{t&&a.set(e)}),[t,a])]}({groupId:o}),f=(()=>{const e=l??g;return m({value:e,tabValues:a})?e:null})();(0,r.A)((()=>{f&&d(f)}),[f]);return{selectedValue:i,selectValue:(0,s.useCallback)((e=>{if(!m({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);d(e),u(e),p(e)}),[u,p,a]),tabValues:a}}var p=t(92303);const f={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var y=t(74848);function M(e){let{className:n,block:t,selectedValue:s,selectValue:i,tabValues:r}=e;const d=[],{blockElementScrollPositionUntilNextRender:l}=(0,a.a_)(),c=e=>{const n=e.currentTarget,t=d.indexOf(n),o=r[t].value;o!==s&&(l(n),i(o))},u=e=>{let n=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const t=d.indexOf(e.currentTarget)+1;n=d[t]??d[0];break}case"ArrowLeft":{const t=d.indexOf(e.currentTarget)-1;n=d[t]??d[d.length-1];break}}n?.focus()};return(0,y.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,o.A)("tabs",{"tabs--block":t},n),children:r.map((e=>{let{value:n,label:t,attributes:a}=e;return(0,y.jsx)("li",{role:"tab",tabIndex:s===n?0:-1,"aria-selected":s===n,ref:e=>d.push(e),onKeyDown:u,onClick:c,...a,className:(0,o.A)("tabs__item",f.tabItem,a?.className,{"tabs__item--active":s===n}),children:t??n},n)}))})}function x(e){let{lazy:n,children:t,selectedValue:o}=e;const a=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=a.find((e=>e.props.value===o));return e?(0,s.cloneElement)(e,{className:"margin-top--md"}):null}return(0,y.jsx)("div",{className:"margin-top--md",children:a.map(((e,n)=>(0,s.cloneElement)(e,{key:n,hidden:e.props.value!==o})))})}function j(e){const n=g(e);return(0,y.jsxs)("div",{className:(0,o.A)("tabs-container",f.tabList),children:[(0,y.jsx)(M,{...n,...e}),(0,y.jsx)(x,{...n,...e})]})}function v(e){const n=(0,p.A)();return(0,y.jsx)(j,{...e,children:u(e.children)},String(n))}},89236:(e,n,t)=>{t.d(n,{A:()=>a});var s=t(19365),o=(t(96540),t(74848));function a(e){return(0,o.jsx)(o.Fragment,{children:(0,o.jsx)(s.A,{className:"tw-rounded-md",...e})})}},27064:(e,n,t)=>{t.d(n,{A:()=>a});var s=t(11470),o=(t(96540),t(74848));function a(e){return(0,o.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,o.jsx)(s.A,{...e})})}},94787:(e,n,t)=>{t.d(n,{A:()=>s});const s=t.p+"assets/images/enable_mod_management-d7351b002e16e9781646be8381e57e12.png"},34796:(e,n,t)=>{t.d(n,{A:()=>s});const s=t.p+"assets/images/fetch_external_updates-815fb53cb41b2a7529c4ab0f6484f5dc.png"},68413:(e,n,t)=>{t.d(n,{A:()=>s});const s=t.p+"assets/images/query_user_installations-719f403a60b8a2a20541dd4d547fefd9.png"},68260:(e,n,t)=>{t.d(n,{A:()=>s});const s=t.p+"assets/images/query_user_subscriptions-c0a465f3e522e5bbb9f8d0a28ddd47f5.png"},84272:(e,n,t)=>{t.d(n,{A:()=>s});const s=t.p+"assets/images/subscribe_to_mod-9f9f05d9c9ffd8d4ca2d2f6d383f25a3.png"},66738:(e,n,t)=>{t.d(n,{A:()=>s});const s=t.p+"assets/images/unsubscribe_from_mod-ad13b0ee52b1ada4ab5713683a5e42f2.png"},28453:(e,n,t)=>{t.d(n,{R:()=>i,x:()=>r});var s=t(96540);const o={},a=s.createContext(o);function i(e){const n=s.useContext(a);return s.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:i(e.components),s.createElement(a.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/c5d35168.34c70772.js b/Doc/assets/js/c5d35168.34c70772.js deleted file mode 100644 index ae1f5d59..00000000 --- a/Doc/assets/js/c5d35168.34c70772.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[8156],{69285:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>c,contentTitle:()=>l,default:()=>m,frontMatter:()=>o,metadata:()=>u,toc:()=>d});var i=t(74848),s=t(28453),a=t(27064),r=t(89236);const o={id:"ue-user-authentication",title:"User Authentication",slug:"/unreal/getting-started/user-authentication",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/user-authentication.mdx"},l=void 0,u={id:"getting-started/ue-user-authentication",title:"User Authentication",description:"Best practices",source:"@site/public/en-us/getting-started/user-authentication.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/user-authentication",permalink:"/unreal/getting-started/user-authentication",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/user-authentication.mdx",tags:[],version:"current",frontMatter:{id:"ue-user-authentication",title:"User Authentication",slug:"/unreal/getting-started/user-authentication",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/user-authentication.mdx"},sidebar:"sidebar",previous:{title:"Initialization & Teardown",permalink:"/unreal/getting-started/initialization"},next:{title:"Browsing Mods",permalink:"/unreal/getting-started/browsing-mods"}},c={},d=[{value:"Best practices",id:"best-practices",level:2},{value:"Recommended flow",id:"recommended-flow",level:3},{value:"Terms & user consent",id:"terms--user-consent",level:2},{value:"Authentication methods",id:"authentication-methods",level:2},{value:"Email authentication",id:"email-authentication",level:3},{value:"Requesting an authentication code",id:"requesting-an-authentication-code",level:4},{value:"Submitting an authentication code",id:"submitting-an-authentication-code",level:4},{value:"Single sign-on authentication",id:"single-sign-on-authentication",level:3},{value:"Using the Unreal Online Subsystem",id:"using-the-unreal-online-subsystem",level:2},{value:"Token Lifetime & Re-Authentication",id:"token-lifetime--re-authentication",level:2}];function h(e){const n={a:"a",admonition:"admonition",code:"code",h2:"h2",h3:"h3",h4:"h4",img:"img",li:"li",p:"p",pre:"pre",ul:"ul",...(0,s.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"best-practices",children:"Best practices"}),"\n",(0,i.jsxs)(n.p,{children:["We recommend creating a new game-instance based subsystem (e.g. ",(0,i.jsx)(n.code,{children:"ModioManagerSubsystem"}),") that depends on ",(0,i.jsx)(n.code,{children:"ModioSubsystem"}),", and is responsible for managing the full initialization and authentication flow for your game."]}),"\n",(0,i.jsx)(n.h3,{id:"recommended-flow",children:"Recommended flow"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Create a subsystem to manage mod.io initialization and authentication."}),"\n",(0,i.jsxs)(n.li,{children:["Perform ",(0,i.jsx)(n.a,{href:"#using-the-unreal-online-subsystem",children:"Online Subsystem (OSS) login"})," in your subsystem's ",(0,i.jsx)(n.code,{children:"Initialize"})," function to ensure you have a valid platform auth token ready when its time to authenticate to mod.io using ",(0,i.jsx)(n.a,{href:"#single-sign-on-authentication",children:"single sign-on (SSO)"}),".","\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Log in with the online identity provider."}),"\n",(0,i.jsx)(n.li,{children:"On success, get a linked account token for the current user."}),"\n",(0,i.jsx)(n.li,{children:"On success, get an authentication token from the linked account or from the online identity."}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.a,{href:"initialization",children:"Initialize"})," the mod.io plugin with the required environment parameters for the current platform."]}),"\n",(0,i.jsxs)(n.li,{children:["Display the mod.io ",(0,i.jsx)(n.a,{href:"/terms/",children:"Terms of Service"}),". Ensure they are accepted by the user."]}),"\n",(0,i.jsxs)(n.li,{children:["Authenticate the user using ",(0,i.jsx)(n.a,{href:"#single-sign-on-authentication",children:"SSO"}),"."]}),"\n"]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsx)(n.p,{children:"If the Online Subsystem login fails at any point in the above process, fall back to simply initializing the plugin and offer email authentication as an alternative."})}),"\n",(0,i.jsx)(n.h2,{id:"terms--user-consent",children:"Terms & user consent"}),"\n",(0,i.jsxs)(n.p,{children:["Users must accept the mod.io Terms of Service before using any mod.io services. The Terms of Service can be fetched from mod.io using ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_GetTermsOfUseAsync",children:(0,i.jsx)(n.code,{children:"GetTermsOfUseAsync"})}),". Before attempting authentication, ensure these terms are displayed to and accepted by the user."]}),"\n",(0,i.jsxs)(n.p,{children:["See our full ",(0,i.jsx)(n.a,{href:"/terms/",children:"Terms & User Consent"})," documentation for further details."]}),"\n",(0,i.jsx)(n.h2,{id:"authentication-methods",children:"Authentication methods"}),"\n",(0,i.jsx)(n.p,{children:"The mod.io Unreal Engine plugin provides two ways for users to create an account to use the service: email authentication, and single sign-on (SSO) through an external authentication partner."}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsx)(n.p,{children:"For the best user experience, we recommend using SSO wherever possible."})}),"\n",(0,i.jsx)(n.h3,{id:"email-authentication",children:"Email authentication"}),"\n",(0,i.jsx)(n.p,{children:"Users can authenticate to mod.io with just an email address. This involves:"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Requesting a one-time authentication code be sent from mod.io to an email address provided by the user."}),"\n",(0,i.jsx)(n.li,{children:"Submitting the received code to mod.io."}),"\n"]}),"\n",(0,i.jsx)(n.h4,{id:"requesting-an-authentication-code",children:"Requesting an authentication code"}),"\n",(0,i.jsxs)(n.p,{children:["Request a one-time authentication code by calling ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_RequestEmailAuthCodeAsync",children:(0,i.jsx)(n.code,{children:"RequestEmailAuthCodeAsync"})}),". Pass in a user-provided email address as the first argument."]}),"\n",(0,i.jsxs)(a.A,{groupId:"languages",children:[(0,i.jsx)(r.A,{value:"cpp",label:"C++",default:!0,children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::RequestEmailAuthCode(const FString &EmailAddress)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->RequestEmailAuthCodeAsync(FModioEmailAddress(*EmailAddress), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnRequestEmailAuthCodeComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnRequestEmailAuthCodeComplete(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("Email auth code request complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n'})})}),(0,i.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"request_email_auth_code",src:t(98069).A+"",width:"1137",height:"597"})})})]}),"\n",(0,i.jsxs)(n.p,{children:["When your callback is invoked, inspect the ",(0,i.jsx)(n.code,{children:"ModioErrorCode"})," for a successful result. On success, prompt the user to check their emails and enter the authentication code they have received into your game."]}),"\n",(0,i.jsx)(n.h4,{id:"submitting-an-authentication-code",children:"Submitting an authentication code"}),"\n",(0,i.jsxs)(n.p,{children:["To finalize the user's authentication, submit the code they have entered to mod.io by calling ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_AuthenticateUserEmailAsync",children:(0,i.jsx)(n.code,{children:"AuthenticateUserEmailAsync"})}),". Pass the user's code in as the first argument."]}),"\n",(0,i.jsxs)(a.A,{groupId:"languages",children:[(0,i.jsx)(r.A,{value:"cpp",label:"C++",children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::AuthenticateUserEmail(const FString& AuthCode)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->AuthenticateUserEmailAsync(FModioEmailAuthCode(*AuthCode), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnAuthenticateUserEmailComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnAuthenticateUserEmailComplete(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("Email auth code submission complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n'})})}),(0,i.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"authenticate_user_email",src:t(534).A+"",width:"1170",height:"565"})})})]}),"\n",(0,i.jsxs)(n.p,{children:["When your callback is invoked, inspect the ",(0,i.jsx)(n.code,{children:"ModioErrorCode"})," for a successful result. On success, user authentication is complete."]}),"\n",(0,i.jsx)(n.h3,{id:"single-sign-on-authentication",children:"Single sign-on authentication"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine plugin features single sign-on (SSO) authentication from a number of ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_emodioauthenticationprovider",children:"external providers"}),". These include:"]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/gdk/#authentication",children:"Xbox Live"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/steam/authentication",children:"Steam"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/gog/authentication",children:"GOG Galaxy"})}),"\n",(0,i.jsx)(n.li,{children:"itch.io"}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/switch/#authentication",children:"Nintendo Switch"})}),"\n",(0,i.jsx)(n.li,{children:"Discord"}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/playstation/#authentication",children:"PlayStation\u2122Network"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/epic/authentication",children:"Epic Online Services"})}),"\n",(0,i.jsx)(n.li,{children:"Oculus"}),"\n",(0,i.jsx)(n.li,{children:"OpenID"}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/google/authentication",children:"Google"})}),"\n"]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["OpenID is a premium feature. If you are interested in mod.io premium features, please contact ",(0,i.jsx)(n.a,{href:"mailto:developers@mod.io",children:"developers@mod.io"}),"."]})}),"\n",(0,i.jsxs)(n.p,{children:["Each platform has their own requirements and prerequisites for performing SSO. Platform-specific authentication can be found in the respective ",(0,i.jsx)(n.a,{href:"/platforms/",children:"platform documentation"}),". SSO is performed by calling ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_AuthenticateUserExternalAsync",children:(0,i.jsx)(n.code,{children:"AuthenticateUserExternalAsync"})}),". This function takes a ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modioauthenticationparams",children:(0,i.jsx)(n.code,{children:"ModioAuthenticationParams"})})," containing provider-specific parameters, and a ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_emodioauthenticationprovider",children:(0,i.jsx)(n.code,{children:"ModioAuthenticationProvider"})})," indicating the provider you're using."]}),"\n",(0,i.jsxs)(n.p,{children:[(0,i.jsx)(n.a,{href:"#using-the-unreal-online-subsystem",children:"Unreal Engine's Online Subsystem"})," provides a convenient way to get a token for ",(0,i.jsx)(n.code,{children:"ModioAuthenticationParams.AuthToken"}),". ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#GetDefaultAuthProviderForCurrentPlatform",children:(0,i.jsx)(n.code,{children:"GetDefaultAuthProviderForCurrentPlatform"})})," can be used to get the default provider for the current platform."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsx)(n.p,{children:"The plugin will automatically URL encode parameters (such as the auth token) when making the request."})}),"\n",(0,i.jsxs)(a.A,{groupId:"languages",children:[(0,i.jsx)(r.A,{value:"cpp",label:"C++",children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::AuthenticateWithSSO(const FModioAuthenticationParams& AuthParams)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->AuthenticateUserExternalAsync(AuthParams, UModioPlatformHelpersLibrary::GetDefaultAuthProviderForCurrentPlatform(), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::AuthenticateUserExternalComplete));\n }\n}\n\nvoid UModioManagerSubsystem::AuthenticateUserExternalComplete(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("SSO complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n'})})}),(0,i.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"authenticate_user_email",src:t(66309).A+"",width:"1174",height:"681"})})})]}),"\n",(0,i.jsx)(n.h2,{id:"using-the-unreal-online-subsystem",children:"Using the Unreal Online Subsystem"}),"\n",(0,i.jsxs)(n.p,{children:["Since Unreal Engine 5.2, most of the authentication methods supported by mod.io are abstracted by Unreal's ",(0,i.jsx)(n.a,{href:"https://dev.epicgames.com/documentation/en-us/unreal-engine/API/Plugins/OnlineSubsystem/IOnlineSubsystem",children:"Online Subsystem"}),". Below are some code snippets demonstrating the flow to get a linked account token from Unreal's Online Subsystem."]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["Create a subsystem to manage mod.io initialization and authentication that depends on the ",(0,i.jsx)(n.code,{children:"ModioSubsystem"}),". During your subsystem's initialization, perform an Online Subsystem (OSS) login. This guarantees you will have a valid platform auth token ready when needed for mod.io SSO authentication."]}),"\n"]}),"\n",(0,i.jsx)(a.A,{"group-id":"languages",children:(0,i.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:"const IOnlineSubsystem* OnlineSubsystem = IOnlineSubsystem::GetByPlatform();\nif (OnlineSubsystem != nullptr && OnlineSubsystem->GetIdentityInterface() != nullptr)\n{\n // Get the Identity interface from our OnlineSubsystem and perform login\n const IOnlineIdentityPtr OnlineIdentity = OnlineSubsystem->GetIdentityInterface();\n OnlineIdentity->AddOnLoginCompleteDelegate_Handle(\n \t0, FOnLoginCompleteDelegate::CreateLambda(\n \t\t [this](int32 LocalUserNum, bool bWasSuccessful, const FUniqueNetId& UserId, const FString& Error) {\n \t\t\t this->OnSubsystemLoginComplete(LocalUserNum, bWasSuccessful, UserId, Error);\n \t\t }));\n OnlineIdentity->Login(0, FOnlineAccountCredentials());\n}\nelse\n{\n // If we don't have a valid OnlineSubsystem, just initialize the mod.io plugin\n InitializeModio();\n}\n"})})})}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"After a successful OSS login, request the linked account token."}),"\n"]}),"\n",(0,i.jsx)(a.A,{"group-id":"languages",children:(0,i.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::OnSubsystemLoginComplete(int32 LocalUserNum, bool bWasSuccessful, const FUniqueNetId& UserId, const FString& Error)\n{\n if (bWasSuccessful)\n {\n \tauto OnlineIdentity = IOnlineSubsystem::GetByPlatform()->GetIdentityInterface();\n \tOnlineIdentity->GetLinkedAccountAuthToken(0, FString(), IOnlineIdentity::FOnGetLinkedAccountAuthTokenCompleteDelegate::CreateUObject(this, &UModioManagerSubsystem::OnGetLinkedAccountTokenComplete)); \n }\n}\n"})})})}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["On success, the callback received from ",(0,i.jsx)(n.code,{children:"GetLinkedAccountAuthToken"})," will contain a token that you can use to perform SSO. This will contain either a byte array ",(0,i.jsx)(n.code,{children:"TokenData"})," (e.g. for Steam SSO) or ",(0,i.jsx)(n.code,{children:"TokenString"})," for a plaintext string. These need to be handled appropriately."]}),"\n"]}),"\n",(0,i.jsx)(a.A,{"group-id":"languages",children:(0,i.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::OnGetLinkedAccountTokenComplete(int LocalUserNum, bool bWasSuccessful, const FExternalAuthToken& AuthToken)\n{\n if (AuthToken.IsValid())\n {\n \tif (AuthToken.HasTokenData())\n \t{\n \t\tAuthTokenString = FBase64::Encode(AuthToken.TokenData);\t\t\n \t} \n \telse if (AuthToken.HasTokenString())\n \t{\n \t\tAuthTokenString = AuthToken.TokenString;\n \t}\n }\n}\n"})})})}),"\n",(0,i.jsx)(n.h2,{id:"token-lifetime--re-authentication",children:"Token Lifetime & Re-Authentication"}),"\n",(0,i.jsxs)(n.p,{children:["By default, tokens issued via email token exchange have a lifetime of 1 year. You can verify that a user has been successfully authenticated with ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#K2_VerifyUserAuthenticationAsync",children:(0,i.jsx)(n.code,{children:"VerifyUserAuthenticationAsync"})}),"."]}),"\n",(0,i.jsx)(n.p,{children:"If a user is not authenticated for any reason (e.g. their token has been invalidated by changing their password, or their profile data has been removed from their device) then you should prompt them to re-authenticate."})]})}function m(e={}){const{wrapper:n}={...(0,s.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},19365:(e,n,t)=>{t.d(n,{A:()=>r});t(96540);var i=t(18215);const s={tabItem:"tabItem_Ymn6"};var a=t(74848);function r(e){let{children:n,hidden:t,className:r}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,i.A)(s.tabItem,r),hidden:t,children:n})}},11470:(e,n,t)=>{t.d(n,{A:()=>S});var i=t(96540),s=t(18215),a=t(23104),r=t(56347),o=t(205),l=t(57485),u=t(31682),c=t(70679);function d(e){return i.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,i.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:n,children:t}=e;return(0,i.useMemo)((()=>{const e=n??function(e){return d(e).map((e=>{let{props:{value:n,label:t,attributes:i,default:s}}=e;return{value:n,label:t,attributes:i,default:s}}))}(t);return function(e){const n=(0,u.X)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function m(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function p(e){let{queryString:n=!1,groupId:t}=e;const s=(0,r.W6)(),a=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,l.aZ)(a),(0,i.useCallback)((e=>{if(!a)return;const n=new URLSearchParams(s.location.search);n.set(a,e),s.replace({...s.location,search:n.toString()})}),[a,s])]}function g(e){const{defaultValue:n,queryString:t=!1,groupId:s}=e,a=h(e),[r,l]=(0,i.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!m({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const i=t.find((e=>e.default))??t[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:n,tabValues:a}))),[u,d]=p({queryString:t,groupId:s}),[g,f]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[s,a]=(0,c.Dv)(t);return[s,(0,i.useCallback)((e=>{t&&a.set(e)}),[t,a])]}({groupId:s}),b=(()=>{const e=u??g;return m({value:e,tabValues:a})?e:null})();(0,o.A)((()=>{b&&l(b)}),[b]);return{selectedValue:r,selectValue:(0,i.useCallback)((e=>{if(!m({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),d(e),f(e)}),[d,f,a]),tabValues:a}}var f=t(92303);const b={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var x=t(74848);function y(e){let{className:n,block:t,selectedValue:i,selectValue:r,tabValues:o}=e;const l=[],{blockElementScrollPositionUntilNextRender:u}=(0,a.a_)(),c=e=>{const n=e.currentTarget,t=l.indexOf(n),s=o[t].value;s!==i&&(u(n),r(s))},d=e=>{let n=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const t=l.indexOf(e.currentTarget)+1;n=l[t]??l[0];break}case"ArrowLeft":{const t=l.indexOf(e.currentTarget)-1;n=l[t]??l[l.length-1];break}}n?.focus()};return(0,x.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,s.A)("tabs",{"tabs--block":t},n),children:o.map((e=>{let{value:n,label:t,attributes:a}=e;return(0,x.jsx)("li",{role:"tab",tabIndex:i===n?0:-1,"aria-selected":i===n,ref:e=>l.push(e),onKeyDown:d,onClick:c,...a,className:(0,s.A)("tabs__item",b.tabItem,a?.className,{"tabs__item--active":i===n}),children:t??n},n)}))})}function j(e){let{lazy:n,children:t,selectedValue:s}=e;const a=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=a.find((e=>e.props.value===s));return e?(0,i.cloneElement)(e,{className:"margin-top--md"}):null}return(0,x.jsx)("div",{className:"margin-top--md",children:a.map(((e,n)=>(0,i.cloneElement)(e,{key:n,hidden:e.props.value!==s})))})}function v(e){const n=g(e);return(0,x.jsxs)("div",{className:(0,s.A)("tabs-container",b.tabList),children:[(0,x.jsx)(y,{...n,...e}),(0,x.jsx)(j,{...n,...e})]})}function S(e){const n=(0,f.A)();return(0,x.jsx)(v,{...e,children:d(e.children)},String(n))}},89236:(e,n,t)=>{t.d(n,{A:()=>a});var i=t(19365),s=(t(96540),t(74848));function a(e){return(0,s.jsx)(s.Fragment,{children:(0,s.jsx)(i.A,{className:"tw-rounded-md",...e})})}},27064:(e,n,t)=>{t.d(n,{A:()=>a});var i=t(11470),s=(t(96540),t(74848));function a(e){return(0,s.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,s.jsx)(i.A,{...e})})}},534:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/authenticate_user_email-d2b0b6de82e9ce9673d7c15f3d96f372.png"},66309:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/authenticate_user_external-d0c4154affa4cf479cbe902c72445329.png"},98069:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/request_email_auth_code-8c01afd7910450ea8f357c12db8fbf78.png"},28453:(e,n,t)=>{t.d(n,{R:()=>r,x:()=>o});var i=t(96540);const s={},a=i.createContext(s);function r(e){const n=i.useContext(a);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function o(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:r(e.components),i.createElement(a.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/c5d35168.d3253ce3.js b/Doc/assets/js/c5d35168.d3253ce3.js new file mode 100644 index 00000000..f1f4220d --- /dev/null +++ b/Doc/assets/js/c5d35168.d3253ce3.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[8156],{69285:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>c,contentTitle:()=>l,default:()=>m,frontMatter:()=>o,metadata:()=>u,toc:()=>d});var i=t(74848),a=t(28453),s=t(27064),r=t(89236);const o={id:"ue-user-authentication",title:"User Authentication",slug:"/unreal/getting-started/user-authentication",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/user-authentication.mdx"},l=void 0,u={id:"getting-started/ue-user-authentication",title:"User Authentication",description:"Best practices",source:"@site/public/en-us/getting-started/user-authentication.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/user-authentication",permalink:"/unreal/getting-started/user-authentication",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/user-authentication.mdx",tags:[],version:"current",frontMatter:{id:"ue-user-authentication",title:"User Authentication",slug:"/unreal/getting-started/user-authentication",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/user-authentication.mdx"},sidebar:"sidebar",previous:{title:"Initialization & Teardown",permalink:"/unreal/getting-started/initialization"},next:{title:"Browsing Mods",permalink:"/unreal/getting-started/browsing-mods"}},c={},d=[{value:"Best practices",id:"best-practices",level:2},{value:"Recommended flow",id:"recommended-flow",level:3},{value:"Terms & user consent",id:"terms--user-consent",level:2},{value:"Authentication methods",id:"authentication-methods",level:2},{value:"Email authentication",id:"email-authentication",level:3},{value:"Requesting an authentication code",id:"requesting-an-authentication-code",level:4},{value:"Submitting an authentication code",id:"submitting-an-authentication-code",level:4},{value:"Single sign-on authentication",id:"single-sign-on-authentication",level:3},{value:"Using the Unreal Online Subsystem",id:"using-the-unreal-online-subsystem",level:2},{value:"Token Lifetime & Re-Authentication",id:"token-lifetime--re-authentication",level:2}];function h(e){const n={a:"a",admonition:"admonition",code:"code",h2:"h2",h3:"h3",h4:"h4",img:"img",li:"li",p:"p",pre:"pre",ul:"ul",...(0,a.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"best-practices",children:"Best practices"}),"\n",(0,i.jsxs)(n.p,{children:["We recommend creating a new game-instance based subsystem (e.g. ",(0,i.jsx)(n.code,{children:"ModioManagerSubsystem"}),") that depends on ",(0,i.jsx)(n.code,{children:"ModioSubsystem"}),", and is responsible for managing the full initialization and authentication flow for your game."]}),"\n",(0,i.jsx)(n.h3,{id:"recommended-flow",children:"Recommended flow"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Create a subsystem to manage mod.io initialization and authentication."}),"\n",(0,i.jsxs)(n.li,{children:["Perform ",(0,i.jsx)(n.a,{href:"#using-the-unreal-online-subsystem",children:"Online Subsystem (OSS) login"})," in your subsystem's ",(0,i.jsx)(n.code,{children:"Initialize"})," function to ensure you have a valid platform auth token ready when its time to authenticate to mod.io using ",(0,i.jsx)(n.a,{href:"#single-sign-on-authentication",children:"single sign-on (SSO)"}),".","\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Log in with the online identity provider."}),"\n",(0,i.jsx)(n.li,{children:"On success, get a linked account token for the current user."}),"\n",(0,i.jsx)(n.li,{children:"On success, get an authentication token from the linked account or from the online identity."}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:[(0,i.jsx)(n.a,{href:"initialization",children:"Initialize"})," the mod.io plugin with the required environment parameters for the current platform."]}),"\n",(0,i.jsxs)(n.li,{children:["Display the mod.io ",(0,i.jsx)(n.a,{href:"/terms/",children:"Terms of Service"}),". Ensure they are accepted by the user."]}),"\n",(0,i.jsxs)(n.li,{children:["Authenticate the user using ",(0,i.jsx)(n.a,{href:"#single-sign-on-authentication",children:"SSO"}),"."]}),"\n"]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsx)(n.p,{children:"If the Online Subsystem login fails at any point in the above process, fall back to simply initializing the plugin and offer email authentication as an alternative."})}),"\n",(0,i.jsx)(n.h2,{id:"terms--user-consent",children:"Terms & user consent"}),"\n",(0,i.jsxs)(n.p,{children:["Users must accept the mod.io Terms of Service before using any mod.io services. The Terms of Service can be fetched from mod.io using ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#gettermsofuseasync",children:(0,i.jsx)(n.code,{children:"GetTermsOfUseAsync"})}),". Before attempting authentication, ensure these terms are displayed to and accepted by the user."]}),"\n",(0,i.jsxs)(n.p,{children:["See our full ",(0,i.jsx)(n.a,{href:"/terms/",children:"Terms & User Consent"})," documentation for further details."]}),"\n",(0,i.jsx)(n.h2,{id:"authentication-methods",children:"Authentication methods"}),"\n",(0,i.jsx)(n.p,{children:"The mod.io Unreal Engine plugin provides two ways for users to create an account to use the service: email authentication, and single sign-on (SSO) through an external authentication partner."}),"\n",(0,i.jsx)(n.admonition,{type:"tip",children:(0,i.jsx)(n.p,{children:"For the best user experience, we recommend using SSO wherever possible."})}),"\n",(0,i.jsx)(n.h3,{id:"email-authentication",children:"Email authentication"}),"\n",(0,i.jsx)(n.p,{children:"Users can authenticate to mod.io with just an email address. This involves:"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"Requesting a one-time authentication code be sent from mod.io to an email address provided by the user."}),"\n",(0,i.jsx)(n.li,{children:"Submitting the received code to mod.io."}),"\n"]}),"\n",(0,i.jsx)(n.h4,{id:"requesting-an-authentication-code",children:"Requesting an authentication code"}),"\n",(0,i.jsxs)(n.p,{children:["Request a one-time authentication code by calling ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#requestemailauthcodeasync",children:(0,i.jsx)(n.code,{children:"RequestEmailAuthCodeAsync"})}),". Pass in a user-provided email address as the first argument."]}),"\n",(0,i.jsxs)(s.A,{groupId:"languages",children:[(0,i.jsx)(r.A,{value:"cpp",label:"C++",default:!0,children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::RequestEmailAuthCode(const FString &EmailAddress)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->RequestEmailAuthCodeAsync(FModioEmailAddress(*EmailAddress), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnRequestEmailAuthCodeComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnRequestEmailAuthCodeComplete(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("Email auth code request complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n'})})}),(0,i.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"request_email_auth_code",src:t(46524).A+"",width:"1137",height:"597"})})})]}),"\n",(0,i.jsxs)(n.p,{children:["When your callback is invoked, inspect the ",(0,i.jsx)(n.code,{children:"ModioErrorCode"})," for a successful result. On success, prompt the user to check their emails and enter the authentication code they have received into your game."]}),"\n",(0,i.jsx)(n.h4,{id:"submitting-an-authentication-code",children:"Submitting an authentication code"}),"\n",(0,i.jsxs)(n.p,{children:["To finalize the user's authentication, submit the code they have entered to mod.io by calling ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#authenticateuseremailasync",children:(0,i.jsx)(n.code,{children:"AuthenticateUserEmailAsync"})}),". Pass the user's code in as the first argument."]}),"\n",(0,i.jsxs)(s.A,{groupId:"languages",children:[(0,i.jsx)(r.A,{value:"cpp",label:"C++",children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::AuthenticateUserEmail(const FString& AuthCode)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->AuthenticateUserEmailAsync(FModioEmailAuthCode(*AuthCode), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnAuthenticateUserEmailComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnAuthenticateUserEmailComplete(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("Email auth code submission complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n'})})}),(0,i.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"authenticate_user_email",src:t(73855).A+"",width:"1170",height:"565"})})})]}),"\n",(0,i.jsxs)(n.p,{children:["When your callback is invoked, inspect the ",(0,i.jsx)(n.code,{children:"ModioErrorCode"})," for a successful result. On success, user authentication is complete."]}),"\n",(0,i.jsx)(n.h3,{id:"single-sign-on-authentication",children:"Single sign-on authentication"}),"\n",(0,i.jsxs)(n.p,{children:["The mod.io Unreal Engine plugin features single sign-on (SSO) authentication from a number of ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#EModioAuthenticationProvider",children:"external providers"}),". These include:"]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/gdk/#authentication",children:"Xbox Live"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/steam/authentication",children:"Steam"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/gog/authentication",children:"GOG Galaxy"})}),"\n",(0,i.jsx)(n.li,{children:"itch.io"}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/switch/#authentication",children:"Nintendo Switch"})}),"\n",(0,i.jsx)(n.li,{children:"Discord"}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/playstation/#authentication",children:"PlayStation\u2122Network"})}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/epic/authentication",children:"Epic Online Services"})}),"\n",(0,i.jsx)(n.li,{children:"Oculus"}),"\n",(0,i.jsx)(n.li,{children:"OpenID"}),"\n",(0,i.jsx)(n.li,{children:(0,i.jsx)(n.a,{href:"/platforms/google/authentication",children:"Google"})}),"\n"]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["OpenID is a premium feature. If you are interested in mod.io premium features, please contact ",(0,i.jsx)(n.a,{href:"mailto:developers@mod.io",children:"developers@mod.io"}),"."]})}),"\n",(0,i.jsxs)(n.p,{children:["Each platform has their own requirements and prerequisites for performing SSO. Platform-specific authentication can be found in the respective ",(0,i.jsx)(n.a,{href:"/platforms/",children:"platform documentation"}),". SSO is performed by calling ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#authenticateuserexternalasync",children:(0,i.jsx)(n.code,{children:"AuthenticateUserExternalAsync"})}),". This function takes a ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#modioauthenticationparams",children:(0,i.jsx)(n.code,{children:"ModioAuthenticationParams"})})," containing provider-specific parameters, and a ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#EModioAuthenticationProvider",children:(0,i.jsx)(n.code,{children:"ModioAuthenticationProvider"})})," indicating the provider you're using."]}),"\n",(0,i.jsxs)(n.p,{children:[(0,i.jsx)(n.a,{href:"#using-the-unreal-online-subsystem",children:"Unreal Engine's Online Subsystem"})," provides a convenient way to get a token for ",(0,i.jsx)(n.code,{children:"ModioAuthenticationParams.AuthToken"}),". ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#get-default-auth-provider-for-current-platform",children:(0,i.jsx)(n.code,{children:"GetDefaultAuthProviderForCurrentPlatform"})})," can be used to get the default provider for the current platform."]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsx)(n.p,{children:"The plugin will automatically URL encode parameters (such as the auth token) when making the request."})}),"\n",(0,i.jsxs)(s.A,{groupId:"languages",children:[(0,i.jsx)(r.A,{value:"cpp",label:"C++",children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:'void UModioManagerSubsystem::AuthenticateWithSSO(const FModioAuthenticationParams& AuthParams)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->AuthenticateUserExternalAsync(AuthParams, UModioPlatformHelpersLibrary::GetDefaultAuthProviderForCurrentPlatform(), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::AuthenticateUserExternalComplete));\n }\n}\n\nvoid UModioManagerSubsystem::AuthenticateUserExternalComplete(FModioErrorCode ErrorCode)\n{\n UE_LOG(LogModioGame, Log, TEXT("SSO complete with result: %s"), *ErrorCode.GetErrorMessage());\n}\n'})})}),(0,i.jsx)(r.A,{value:"blueprint",label:"Blueprint",children:(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"authenticate_user_email",src:t(47042).A+"",width:"1174",height:"681"})})})]}),"\n",(0,i.jsx)(n.h2,{id:"using-the-unreal-online-subsystem",children:"Using the Unreal Online Subsystem"}),"\n",(0,i.jsxs)(n.p,{children:["Since Unreal Engine 5.2, most of the authentication methods supported by mod.io are abstracted by Unreal's ",(0,i.jsx)(n.a,{href:"https://dev.epicgames.com/documentation/en-us/unreal-engine/API/Plugins/OnlineSubsystem/IOnlineSubsystem",children:"Online Subsystem"}),". Below are some code snippets demonstrating the flow to get a linked account token from Unreal's Online Subsystem."]}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["Create a subsystem to manage mod.io initialization and authentication that depends on the ",(0,i.jsx)(n.code,{children:"ModioSubsystem"}),". During your subsystem's initialization, perform an Online Subsystem (OSS) login. This guarantees you will have a valid platform auth token ready when needed for mod.io SSO authentication."]}),"\n"]}),"\n",(0,i.jsx)(s.A,{"group-id":"languages",children:(0,i.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:"const IOnlineSubsystem* OnlineSubsystem = IOnlineSubsystem::GetByPlatform();\nif (OnlineSubsystem != nullptr && OnlineSubsystem->GetIdentityInterface() != nullptr)\n{\n // Get the Identity interface from our OnlineSubsystem and perform login\n const IOnlineIdentityPtr OnlineIdentity = OnlineSubsystem->GetIdentityInterface();\n OnlineIdentity->AddOnLoginCompleteDelegate_Handle(\n \t0, FOnLoginCompleteDelegate::CreateLambda(\n \t\t [this](int32 LocalUserNum, bool bWasSuccessful, const FUniqueNetId& UserId, const FString& Error) {\n \t\t\t this->OnSubsystemLoginComplete(LocalUserNum, bWasSuccessful, UserId, Error);\n \t\t }));\n OnlineIdentity->Login(0, FOnlineAccountCredentials());\n}\nelse\n{\n // If we don't have a valid OnlineSubsystem, just initialize the mod.io plugin\n InitializeModio();\n}\n"})})})}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsx)(n.li,{children:"After a successful OSS login, request the linked account token."}),"\n"]}),"\n",(0,i.jsx)(s.A,{"group-id":"languages",children:(0,i.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::OnSubsystemLoginComplete(int32 LocalUserNum, bool bWasSuccessful, const FUniqueNetId& UserId, const FString& Error)\n{\n if (bWasSuccessful)\n {\n \tauto OnlineIdentity = IOnlineSubsystem::GetByPlatform()->GetIdentityInterface();\n \tOnlineIdentity->GetLinkedAccountAuthToken(0, FString(), IOnlineIdentity::FOnGetLinkedAccountAuthTokenCompleteDelegate::CreateUObject(this, &UModioManagerSubsystem::OnGetLinkedAccountTokenComplete)); \n }\n}\n"})})})}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["On success, the callback received from ",(0,i.jsx)(n.code,{children:"GetLinkedAccountAuthToken"})," will contain a token that you can use to perform SSO. This will contain either a byte array ",(0,i.jsx)(n.code,{children:"TokenData"})," (e.g. for Steam SSO) or ",(0,i.jsx)(n.code,{children:"TokenString"})," for a plaintext string. These need to be handled appropriately."]}),"\n"]}),"\n",(0,i.jsx)(s.A,{"group-id":"languages",children:(0,i.jsx)(r.A,{value:"c++",label:"C++",default:!0,children:(0,i.jsx)(n.pre,{children:(0,i.jsx)(n.code,{className:"language-cpp",children:"void UModioManagerSubsystem::OnGetLinkedAccountTokenComplete(int LocalUserNum, bool bWasSuccessful, const FExternalAuthToken& AuthToken)\n{\n if (AuthToken.IsValid())\n {\n \tif (AuthToken.HasTokenData())\n \t{\n \t\tAuthTokenString = FBase64::Encode(AuthToken.TokenData);\t\t\n \t} \n \telse if (AuthToken.HasTokenString())\n \t{\n \t\tAuthTokenString = AuthToken.TokenString;\n \t}\n }\n}\n"})})})}),"\n",(0,i.jsx)(n.h2,{id:"token-lifetime--re-authentication",children:"Token Lifetime & Re-Authentication"}),"\n",(0,i.jsxs)(n.p,{children:["By default, tokens issued via email token exchange have a lifetime of 1 year. You can verify that a user has been successfully authenticated with ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#verifyuserauthenticationasync",children:(0,i.jsx)(n.code,{children:"VerifyUserAuthenticationAsync"})}),"."]}),"\n",(0,i.jsx)(n.p,{children:"If a user is not authenticated for any reason (e.g. their token has been invalidated by changing their password, or their profile data has been removed from their device) then you should prompt them to re-authenticate."})]})}function m(e={}){const{wrapper:n}={...(0,a.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},19365:(e,n,t)=>{t.d(n,{A:()=>r});t(96540);var i=t(18215);const a={tabItem:"tabItem_Ymn6"};var s=t(74848);function r(e){let{children:n,hidden:t,className:r}=e;return(0,s.jsx)("div",{role:"tabpanel",className:(0,i.A)(a.tabItem,r),hidden:t,children:n})}},11470:(e,n,t)=>{t.d(n,{A:()=>S});var i=t(96540),a=t(18215),s=t(23104),r=t(56347),o=t(205),l=t(57485),u=t(31682),c=t(70679);function d(e){return i.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,i.isValidElement)(e)&&function(e){const{props:n}=e;return!!n&&"object"==typeof n&&"value"in n}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:n,children:t}=e;return(0,i.useMemo)((()=>{const e=n??function(e){return d(e).map((e=>{let{props:{value:n,label:t,attributes:i,default:a}}=e;return{value:n,label:t,attributes:i,default:a}}))}(t);return function(e){const n=(0,u.X)(e,((e,n)=>e.value===n.value));if(n.length>0)throw new Error(`Docusaurus error: Duplicate values "${n.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[n,t])}function m(e){let{value:n,tabValues:t}=e;return t.some((e=>e.value===n))}function g(e){let{queryString:n=!1,groupId:t}=e;const a=(0,r.W6)(),s=function(e){let{queryString:n=!1,groupId:t}=e;if("string"==typeof n)return n;if(!1===n)return null;if(!0===n&&!t)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return t??null}({queryString:n,groupId:t});return[(0,l.aZ)(s),(0,i.useCallback)((e=>{if(!s)return;const n=new URLSearchParams(a.location.search);n.set(s,e),a.replace({...a.location,search:n.toString()})}),[s,a])]}function p(e){const{defaultValue:n,queryString:t=!1,groupId:a}=e,s=h(e),[r,l]=(0,i.useState)((()=>function(e){let{defaultValue:n,tabValues:t}=e;if(0===t.length)throw new Error("Docusaurus error: the component requires at least one children component");if(n){if(!m({value:n,tabValues:t}))throw new Error(`Docusaurus error: The has a defaultValue "${n}" but none of its children has the corresponding value. Available values are: ${t.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return n}const i=t.find((e=>e.default))??t[0];if(!i)throw new Error("Unexpected error: 0 tabValues");return i.value}({defaultValue:n,tabValues:s}))),[u,d]=g({queryString:t,groupId:a}),[p,f]=function(e){let{groupId:n}=e;const t=function(e){return e?`docusaurus.tab.${e}`:null}(n),[a,s]=(0,c.Dv)(t);return[a,(0,i.useCallback)((e=>{t&&s.set(e)}),[t,s])]}({groupId:a}),b=(()=>{const e=u??p;return m({value:e,tabValues:s})?e:null})();(0,o.A)((()=>{b&&l(b)}),[b]);return{selectedValue:r,selectValue:(0,i.useCallback)((e=>{if(!m({value:e,tabValues:s}))throw new Error(`Can't select invalid tab value=${e}`);l(e),d(e),f(e)}),[d,f,s]),tabValues:s}}var f=t(92303);const b={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var x=t(74848);function y(e){let{className:n,block:t,selectedValue:i,selectValue:r,tabValues:o}=e;const l=[],{blockElementScrollPositionUntilNextRender:u}=(0,s.a_)(),c=e=>{const n=e.currentTarget,t=l.indexOf(n),a=o[t].value;a!==i&&(u(n),r(a))},d=e=>{let n=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const t=l.indexOf(e.currentTarget)+1;n=l[t]??l[0];break}case"ArrowLeft":{const t=l.indexOf(e.currentTarget)-1;n=l[t]??l[l.length-1];break}}n?.focus()};return(0,x.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,a.A)("tabs",{"tabs--block":t},n),children:o.map((e=>{let{value:n,label:t,attributes:s}=e;return(0,x.jsx)("li",{role:"tab",tabIndex:i===n?0:-1,"aria-selected":i===n,ref:e=>l.push(e),onKeyDown:d,onClick:c,...s,className:(0,a.A)("tabs__item",b.tabItem,s?.className,{"tabs__item--active":i===n}),children:t??n},n)}))})}function j(e){let{lazy:n,children:t,selectedValue:a}=e;const s=(Array.isArray(t)?t:[t]).filter(Boolean);if(n){const e=s.find((e=>e.props.value===a));return e?(0,i.cloneElement)(e,{className:"margin-top--md"}):null}return(0,x.jsx)("div",{className:"margin-top--md",children:s.map(((e,n)=>(0,i.cloneElement)(e,{key:n,hidden:e.props.value!==a})))})}function v(e){const n=p(e);return(0,x.jsxs)("div",{className:(0,a.A)("tabs-container",b.tabList),children:[(0,x.jsx)(y,{...n,...e}),(0,x.jsx)(j,{...n,...e})]})}function S(e){const n=(0,f.A)();return(0,x.jsx)(v,{...e,children:d(e.children)},String(n))}},89236:(e,n,t)=>{t.d(n,{A:()=>s});var i=t(19365),a=(t(96540),t(74848));function s(e){return(0,a.jsx)(a.Fragment,{children:(0,a.jsx)(i.A,{className:"tw-rounded-md",...e})})}},27064:(e,n,t)=>{t.d(n,{A:()=>s});var i=t(11470),a=(t(96540),t(74848));function s(e){return(0,a.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,a.jsx)(i.A,{...e})})}},73855:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/authenticate_user_email-d2b0b6de82e9ce9673d7c15f3d96f372.png"},47042:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/authenticate_user_external-d0c4154affa4cf479cbe902c72445329.png"},46524:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/request_email_auth_code-8c01afd7910450ea8f357c12db8fbf78.png"},28453:(e,n,t)=>{t.d(n,{R:()=>r,x:()=>o});var i=t(96540);const a={},s=i.createContext(a);function r(e){const n=i.useContext(s);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function o(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(a):e.components||a:r(e.components),i.createElement(s.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/cddf829a.11002d8d.js b/Doc/assets/js/cddf829a.11002d8d.js deleted file mode 100644 index 4812878a..00000000 --- a/Doc/assets/js/cddf829a.11002d8d.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[8840],{64088:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>c,contentTitle:()=>l,default:()=>m,frontMatter:()=>i,metadata:()=>u,toc:()=>d});var s=n(74848),r=n(28453),a=n(27064),o=n(89236);const i={id:"ue-monetization",title:"Monetization",slug:"/unreal/getting-started/monetization",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/monetization.mdx"},l=void 0,u={id:"getting-started/ue-monetization",title:"Monetization",description:"The mod.io Unreal Engine Plugin supports a range of monetization features, allowing you to sell a per-game virtual currency to your players that they can use to purchase mods, with a share of the revenue split between creators and your studio. An overview of the mod.io monetization system is available here.",source:"@site/public/en-us/getting-started/monetization.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/monetization",permalink:"/unreal/getting-started/monetization",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/monetization.mdx",tags:[],version:"current",frontMatter:{id:"ue-monetization",title:"Monetization",slug:"/unreal/getting-started/monetization",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/monetization.mdx"},sidebar:"sidebar",previous:{title:"Error Handling",permalink:"/unreal/getting-started/error-handling"},next:{title:"Temporary Mod Sets",permalink:"/unreal/getting-started/temporary-mods"}},c={},d=[{value:"Initialization",id:"initialization",level:2},{value:"Getting the user's wallet",id:"getting-the-users-wallet",level:2},{value:"Querying mods",id:"querying-mods",level:2},{value:"Purchasing mods",id:"purchasing-mods",level:2},{value:"Showing user purchases",id:"showing-user-purchases",level:2},{value:"Getting a user delegation token",id:"getting-a-user-delegation-token",level:2}];function h(e){const t={a:"a",code:"code",h2:"h2",img:"img",p:"p",pre:"pre",...(0,r.R)(),...e.components};return(0,s.jsxs)(s.Fragment,{children:[(0,s.jsxs)(t.p,{children:["The mod.io Unreal Engine Plugin supports a range of monetization features, allowing you to sell a per-game virtual currency to your players that they can use to purchase mods, with a share of the revenue split between creators and your studio. An overview of the mod.io monetization system is available ",(0,s.jsx)(t.a,{href:"/monetization/",children:"here"}),"."]}),"\n",(0,s.jsxs)(t.p,{children:["Every platform requires specific setup for monetization features to work, particularly with respect to the virtual currency configuration and API calls. The following documentation is generically applicable. Platform-specific information is available in the relevant ",(0,s.jsx)(t.a,{href:"/platforms/",children:"platform documentation section"}),"."]}),"\n",(0,s.jsx)(t.h2,{id:"initialization",children:"Initialization"}),"\n",(0,s.jsxs)(t.p,{children:["The mod.io monetization features are enabled during the onboarding process on your ",(0,s.jsx)(t.a,{href:"https://mod.io/g",children:"game profile"}),"."]}),"\n",(0,s.jsxs)(t.p,{children:["Ensure that you have set the appropriate ",(0,s.jsx)(t.code,{children:"Portal"})," when ",(0,s.jsx)(t.a,{href:"initialization",children:"initializing the plugin"}),". For instance, on Steam you must initialize with ",(0,s.jsx)(t.code,{children:"EModioPortal::Steam"})," to redeem entitlements for Steam."]}),"\n",(0,s.jsx)(t.h2,{id:"getting-the-users-wallet",children:"Getting the user's wallet"}),"\n",(0,s.jsxs)(t.p,{children:["On startup, you can make a call to ",(0,s.jsx)(t.code,{children:"UModioSubsystem::GetUserWalletBalanceAsync"})," to get the balance of the current user's wallet. If no wallet exists for the user, one will be automatically created for them. This call returns the user's wallet balance for the current game."]}),"\n",(0,s.jsxs)(t.p,{children:["The only time you need to make this call is on start-up. We recommend that you cache the value of this result in your game code rather than making consistent calls to ",(0,s.jsx)(t.code,{children:"UModioSubsystem::GetUserWalletBalanceAsync"}),", and update your local state from the return values of other calls that affect wallet balance."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(t.p,{children:(0,s.jsx)(t.img,{alt:"get_user_wallet",src:n(80814).A+"",width:"598",height:"414"})})}),(0,s.jsx)(o.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(t.pre,{children:(0,s.jsx)(t.code,{className:"language-cpp",children:"void UModioManagerSubsystem::GetUserWallet()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->GetUserWalletBalanceAsync(FOnGetUserWalletBalanceDelegate::CreateUObject(this, &UModioManagerSubsystem::OnGetUserWalletComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnGetUserWalletComplete(FModioErrorCode ErrorCode, FModioOptionalUInt64 WalletBalance)\n{\n if (!ErrorCode)\n {\n \t// Wallet balance successfully retrieved\n }\n}\n"})})})]}),"\n",(0,s.jsx)(t.h2,{id:"querying-mods",children:"Querying mods"}),"\n",(0,s.jsxs)(t.p,{children:["As part ",(0,s.jsx)(t.code,{children:"UModioSubsystem::ListAllModsAsync"}),", you can include an additional filter for whether you list paid mods. By default, only free mods are shown. Set ",(0,s.jsx)(t.code,{children:"RevenueType"})," on the ",(0,s.jsx)(t.code,{children:"ModioFilterParams"})," object passed to ",(0,s.jsx)(t.code,{children:"UModioSubsystem::ListAllModsAsync"})," to include free and paid content, or just paid content. All mods returned will have a ",(0,s.jsx)(t.code,{children:"Price"})," property, indicating the virtual currency price that must be paid to purchase that mod."]}),"\n",(0,s.jsx)(t.p,{children:"Filtering for Paid/Unpaid content is not currently exposed to Blueprint."}),"\n",(0,s.jsx)(t.h2,{id:"purchasing-mods",children:"Purchasing mods"}),"\n",(0,s.jsxs)(t.p,{children:["Call ",(0,s.jsx)(t.code,{children:"UModioSubsystem::PurchaseModAsync"})," to purchase a specified mod. ",(0,s.jsx)(t.code,{children:"PurchaseModAsync"})," takes two parameters: the ",(0,s.jsx)(t.code,{children:"ModioModID"})," of the mod to purchase, and the ",(0,s.jsx)(t.code,{children:"ExpectedPrice"}),", which is the price displayed to the user from ",(0,s.jsx)(t.code,{children:"UModioSubsystem::ListAllModsAsync"}),". You must include this parameter for safety so the user is not charged more or less than the price displayed to them, in case the price of the mod has changed between the call to ",(0,s.jsx)(t.code,{children:"ListAllModsAsync"})," and purchase time."]}),"\n",(0,s.jsx)(t.p,{children:"Once a mod is purchased, it is automatically subscribed to for the user."}),"\n",(0,s.jsxs)(t.p,{children:["You should validate that the user has enough virtual currency to make the purchase by comparing it to the balance you received from ",(0,s.jsx)(t.code,{children:"UModioSubsystem::GetUserWalletBalanceAsync"}),". This is purely for user experience (e.g. for graying out the purchase button in the UI, or upselling the user a virtual currency pack); ",(0,s.jsx)(t.code,{children:"UModioSubsystem::PurchaseModAsync"})," will return an error if the user has insufficient funds in their wallet."]}),"\n",(0,s.jsxs)(t.p,{children:["An updated wallet balance (with the purchase amount subtracted) is returned in the callback of ",(0,s.jsx)(t.code,{children:"UModioSubsystem::PurchaseModAsync"}),"."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(t.p,{children:(0,s.jsx)(t.img,{alt:"purchase_mod",src:n(15154).A+"",width:"617",height:"442"})})}),(0,s.jsx)(o.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(t.pre,{children:(0,s.jsx)(t.code,{className:"language-cpp",children:"void UModioManagerSubsystem::PurchaseMod(FModioModID ModId, FModioUnsigned64 ExpectedPrice)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->PurchaseModAsync(ModId, ExpectedPrice, FOnPurchaseModDelegate::CreateUObject(this, &UModioManagerSubsystem::OnPurchaseModComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnPurchaseModComplete(FModioErrorCode ErrorCode, FModioOptionalTransactionRecord Transaction)\n{\n if (!ErrorCode)\n {\n \t// Mod purchase successful\n }\n}\n"})})})]}),"\n",(0,s.jsx)(t.h2,{id:"showing-user-purchases",children:"Showing user purchases"}),"\n",(0,s.jsxs)(t.p,{children:["While purchased mods are automatically subscribed and installed at purchase time, the user can freely unsubscribe and uninstall purchased mods and they and they will remain owned and purchased by the user. They must re-subscribe to the mod to have it re-installed. Use ",(0,s.jsx)(t.code,{children:"UModioSubsystem::FetchUserPurchasesAsync"})," to fetch an updated list of a user's purchased mods from the server. After a successful call, you can then display the user's purchased mods with ",(0,s.jsx)(t.code,{children:"UModioSubsystem::QueryUserPurchasedMods"}),", allowing re-subscription as desired."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(t.p,{children:(0,s.jsx)(t.img,{alt:"show_user_purchases",src:n(97910).A+"",width:"580",height:"481"})})}),(0,s.jsx)(o.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(t.pre,{children:(0,s.jsx)(t.code,{className:"language-cpp",children:"void UModioManagerSubsystem::FetchUserPurchases()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->FetchUserPurchasesAsync(FOnFetchUserPurchasesDelegate::CreateUObject(this, &UModioManagerSubsystem::OnFetchUserPurchasesComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnFetchUserPurchasesComplete(FModioErrorCode ErrorCode)\n{\n if (!ErrorCode)\n {\n \t// Purchases Successfully Fetched\n \tif (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n \t{\n \t\t// We can now access the list of purchased mods directly\n \t\tTMap PurchasedMods = Subsystem->QueryUserPurchasedMods();\n \t}\n }\n}\n"})})})]}),"\n",(0,s.jsx)(t.h2,{id:"getting-a-user-delegation-token",children:"Getting a user delegation token"}),"\n",(0,s.jsxs)(t.p,{children:["User delegation tokens can be used by a backend server for S2S (Server to Server) transactions. You can get one for the current user by calling ",(0,s.jsx)(t.code,{children:"UModioSubsystem::GetUserDelegationTokenAsync"}),", the callback for which contains the token as an ",(0,s.jsx)(t.code,{children:"FString"}),"."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(t.p,{children:(0,s.jsx)(t.img,{alt:"get_user_delegation_token",src:n(94773).A+"",width:"670",height:"448"})})}),(0,s.jsx)(o.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(t.pre,{children:(0,s.jsx)(t.code,{className:"language-cpp",children:"void UModioManagerSubsystem::GetUserDelegationToken()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->GetUserDelegationTokenAsync(FOnGetUserDelegationTokenDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnGetUserDelegationTokenCallback));\n }\n}\n\nvoid UModioManagerSubsystem::OnGetUserDelegationTokenCallback(FModioErrorCode ErrorCode, FString UserDelegationToken)\n{\n if (!ErrorCode)\n {\n \t// Successfully got User Delegation Token\n }\n}\n"})})})]})]})}function m(e={}){const{wrapper:t}={...(0,r.R)(),...e.components};return t?(0,s.jsx)(t,{...e,children:(0,s.jsx)(h,{...e})}):h(e)}},19365:(e,t,n)=>{n.d(t,{A:()=>o});n(96540);var s=n(18215);const r={tabItem:"tabItem_Ymn6"};var a=n(74848);function o(e){let{children:t,hidden:n,className:o}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,s.A)(r.tabItem,o),hidden:n,children:t})}},11470:(e,t,n)=>{n.d(t,{A:()=>M});var s=n(96540),r=n(18215),a=n(23104),o=n(56347),i=n(205),l=n(57485),u=n(31682),c=n(70679);function d(e){return s.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,s.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:t,children:n}=e;return(0,s.useMemo)((()=>{const e=t??function(e){return d(e).map((e=>{let{props:{value:t,label:n,attributes:s,default:r}}=e;return{value:t,label:n,attributes:s,default:r}}))}(n);return function(e){const t=(0,u.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function p(e){let{queryString:t=!1,groupId:n}=e;const r=(0,o.W6)(),a=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l.aZ)(a),(0,s.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(r.location.search);t.set(a,e),r.replace({...r.location,search:t.toString()})}),[a,r])]}function g(e){const{defaultValue:t,queryString:n=!1,groupId:r}=e,a=h(e),[o,l]=(0,s.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const s=n.find((e=>e.default))??n[0];if(!s)throw new Error("Unexpected error: 0 tabValues");return s.value}({defaultValue:t,tabValues:a}))),[u,d]=p({queryString:n,groupId:r}),[g,b]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[r,a]=(0,c.Dv)(n);return[r,(0,s.useCallback)((e=>{n&&a.set(e)}),[n,a])]}({groupId:r}),f=(()=>{const e=u??g;return m({value:e,tabValues:a})?e:null})();(0,i.A)((()=>{f&&l(f)}),[f]);return{selectedValue:o,selectValue:(0,s.useCallback)((e=>{if(!m({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),d(e),b(e)}),[d,b,a]),tabValues:a}}var b=n(92303);const f={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var y=n(74848);function v(e){let{className:t,block:n,selectedValue:s,selectValue:o,tabValues:i}=e;const l=[],{blockElementScrollPositionUntilNextRender:u}=(0,a.a_)(),c=e=>{const t=e.currentTarget,n=l.indexOf(t),r=i[n].value;r!==s&&(u(t),o(r))},d=e=>{let t=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const n=l.indexOf(e.currentTarget)+1;t=l[n]??l[0];break}case"ArrowLeft":{const n=l.indexOf(e.currentTarget)-1;t=l[n]??l[l.length-1];break}}t?.focus()};return(0,y.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.A)("tabs",{"tabs--block":n},t),children:i.map((e=>{let{value:t,label:n,attributes:a}=e;return(0,y.jsx)("li",{role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,ref:e=>l.push(e),onKeyDown:d,onClick:c,...a,className:(0,r.A)("tabs__item",f.tabItem,a?.className,{"tabs__item--active":s===t}),children:n??t},t)}))})}function x(e){let{lazy:t,children:n,selectedValue:r}=e;const a=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===r));return e?(0,s.cloneElement)(e,{className:"margin-top--md"}):null}return(0,y.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,s.cloneElement)(e,{key:t,hidden:e.props.value!==r})))})}function j(e){const t=g(e);return(0,y.jsxs)("div",{className:(0,r.A)("tabs-container",f.tabList),children:[(0,y.jsx)(v,{...t,...e}),(0,y.jsx)(x,{...t,...e})]})}function M(e){const t=(0,b.A)();return(0,y.jsx)(j,{...e,children:d(e.children)},String(t))}},89236:(e,t,n)=>{n.d(t,{A:()=>a});var s=n(19365),r=(n(96540),n(74848));function a(e){return(0,r.jsx)(r.Fragment,{children:(0,r.jsx)(s.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,n)=>{n.d(t,{A:()=>a});var s=n(11470),r=(n(96540),n(74848));function a(e){return(0,r.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,r.jsx)(s.A,{...e})})}},94773:(e,t,n)=>{n.d(t,{A:()=>s});const s=n.p+"assets/images/get_user_delegation_token-9ae69e059f28541bbac934ea02b8f994.png"},80814:(e,t,n)=>{n.d(t,{A:()=>s});const s=n.p+"assets/images/get_user_wallet-3e8cab6076988d8cc7469c3183f8711b.png"},15154:(e,t,n)=>{n.d(t,{A:()=>s});const s=n.p+"assets/images/purchase_mod-d1e2b2ee8586f08eada43e87167ee125.png"},97910:(e,t,n)=>{n.d(t,{A:()=>s});const s=n.p+"assets/images/show_user_purchases-575ecaef025b90bbf3ea30f2c0437cfe.png"},28453:(e,t,n)=>{n.d(t,{R:()=>o,x:()=>i});var s=n(96540);const r={},a=s.createContext(r);function o(e){const t=s.useContext(a);return s.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function i(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:o(e.components),s.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/cddf829a.6ee7539f.js b/Doc/assets/js/cddf829a.6ee7539f.js new file mode 100644 index 00000000..6f696b4f --- /dev/null +++ b/Doc/assets/js/cddf829a.6ee7539f.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[8840],{64088:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>c,contentTitle:()=>l,default:()=>m,frontMatter:()=>i,metadata:()=>d,toc:()=>u});var s=n(74848),r=n(28453),a=n(27064),o=n(89236);const i={id:"ue-monetization",title:"Monetization",slug:"/unreal/getting-started/monetization",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/monetization.mdx"},l=void 0,d={id:"getting-started/ue-monetization",title:"Monetization",description:"The mod.io Unreal Engine Plugin supports a range of monetization features, allowing you to sell a per-game virtual currency to your players that they can use to purchase mods, with a share of the revenue split between creators and your studio. An overview of the mod.io monetization system is available here.",source:"@site/public/en-us/getting-started/monetization.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/monetization",permalink:"/unreal/getting-started/monetization",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/monetization.mdx",tags:[],version:"current",frontMatter:{id:"ue-monetization",title:"Monetization",slug:"/unreal/getting-started/monetization",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/monetization.mdx"},sidebar:"sidebar",previous:{title:"Error Handling",permalink:"/unreal/getting-started/error-handling"},next:{title:"Temporary Mod Sets",permalink:"/unreal/getting-started/temporary-mods"}},c={},u=[{value:"Initialization",id:"initialization",level:2},{value:"Getting the user's wallet",id:"getting-the-users-wallet",level:2},{value:"Querying mods",id:"querying-mods",level:2},{value:"Purchasing mods",id:"purchasing-mods",level:2},{value:"Showing user purchases",id:"showing-user-purchases",level:2},{value:"Getting a user delegation token",id:"getting-a-user-delegation-token",level:2},{value:"Listing and Purchasing Premium Currency",id:"listing-and-purchasing-premium-currency",level:2},{value:"Displaying the Native Platform Store",id:"displaying-the-native-platform-store",level:3},{value:"Displaying Component UI Store Offers",id:"displaying-component-ui-store-offers",level:3},{value:"Consuming User Entitlements",id:"consuming-user-entitlements",level:2}];function h(e){const t={a:"a",code:"code",h2:"h2",h3:"h3",img:"img",p:"p",pre:"pre",...(0,r.R)(),...e.components};return(0,s.jsxs)(s.Fragment,{children:[(0,s.jsxs)(t.p,{children:["The mod.io Unreal Engine Plugin supports a range of monetization features, allowing you to sell a per-game virtual currency to your players that they can use to purchase mods, with a share of the revenue split between creators and your studio. An overview of the mod.io monetization system is available ",(0,s.jsx)(t.a,{href:"/monetization/",children:"here"}),"."]}),"\n",(0,s.jsxs)(t.p,{children:["Every platform requires specific setup for monetization features to work, particularly with respect to the virtual currency configuration and API calls. The following documentation is generically applicable. Platform-specific information is available in the relevant ",(0,s.jsx)(t.a,{href:"/platforms/",children:"platform documentation section"}),"."]}),"\n",(0,s.jsx)(t.h2,{id:"initialization",children:"Initialization"}),"\n",(0,s.jsxs)(t.p,{children:["The mod.io monetization features are enabled during the onboarding process on your ",(0,s.jsx)(t.a,{href:"https://mod.io/g",children:"game profile"}),"."]}),"\n",(0,s.jsxs)(t.p,{children:["Ensure that you have set the appropriate ",(0,s.jsx)(t.code,{children:"Portal"})," when ",(0,s.jsx)(t.a,{href:"initialization",children:"initializing the plugin"}),". For instance, on Steam you must initialize with ",(0,s.jsx)(t.code,{children:"EModioPortal::Steam"})," to redeem entitlements for Steam."]}),"\n",(0,s.jsx)(t.h2,{id:"getting-the-users-wallet",children:"Getting the user's wallet"}),"\n",(0,s.jsxs)(t.p,{children:["On startup, you can make a call to ",(0,s.jsx)(t.code,{children:"UModioSubsystem::GetUserWalletBalanceAsync"})," to get the balance of the current user's wallet. If no wallet exists for the user, one will be automatically created for them. This call returns the user's wallet balance for the current game."]}),"\n",(0,s.jsxs)(t.p,{children:["The only time you need to make this call is on start-up. We recommend that you cache the value of this result in your game code rather than making consistent calls to ",(0,s.jsx)(t.code,{children:"UModioSubsystem::GetUserWalletBalanceAsync"}),", and update your local state from the return values of other calls that affect wallet balance."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(t.p,{children:(0,s.jsx)(t.img,{alt:"get_user_wallet",src:n(93207).A+"",width:"598",height:"414"})})}),(0,s.jsx)(o.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(t.pre,{children:(0,s.jsx)(t.code,{className:"language-cpp",children:"void UModioManagerSubsystem::GetUserWallet()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->GetUserWalletBalanceAsync(FOnGetUserWalletBalanceDelegate::CreateUObject(this, &UModioManagerSubsystem::OnGetUserWalletComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnGetUserWalletComplete(FModioErrorCode ErrorCode, FModioOptionalUInt64 WalletBalance)\n{\n if (!ErrorCode)\n {\n \t// Wallet balance successfully retrieved\n }\n}\n"})})})]}),"\n",(0,s.jsx)(t.h2,{id:"querying-mods",children:"Querying mods"}),"\n",(0,s.jsxs)(t.p,{children:["As part ",(0,s.jsx)(t.code,{children:"UModioSubsystem::ListAllModsAsync"}),", you can include an additional filter for whether you list paid mods. By default, only free mods are shown. Set ",(0,s.jsx)(t.code,{children:"RevenueType"})," on the ",(0,s.jsx)(t.code,{children:"ModioFilterParams"})," object passed to ",(0,s.jsx)(t.code,{children:"UModioSubsystem::ListAllModsAsync"})," to include free and paid content, or just paid content. All mods returned will have a ",(0,s.jsx)(t.code,{children:"Price"})," property, indicating the virtual currency price that must be paid to purchase that mod."]}),"\n",(0,s.jsx)(t.p,{children:"Filtering for Paid/Unpaid content is not currently exposed to Blueprint."}),"\n",(0,s.jsx)(t.h2,{id:"purchasing-mods",children:"Purchasing mods"}),"\n",(0,s.jsxs)(t.p,{children:["Call ",(0,s.jsx)(t.code,{children:"UModioSubsystem::PurchaseModAsync"})," to purchase a specified mod. ",(0,s.jsx)(t.code,{children:"PurchaseModAsync"})," takes two parameters: the ",(0,s.jsx)(t.code,{children:"ModioModID"})," of the mod to purchase, and the ",(0,s.jsx)(t.code,{children:"ExpectedPrice"}),", which is the price displayed to the user from ",(0,s.jsx)(t.code,{children:"UModioSubsystem::ListAllModsAsync"}),". You must include this parameter for safety so the user is not charged more or less than the price displayed to them, in case the price of the mod has changed between the call to ",(0,s.jsx)(t.code,{children:"ListAllModsAsync"})," and purchase time."]}),"\n",(0,s.jsx)(t.p,{children:"Once a mod is purchased, it is automatically subscribed to for the user."}),"\n",(0,s.jsxs)(t.p,{children:["You should validate that the user has enough virtual currency to make the purchase by comparing it to the balance you received from ",(0,s.jsx)(t.code,{children:"UModioSubsystem::GetUserWalletBalanceAsync"}),". This is purely for user experience (e.g. for graying out the purchase button in the UI, or upselling the user a virtual currency pack); ",(0,s.jsx)(t.code,{children:"UModioSubsystem::PurchaseModAsync"})," will return an error if the user has insufficient funds in their wallet."]}),"\n",(0,s.jsxs)(t.p,{children:["An updated wallet balance (with the purchase amount subtracted) is returned in the callback of ",(0,s.jsx)(t.code,{children:"UModioSubsystem::PurchaseModAsync"}),"."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(t.p,{children:(0,s.jsx)(t.img,{alt:"purchase_mod",src:n(793).A+"",width:"617",height:"442"})})}),(0,s.jsx)(o.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(t.pre,{children:(0,s.jsx)(t.code,{className:"language-cpp",children:"void UModioManagerSubsystem::PurchaseMod(FModioModID ModId, FModioUnsigned64 ExpectedPrice)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->PurchaseModAsync(ModId, ExpectedPrice, FOnPurchaseModDelegate::CreateUObject(this, &UModioManagerSubsystem::OnPurchaseModComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnPurchaseModComplete(FModioErrorCode ErrorCode, FModioOptionalTransactionRecord Transaction)\n{\n if (!ErrorCode)\n {\n \t// Mod purchase successful\n }\n}\n"})})})]}),"\n",(0,s.jsx)(t.h2,{id:"showing-user-purchases",children:"Showing user purchases"}),"\n",(0,s.jsxs)(t.p,{children:["While purchased mods are automatically subscribed and installed at purchase time, the user can freely unsubscribe and uninstall purchased mods and they and they will remain owned and purchased by the user. They must re-subscribe to the mod to have it re-installed. Use ",(0,s.jsx)(t.code,{children:"UModioSubsystem::FetchUserPurchasesAsync"})," to fetch an updated list of a user's purchased mods from the server. After a successful call, you can then display the user's purchased mods with ",(0,s.jsx)(t.code,{children:"UModioSubsystem::QueryUserPurchasedMods"}),", allowing re-subscription as desired."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(t.p,{children:(0,s.jsx)(t.img,{alt:"show_user_purchases",src:n(24959).A+"",width:"580",height:"481"})})}),(0,s.jsx)(o.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(t.pre,{children:(0,s.jsx)(t.code,{className:"language-cpp",children:"void UModioManagerSubsystem::FetchUserPurchases()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->FetchUserPurchasesAsync(FOnFetchUserPurchasesDelegate::CreateUObject(this, &UModioManagerSubsystem::OnFetchUserPurchasesComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnFetchUserPurchasesComplete(FModioErrorCode ErrorCode)\n{\n if (!ErrorCode)\n {\n \t// Purchases Successfully Fetched\n \tif (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n \t{\n \t\t// We can now access the list of purchased mods directly\n \t\tTMap PurchasedMods = Subsystem->QueryUserPurchasedMods();\n \t}\n }\n}\n"})})})]}),"\n",(0,s.jsx)(t.h2,{id:"getting-a-user-delegation-token",children:"Getting a user delegation token"}),"\n",(0,s.jsxs)(t.p,{children:["User delegation tokens can be used by a backend server for S2S (Server to Server) transactions. You can get one for the current user by calling ",(0,s.jsx)(t.code,{children:"UModioSubsystem::GetUserDelegationTokenAsync"}),", the callback for which contains the token as an ",(0,s.jsx)(t.code,{children:"FString"}),"."]}),"\n",(0,s.jsxs)(a.A,{"group-id":"languages",children:[(0,s.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(t.p,{children:(0,s.jsx)(t.img,{alt:"get_user_delegation_token",src:n(40956).A+"",width:"670",height:"448"})})}),(0,s.jsx)(o.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(t.pre,{children:(0,s.jsx)(t.code,{className:"language-cpp",children:"void UModioManagerSubsystem::GetUserDelegationToken()\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tSubsystem->GetUserDelegationTokenAsync(FOnGetUserDelegationTokenDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnGetUserDelegationTokenCallback));\n }\n}\n\nvoid UModioManagerSubsystem::OnGetUserDelegationTokenCallback(FModioErrorCode ErrorCode, FString UserDelegationToken)\n{\n if (!ErrorCode)\n {\n \t// Successfully got User Delegation Token\n }\n}\n"})})})]}),"\n",(0,s.jsx)(t.h2,{id:"listing-and-purchasing-premium-currency",children:"Listing and Purchasing Premium Currency"}),"\n",(0,s.jsx)(t.p,{children:"The Monetization premium features mod.io includes the ability to show the relevant platform store (where possible), which allows the user to purchase virtual currency, top up their wallet, and purchase Premium UGC all without leaving the game."}),"\n",(0,s.jsx)(t.p,{children:"This functionality is exposed in one of two ways."}),"\n",(0,s.jsx)(t.h3,{id:"displaying-the-native-platform-store",children:"Displaying the Native Platform Store"}),"\n",(0,s.jsxs)(t.p,{children:["The recommended method is to simply call ",(0,s.jsx)(t.code,{children:"UModioUISubsystem::RequestShowTokenPurchaseUIWithHandler"})," which will attempt to show the appropriate Platform Native storefront via Unreal's ",(0,s.jsx)(t.code,{children:"IOnlineSubsystem"}),". This function will return an ",(0,s.jsx)(t.code,{children:"EModioOpenStoreResult"})," indicating whether call succeeded or not."]}),"\n",(0,s.jsxs)(t.p,{children:[(0,s.jsx)(t.code,{children:"FailedInactive"})," indicates that the Monetization feature is disabled in the base Mod.io plugin settings."]}),"\n",(0,s.jsxs)(t.p,{children:[(0,s.jsx)(t.code,{children:"FailedUnsupportedPlatform"})," means that the current platform has not implemented its native storefront in the Unreal Engine Online Subsystem, and so you will have to implement your own method for getting and displaying purchasable items. An example/template of this can be found in the ComponentUI reference design, see below for more information."]}),"\n",(0,s.jsxs)(t.p,{children:[(0,s.jsx)(t.code,{children:"FailedUnknown"})," is a catch-all for other error types, such as inactive online subsystems, network issues, etc, and you should check the relevant logs for any pertinent information."]}),"\n",(0,s.jsxs)(t.p,{children:[(0,s.jsx)(t.code,{children:"Success"})," means that the platform store UI is being displayed, and you can then listen for the ",(0,s.jsx)(t.code,{children:"Callback"})," to handle any resulting purchases. This ",(0,s.jsx)(t.code,{children:"Callback"})," only indicates whether the Store was opened and closed successfully, not whether the user made a purchase or not. There is also ",(0,s.jsx)(t.code,{children:"UModioUISubsystem::RequestShowTokenPurchaseUI"})," for instances where a Callback is not needed."]}),"\n",(0,s.jsx)(a.A,{"group-id":"languages",children:(0,s.jsx)(o.A,{value:"blueprint",label:"Blueprint",children:(0,s.jsx)(t.p,{children:(0,s.jsx)(t.img,{alt:"ShowPlatformStoreUI",src:n(55494).A+"",width:"1029",height:"445"})})})}),"\n",(0,s.jsx)(t.h3,{id:"displaying-component-ui-store-offers",children:"Displaying Component UI Store Offers"}),"\n",(0,s.jsx)(t.p,{children:"In cases where native store display is not supported, there is a provided template UI component for displaying and listing product offers within the Component UI itself."}),"\n",(0,s.jsxs)(t.p,{children:["Found at ",(0,s.jsx)(t.code,{children:"ModioComponentUI/Content/UI/Templates/Default/WBP_ModioDefaultTokenPackBrowser"})," this Widget provides a framework and example of querying a set of product offers, displaying them, and purchasing them through the platform store."]}),"\n",(0,s.jsx)(t.p,{children:"NOTE: With either the Native or Component UI methods, the refreshing and consumption of purchased Entitlements, as well as the updating of the user's Wallet Balance, is left up to you the developer due to platform limitations."}),"\n",(0,s.jsx)(t.h2,{id:"consuming-user-entitlements",children:"Consuming User Entitlements"}),"\n",(0,s.jsx)(t.p,{children:"User Entitlements are virtual currency packs/products that the user can purchase through their relevant platform store. These Entitlements can then be consumed; in the case of virtual currency, adding balance to their mod.io wallet for use purchasing UGC."}),"\n",(0,s.jsxs)(t.p,{children:["Once a user has purchased a virtual currency pack on a given store those entitlements must be consumed before they can be used. To do so you can call ",(0,s.jsx)(t.code,{children:"UModioSubsystem::RefreshUserEntitlementsAsync"})," which takes in ",(0,s.jsx)(t.code,{children:"FModioEntitlementParams"})," and a ",(0,s.jsx)(t.code,{children:"FOnRefreshUserEntitlementsDelegateFast"})," as a Callback, returning an ",(0,s.jsx)(t.code,{children:"FModioErrorCode"})," and ",(0,s.jsx)(t.code,{children:"TOptional"})," indicating the status of the user\u2019s entitlements."]}),"\n",(0,s.jsxs)(t.p,{children:[(0,s.jsx)(t.code,{children:"FModioEntitlementParams"})," is to be populated with the Delegation Token received from ",(0,s.jsx)(t.code,{children:"UModioSubsystem::GetUserDelegationTokenAsync"}),"."]}),"\n",(0,s.jsx)(a.A,{"group-id":"languages",children:(0,s.jsx)(o.A,{value:"c++",label:"C++",default:!0,children:(0,s.jsx)(t.pre,{children:(0,s.jsx)(t.code,{className:"language-cpp",children:'// Get the relevant platform authentication token prior to making this call.\nvoid UModioManager::RefreshUserEntitlements(const FString& PlatformAuthToken)\n{\n\tif (GEngine->GetEngineSubsystem())\n\t{\n\t\tFModioEntitlementParams Params;\n\t\tParams.ExtendedParams.Add("auth_code", PlatformAuthToken);\n\t\tGEngine->GetEngineSubsystem()->RefreshUserEntitlementsAsync(Params, FOnRefreshUserEntitlementsDelegateFast::CreateUObject(this, &UModioManager::OnRefreshUserEntitlementsCallback));\n\t}\n}\n\nvoid UModioManager::OnRefreshUserEntitlementsCallback(FModioErrorCode ErrorCode, TOptional ConsumptionStatusList)\n{\n\tif (ErrorCode == false)\n\t{\n\t\t// Successfully refreshed entitlements.\n\t}\n}\n'})})})}),"\n",(0,s.jsxs)(t.p,{children:["After entitlements are refreshed successfully any classes properly implementing the ",(0,s.jsx)(t.code,{children:"ModioUIWalletBalanceUpdatedEventReceiver"})," interface will receive the updated balance."]})]})}function m(e={}){const{wrapper:t}={...(0,r.R)(),...e.components};return t?(0,s.jsx)(t,{...e,children:(0,s.jsx)(h,{...e})}):h(e)}},19365:(e,t,n)=>{n.d(t,{A:()=>o});n(96540);var s=n(18215);const r={tabItem:"tabItem_Ymn6"};var a=n(74848);function o(e){let{children:t,hidden:n,className:o}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,s.A)(r.tabItem,o),hidden:n,children:t})}},11470:(e,t,n)=>{n.d(t,{A:()=>M});var s=n(96540),r=n(18215),a=n(23104),o=n(56347),i=n(205),l=n(57485),d=n(31682),c=n(70679);function u(e){return s.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,s.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function h(e){const{values:t,children:n}=e;return(0,s.useMemo)((()=>{const e=t??function(e){return u(e).map((e=>{let{props:{value:t,label:n,attributes:s,default:r}}=e;return{value:t,label:n,attributes:s,default:r}}))}(n);return function(e){const t=(0,d.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function m(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function p(e){let{queryString:t=!1,groupId:n}=e;const r=(0,o.W6)(),a=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,l.aZ)(a),(0,s.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(r.location.search);t.set(a,e),r.replace({...r.location,search:t.toString()})}),[a,r])]}function g(e){const{defaultValue:t,queryString:n=!1,groupId:r}=e,a=h(e),[o,l]=(0,s.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!m({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const s=n.find((e=>e.default))??n[0];if(!s)throw new Error("Unexpected error: 0 tabValues");return s.value}({defaultValue:t,tabValues:a}))),[d,u]=p({queryString:n,groupId:r}),[g,f]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[r,a]=(0,c.Dv)(n);return[r,(0,s.useCallback)((e=>{n&&a.set(e)}),[n,a])]}({groupId:r}),b=(()=>{const e=d??g;return m({value:e,tabValues:a})?e:null})();(0,i.A)((()=>{b&&l(b)}),[b]);return{selectedValue:o,selectValue:(0,s.useCallback)((e=>{if(!m({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);l(e),u(e),f(e)}),[u,f,a]),tabValues:a}}var f=n(92303);const b={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var y=n(74848);function x(e){let{className:t,block:n,selectedValue:s,selectValue:o,tabValues:i}=e;const l=[],{blockElementScrollPositionUntilNextRender:d}=(0,a.a_)(),c=e=>{const t=e.currentTarget,n=l.indexOf(t),r=i[n].value;r!==s&&(d(t),o(r))},u=e=>{let t=null;switch(e.key){case"Enter":c(e);break;case"ArrowRight":{const n=l.indexOf(e.currentTarget)+1;t=l[n]??l[0];break}case"ArrowLeft":{const n=l.indexOf(e.currentTarget)-1;t=l[n]??l[l.length-1];break}}t?.focus()};return(0,y.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.A)("tabs",{"tabs--block":n},t),children:i.map((e=>{let{value:t,label:n,attributes:a}=e;return(0,y.jsx)("li",{role:"tab",tabIndex:s===t?0:-1,"aria-selected":s===t,ref:e=>l.push(e),onKeyDown:u,onClick:c,...a,className:(0,r.A)("tabs__item",b.tabItem,a?.className,{"tabs__item--active":s===t}),children:n??t},t)}))})}function v(e){let{lazy:t,children:n,selectedValue:r}=e;const a=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===r));return e?(0,s.cloneElement)(e,{className:"margin-top--md"}):null}return(0,y.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,s.cloneElement)(e,{key:t,hidden:e.props.value!==r})))})}function j(e){const t=g(e);return(0,y.jsxs)("div",{className:(0,r.A)("tabs-container",b.tabList),children:[(0,y.jsx)(x,{...t,...e}),(0,y.jsx)(v,{...t,...e})]})}function M(e){const t=(0,f.A)();return(0,y.jsx)(j,{...e,children:u(e.children)},String(t))}},89236:(e,t,n)=>{n.d(t,{A:()=>a});var s=n(19365),r=(n(96540),n(74848));function a(e){return(0,r.jsx)(r.Fragment,{children:(0,r.jsx)(s.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,n)=>{n.d(t,{A:()=>a});var s=n(11470),r=(n(96540),n(74848));function a(e){return(0,r.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,r.jsx)(s.A,{...e})})}},55494:(e,t,n)=>{n.d(t,{A:()=>s});const s=n.p+"assets/images/ShowPlatformStoreUI-513a06081ba9b2c613e5e4da85b0ca4b.png"},40956:(e,t,n)=>{n.d(t,{A:()=>s});const s=n.p+"assets/images/get_user_delegation_token-9ae69e059f28541bbac934ea02b8f994.png"},93207:(e,t,n)=>{n.d(t,{A:()=>s});const s=n.p+"assets/images/get_user_wallet-3e8cab6076988d8cc7469c3183f8711b.png"},793:(e,t,n)=>{n.d(t,{A:()=>s});const s=n.p+"assets/images/purchase_mod-d1e2b2ee8586f08eada43e87167ee125.png"},24959:(e,t,n)=>{n.d(t,{A:()=>s});const s=n.p+"assets/images/show_user_purchases-575ecaef025b90bbf3ea30f2c0437cfe.png"},28453:(e,t,n)=>{n.d(t,{R:()=>o,x:()=>i});var s=n(96540);const r={},a=s.createContext(r);function o(e){const t=s.useContext(a);return s.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function i(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:o(e.components),s.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/d99e1931.11845b15.js b/Doc/assets/js/d99e1931.11845b15.js new file mode 100644 index 00000000..35429312 --- /dev/null +++ b/Doc/assets/js/d99e1931.11845b15.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[1462],{20040:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>d,contentTitle:()=>l,default:()=>c,frontMatter:()=>o,metadata:()=>r,toc:()=>a});var i=t(74848),s=t(28453);const o={id:"ue-installation-and-setup",title:"Installation & Setup",slug:"/unreal/installation-and-setup/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/installation-and-setup.mdx"},l=void 0,r={id:"ue-installation-and-setup",title:"Installation & Setup",description:"Adding plugin files to your project",source:"@site/public/en-us/installation-and-setup.mdx",sourceDirName:".",slug:"/unreal/installation-and-setup/",permalink:"/unreal/installation-and-setup/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/installation-and-setup.mdx",tags:[],version:"current",frontMatter:{id:"ue-installation-and-setup",title:"Installation & Setup",slug:"/unreal/installation-and-setup/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/installation-and-setup.mdx"},sidebar:"sidebar",previous:{title:"Overview",permalink:"/unreal/"},next:{title:"Getting Started",permalink:"/unreal/getting-started/"}},d={},a=[{value:"Adding plugin files to your project",id:"adding-plugin-files-to-your-project",level:2},{value:"As a git submodule",id:"as-a-git-submodule",level:3},{value:"In a non-git project, or without submodules",id:"in-a-non-git-project-or-without-submodules",level:3},{value:"Enabling the plugin",id:"enabling-the-plugin",level:2},{value:"Plugin configuration",id:"plugin-configuration",level:2},{value:"Using a background thread",id:"using-a-background-thread",level:2},{value:"Using the Live environment",id:"using-the-live-environment",level:2},{value:"Next Steps",id:"next-steps",level:2}];function A(e){const n={a:"a",admonition:"admonition",code:"code",h2:"h2",h3:"h3",img:"img",li:"li",p:"p",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,s.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"adding-plugin-files-to-your-project",children:"Adding plugin files to your project"}),"\n",(0,i.jsxs)(n.p,{children:["Download the Unreal Engine plugin from the ",(0,i.jsx)(n.a,{href:"https://github.com/modio/modio-ue",children:"mod.io GitHub repository"})," using one of the following methods:"]}),"\n",(0,i.jsx)(n.h3,{id:"as-a-git-submodule",children:"As a git submodule"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["\n",(0,i.jsxs)(n.p,{children:["In the directory with your ",(0,i.jsx)(n.code,{children:".uproject"})," file, add and download the mod.io plugin as a git submodule with the command:"]}),"\n",(0,i.jsx)(n.p,{children:(0,i.jsx)(n.code,{children:"git submodule add https://github.com/modio/modio-ue.git Plugins/Modio"})}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:["\n",(0,i.jsx)(n.p,{children:"Next, initialize our submodules and dependencies:"}),"\n",(0,i.jsx)(n.p,{children:(0,i.jsx)(n.code,{children:"git submodule update --init --recursive"})}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.h3,{id:"in-a-non-git-project-or-without-submodules",children:"In a non-git project, or without submodules"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["Grab the latest release zip from the ",(0,i.jsx)(n.a,{href:"https://github.com/modio/modio-ue/releases",children:"releases section"})," on GitHub and extract the contents to your project\u2019s ",(0,i.jsx)(n.code,{children:"Plugins/Modio"})," directory"]}),"\n"]}),"\n",(0,i.jsxs)(n.p,{children:[(0,i.jsx)(n.img,{alt:"get_latest_release",src:t(23846).A+"",width:"385",height:"235"})," ",(0,i.jsx)(n.img,{alt:"get_latest_release2",src:t(48518).A+"",width:"366",height:"181"})]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsx)(n.p,{children:"GitHub's automatically generated zips will not work! They do not contain our submodule dependencies. Ensure you download the zip file from the releases page as highlighted above."})}),"\n",(0,i.jsx)(n.h2,{id:"enabling-the-plugin",children:"Enabling the plugin"}),"\n",(0,i.jsx)(n.p,{children:"Start the editor and enable the plugin in the Unreal Engine plugin settings."}),"\n",(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"plugin_window",src:t(78892).A+"",width:"1593",height:"633"})}),"\n",(0,i.jsx)(n.h2,{id:"plugin-configuration",children:"Plugin configuration"}),"\n",(0,i.jsx)(n.p,{children:"The plugin requires some initial configuration to behave correctly. You can access these settings in the Project Settings window:"}),"\n",(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"plugin_settings",src:t(80619).A+"",width:"1577",height:"964"})}),"\n",(0,i.jsx)(n.p,{children:"The settings have the following parameters:"}),"\n",(0,i.jsxs)(n.table,{children:[(0,i.jsx)(n.thead,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.th,{children:"Property"}),(0,i.jsx)(n.th,{children:"Description"})]})}),(0,i.jsxs)(n.tbody,{children:[(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Game Id"}),(0,i.jsx)(n.td,{children:"Your mod.io-provided Game Id for the target environment"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"API Key"}),(0,i.jsx)(n.td,{children:"Your mod.io-provided API key for the target environment"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Environment"}),(0,i.jsxs)(n.td,{children:["Your target environment, ",(0,i.jsx)(n.code,{children:"Live"})," or ",(0,i.jsx)(n.code,{children:"Test"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Log Level"}),(0,i.jsx)(n.td,{children:"The default logging level to use. Messages with a lower log level will be silently discarded."})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Portal"}),(0,i.jsx)(n.td,{children:"The default portal to use. This usually corresponds to the store your game will be distributed through."})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Use Background Thread"}),(0,i.jsxs)(n.td,{children:["Enables or disables the use of a ",(0,i.jsx)(n.a,{href:"#using-a-background-thread",children:"background thread"})," for the plugin's work."]})]})]})]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["The Test environment is available for usage in limited circumstances only. ",(0,i.jsx)(n.strong,{children:(0,i.jsx)(n.a,{href:"/unreal/installation-and-setup/#using-the-live-environment",children:"All games should be set up on the Live environment"})}),". Set your game to ",(0,i.jsx)(n.strong,{children:"hidden"})," to restrict access during the testing phase. If you require access to Test, please contact us."]})}),"\n",(0,i.jsx)(n.h2,{id:"using-a-background-thread",children:"Using a background thread"}),"\n",(0,i.jsxs)(n.p,{children:[(0,i.jsx)(n.a,{href:"/unreal/refdocs/#run-pending-handlers",children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})})," is responsible for running any pending plugin work, including invoking callbacks passed to asynchronous operations. With ",(0,i.jsx)(n.code,{children:"Use Background Thread"})," set to ",(0,i.jsx)(n.code,{children:"true"}),", the mod.io plugin will automatically spin up a background thread and continually call ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"})," for you. This decouples the frequency of the plugin\u2019s work from your game's main thread, and improves performance of the plugin overall."]}),"\n",(0,i.jsxs)(n.p,{children:["Calling methods from the ",(0,i.jsx)(n.a,{href:"/unreal/refdocs/#modiosubsystem",children:(0,i.jsx)(n.code,{children:"ModioSubsystem"})})," will marshall all callbacks back to the game thread. All other plugin call guarantees remain the same."]}),"\n",(0,i.jsxs)(n.h2,{id:"using-the-live-environment",children:["Using the ",(0,i.jsx)(n.code,{children:"Live"})," environment"]}),"\n",(0,i.jsxs)(n.p,{children:["All games should be set up on the ",(0,i.jsx)(n.code,{children:"Live"})," environment for full access to the mod.io REST API's capabilities."]}),"\n",(0,i.jsxs)(n.p,{children:["If your game or it's modding capabilities are not yet publicly available, set your game to ",(0,i.jsx)(n.strong,{children:"hidden"})," on the ",(0,i.jsx)(n.code,{children:"Live"})," environment. This allows you to restrict access to specific accounts and perform QA against the production environment without exposing your title to the public."]}),"\n",(0,i.jsxs)(n.p,{children:["Once you are ready, change the state from ",(0,i.jsx)(n.strong,{children:"hidden"})," to ",(0,i.jsx)(n.strong,{children:"public"})," for a full release of your game\u2019s modding capabilities."]}),"\n",(0,i.jsx)(n.h2,{id:"next-steps",children:"Next Steps"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["Our ",(0,i.jsx)(n.a,{href:"/unreal/getting-started/",children:"Getting Started Guide"})," page contains a series of quick-start guides with code samples demonstrating the plugin's core functionality."]}),"\n",(0,i.jsxs)(n.li,{children:["A sample project demonstrating basic mod.io functionality is available ",(0,i.jsx)(n.a,{href:"https://go.mod.io/ue5-sample",children:"here"}),"."]}),"\n"]})]})}function c(e={}){const{wrapper:n}={...(0,s.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(A,{...e})}):A(e)}},23846:(e,n,t)=>{t.d(n,{A:()=>i});const i=""},48518:(e,n,t)=>{t.d(n,{A:()=>i});const i=""},80619:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/plugin_settings-031f6d03534360a3799d7adbf8180173.png"},78892:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/plugin_window-74a3e99e913b1314a247702723f4af28.png"},28453:(e,n,t)=>{t.d(n,{R:()=>l,x:()=>r});var i=t(96540);const s={},o=i.createContext(s);function l(e){const n=i.useContext(o);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:l(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/d99e1931.ad6d4183.js b/Doc/assets/js/d99e1931.ad6d4183.js deleted file mode 100644 index f36cecc2..00000000 --- a/Doc/assets/js/d99e1931.ad6d4183.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[1462],{20040:(e,n,t)=>{t.r(n),t.d(n,{assets:()=>d,contentTitle:()=>l,default:()=>c,frontMatter:()=>o,metadata:()=>r,toc:()=>a});var i=t(74848),s=t(28453);const o={id:"ue-installation-and-setup",title:"Installation & Setup",slug:"/unreal/installation-and-setup/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/installation-and-setup.mdx"},l=void 0,r={id:"ue-installation-and-setup",title:"Installation & Setup",description:"Adding plugin files to your project",source:"@site/public/en-us/installation-and-setup.mdx",sourceDirName:".",slug:"/unreal/installation-and-setup/",permalink:"/unreal/installation-and-setup/",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/installation-and-setup.mdx",tags:[],version:"current",frontMatter:{id:"ue-installation-and-setup",title:"Installation & Setup",slug:"/unreal/installation-and-setup/",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/installation-and-setup.mdx"},sidebar:"sidebar",previous:{title:"Overview",permalink:"/unreal/"},next:{title:"Getting Started",permalink:"/unreal/getting-started/"}},d={},a=[{value:"Adding plugin files to your project",id:"adding-plugin-files-to-your-project",level:2},{value:"As a git submodule",id:"as-a-git-submodule",level:3},{value:"In a non-git project, or without submodules",id:"in-a-non-git-project-or-without-submodules",level:3},{value:"Enabling the plugin",id:"enabling-the-plugin",level:2},{value:"Plugin configuration",id:"plugin-configuration",level:2},{value:"Using a background thread",id:"using-a-background-thread",level:2},{value:"Using the Live environment",id:"using-the-live-environment",level:2},{value:"Next Steps",id:"next-steps",level:2}];function A(e){const n={a:"a",admonition:"admonition",code:"code",h2:"h2",h3:"h3",img:"img",li:"li",p:"p",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,s.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(n.h2,{id:"adding-plugin-files-to-your-project",children:"Adding plugin files to your project"}),"\n",(0,i.jsxs)(n.p,{children:["Download the Unreal Engine plugin from the ",(0,i.jsx)(n.a,{href:"https://github.com/modio/modio-ue",children:"mod.io GitHub repository"})," using one of the following methods:"]}),"\n",(0,i.jsx)(n.h3,{id:"as-a-git-submodule",children:"As a git submodule"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["\n",(0,i.jsxs)(n.p,{children:["In the directory with your ",(0,i.jsx)(n.code,{children:".uproject"})," file, add and download the mod.io plugin as a git submodule with the command:"]}),"\n",(0,i.jsx)(n.p,{children:(0,i.jsx)(n.code,{children:"git submodule add https://github.com/modio/modio-ue.git Plugins/Modio"})}),"\n"]}),"\n",(0,i.jsxs)(n.li,{children:["\n",(0,i.jsx)(n.p,{children:"Next, initialize our submodules and dependencies:"}),"\n",(0,i.jsx)(n.p,{children:(0,i.jsx)(n.code,{children:"git submodule update --init --recursive"})}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(n.h3,{id:"in-a-non-git-project-or-without-submodules",children:"In a non-git project, or without submodules"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["Grab the latest release zip from the ",(0,i.jsx)(n.a,{href:"https://github.com/modio/modio-ue/releases",children:"releases section"})," on GitHub and extract the contents to your project\u2019s ",(0,i.jsx)(n.code,{children:"Plugins/Modio"})," directory"]}),"\n"]}),"\n",(0,i.jsxs)(n.p,{children:[(0,i.jsx)(n.img,{alt:"get_latest_release",src:t(73649).A+"",width:"385",height:"235"})," ",(0,i.jsx)(n.img,{alt:"get_latest_release2",src:t(77655).A+"",width:"366",height:"181"})]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsx)(n.p,{children:"GitHub's automatically generated zips will not work! They do not contain our submodule dependencies. Ensure you download the zip file from the releases page as highlighted above."})}),"\n",(0,i.jsx)(n.h2,{id:"enabling-the-plugin",children:"Enabling the plugin"}),"\n",(0,i.jsx)(n.p,{children:"Start the editor and enable the plugin in the Unreal Engine plugin settings."}),"\n",(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"plugin_window",src:t(80101).A+"",width:"1593",height:"633"})}),"\n",(0,i.jsx)(n.h2,{id:"plugin-configuration",children:"Plugin configuration"}),"\n",(0,i.jsx)(n.p,{children:"The plugin requires some initial configuration to behave correctly. You can access these settings in the Project Settings window:"}),"\n",(0,i.jsx)(n.p,{children:(0,i.jsx)(n.img,{alt:"plugin_settings",src:t(17834).A+"",width:"1577",height:"964"})}),"\n",(0,i.jsx)(n.p,{children:"The settings have the following parameters:"}),"\n",(0,i.jsxs)(n.table,{children:[(0,i.jsx)(n.thead,{children:(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.th,{children:"Property"}),(0,i.jsx)(n.th,{children:"Description"})]})}),(0,i.jsxs)(n.tbody,{children:[(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Game Id"}),(0,i.jsx)(n.td,{children:"Your mod.io-provided Game Id for the target environment"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"API Key"}),(0,i.jsx)(n.td,{children:"Your mod.io-provided API key for the target environment"})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Environment"}),(0,i.jsxs)(n.td,{children:["Your target environment, ",(0,i.jsx)(n.code,{children:"Live"})," or ",(0,i.jsx)(n.code,{children:"Test"})]})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Log Level"}),(0,i.jsx)(n.td,{children:"The default logging level to use. Messages with a lower log level will be silently discarded."})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Portal"}),(0,i.jsx)(n.td,{children:"The default portal to use. This usually corresponds to the store your game will be distributed through."})]}),(0,i.jsxs)(n.tr,{children:[(0,i.jsx)(n.td,{children:"Use Background Thread"}),(0,i.jsxs)(n.td,{children:["Enables or disables the use of a ",(0,i.jsx)(n.a,{href:"#using-a-background-thread",children:"background thread"})," for the plugin's work."]})]})]})]}),"\n",(0,i.jsx)(n.admonition,{type:"note",children:(0,i.jsxs)(n.p,{children:["The Test environment is available for usage in limited circumstances only. ",(0,i.jsx)(n.strong,{children:(0,i.jsx)(n.a,{href:"/unreal/installation-and-setup/#using-the-live-environment",children:"All games should be set up on the Live environment"})}),". Set your game to ",(0,i.jsx)(n.strong,{children:"hidden"})," to restrict access during the testing phase. If you require access to Test, please contact us."]})}),"\n",(0,i.jsx)(n.h2,{id:"using-a-background-thread",children:"Using a background thread"}),"\n",(0,i.jsxs)(n.p,{children:[(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#RunPendingHandlers",children:(0,i.jsx)(n.code,{children:"RunPendingHandlers"})})," is responsible for running any pending plugin work, including invoking callbacks passed to asynchronous operations. With ",(0,i.jsx)(n.code,{children:"Use Background Thread"})," set to ",(0,i.jsx)(n.code,{children:"true"}),", the mod.io plugin will automatically spin up a background thread and continually call ",(0,i.jsx)(n.code,{children:"RunPendingHandlers"})," for you. This decouples the frequency of the plugin\u2019s work from your game's main thread, and improves performance of the plugin overall."]}),"\n",(0,i.jsxs)(n.p,{children:["Calling methods from the ",(0,i.jsx)(n.a,{href:"https://docs.mod.io/unrealref/#_modiosubsystem",children:(0,i.jsx)(n.code,{children:"ModioSubsystem"})})," will marshall all callbacks back to the game thread. All other plugin call guarantees remain the same."]}),"\n",(0,i.jsxs)(n.h2,{id:"using-the-live-environment",children:["Using the ",(0,i.jsx)(n.code,{children:"Live"})," environment"]}),"\n",(0,i.jsxs)(n.p,{children:["All games should be set up on the ",(0,i.jsx)(n.code,{children:"Live"})," environment for full access to the mod.io REST API's capabilities."]}),"\n",(0,i.jsxs)(n.p,{children:["If your game or it's modding capabilities are not yet publicly available, set your game to ",(0,i.jsx)(n.strong,{children:"hidden"})," on the ",(0,i.jsx)(n.code,{children:"Live"})," environment. This allows you to restrict access to specific accounts and perform QA against the production environment without exposing your title to the public."]}),"\n",(0,i.jsxs)(n.p,{children:["Once you are ready, change the state from ",(0,i.jsx)(n.strong,{children:"hidden"})," to ",(0,i.jsx)(n.strong,{children:"public"})," for a full release of your game\u2019s modding capabilities."]}),"\n",(0,i.jsx)(n.h2,{id:"next-steps",children:"Next Steps"}),"\n",(0,i.jsxs)(n.ul,{children:["\n",(0,i.jsxs)(n.li,{children:["Our ",(0,i.jsx)(n.a,{href:"/unreal/getting-started/",children:"Getting Started Guide"})," page contains a series of quick-start guides with code samples demonstrating the plugin's core functionality."]}),"\n",(0,i.jsxs)(n.li,{children:["A sample project demonstrating basic mod.io functionality is available ",(0,i.jsx)(n.a,{href:"https://go.mod.io/ue5-sample",children:"here"}),"."]}),"\n"]})]})}function c(e={}){const{wrapper:n}={...(0,s.R)(),...e.components};return n?(0,i.jsx)(n,{...e,children:(0,i.jsx)(A,{...e})}):A(e)}},73649:(e,n,t)=>{t.d(n,{A:()=>i});const i=""},77655:(e,n,t)=>{t.d(n,{A:()=>i});const i=""},17834:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/plugin_settings-031f6d03534360a3799d7adbf8180173.png"},80101:(e,n,t)=>{t.d(n,{A:()=>i});const i=t.p+"assets/images/plugin_window-74a3e99e913b1314a247702723f4af28.png"},28453:(e,n,t)=>{t.d(n,{R:()=>l,x:()=>r});var i=t(96540);const s={},o=i.createContext(s);function l(e){const n=i.useContext(o);return i.useMemo((function(){return"function"==typeof e?e(n):{...n,...e}}),[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(s):e.components||s:l(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/fb32500e.b8d34aff.js b/Doc/assets/js/fb32500e.b8d34aff.js deleted file mode 100644 index 56a17cdc..00000000 --- a/Doc/assets/js/fb32500e.b8d34aff.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[8950],{44149:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>d,default:()=>h,frontMatter:()=>s,metadata:()=>l,toc:()=>c});var o=n(74848),r=n(28453),a=n(27064),i=n(89236);const s={id:"ue-edit-mods",title:"Edit an Existing Mod",slug:"/unreal/getting-started/edit-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/edit-mods.mdx"},d=void 0,l={id:"getting-started/ue-edit-mods",title:"Edit an Existing Mod",description:"Mod details can be edited in-game using SubmitModChangesAsync. This function allows you to edit multiple parameters with a single call. It takes a ModioModID of the mod to edit, a ModioEditModParams containing one or more parameters to be altered, and a callback that will contain an optional updated ModioModInfo object on success.",source:"@site/public/en-us/getting-started/edit-mods.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/edit-mods",permalink:"/unreal/getting-started/edit-mods",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/edit-mods.mdx",tags:[],version:"current",frontMatter:{id:"ue-edit-mods",title:"Edit an Existing Mod",slug:"/unreal/getting-started/edit-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/getting-started/edit-mods.mdx"},sidebar:"sidebar",previous:{title:"In-Game Mod Submission",permalink:"/unreal/getting-started/submit-mods"},next:{title:"Error Handling",permalink:"/unreal/getting-started/error-handling"}},u={},c=[];function m(e){const t={a:"a",admonition:"admonition",code:"code",img:"img",p:"p",pre:"pre",strong:"strong",...(0,r.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsxs)(t.p,{children:["Mod details can be edited in-game using ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_SubmitModChangesAsync",children:(0,o.jsx)(t.code,{children:"SubmitModChangesAsync"})}),". This function allows you to edit multiple parameters with a single call. It takes a ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#_modiomodid",children:(0,o.jsx)(t.code,{children:"ModioModID"})})," of the mod to edit, a ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#_modioeditmodparams",children:(0,o.jsx)(t.code,{children:"ModioEditModParams"})})," containing one or more parameters to be altered, and a callback that will contain an optional updated ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#_modiomodinfo",children:(0,o.jsx)(t.code,{children:"ModioModInfo"})})," object on success."]}),"\n",(0,o.jsx)(t.admonition,{type:"note",children:(0,o.jsxs)(t.p,{children:["To update the mod file itself, use ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#K2_SubmitNewModFileForMod",children:(0,o.jsx)(t.strong,{children:(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"})})}),". See ",(0,o.jsx)(t.strong,{children:(0,o.jsx)(t.a,{href:"submit-mods#submitting-a-file-for-a-mod",children:"Submitting a file for a mod"})})," for more information."]})}),"\n",(0,o.jsxs)(a.A,{"group-id":"languages",children:[(0,o.jsx)(i.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(t.p,{children:(0,o.jsx)(t.img,{alt:"edit_mod",src:n(70627).A+"",width:"1155",height:"480"})})}),(0,o.jsxs)(i.A,{value:"c++",label:"C++",default:!0,children:[(0,o.jsxs)(t.p,{children:["Note that it would be more appropriate to pass an ",(0,o.jsx)(t.a,{href:"https://docs.mod.io/unrealref/#_modioeditmodparams",children:(0,o.jsx)(t.code,{children:"FModioEditModParams"})})," with your desired parameters into ",(0,o.jsx)(t.code,{children:"UModioManagerSubsystem::EditMod()"}),". This example shows their creation within the function to illustrate their use."]}),(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-cpp",children:'void UModioManagerSubsystem::EditMod(FModioModID ModID)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tFModioEditModParams EditParams;\n\n \t// Add one or more parameters to edit\n \tEditParams.Name = TEXT("My Edited Mod Name");\n \tEditParams.Summary = TEXT("My edited summary");\n \t\n \tSubsystem->SubmitModChangesAsync(ModID, EditParams, FOnGetModInfoDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnSubmitModChangesComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnSubmitModChangesComplete(FModioErrorCode ErrorCode, FModioOptionalModInfo UpdatedInfo)\n{\n if (!ErrorCode)\n {\n \t// Mod successfully updated. Can display new details from UpdatedInfo etc.\n }\n}\n\n'})})]})]})]})}function h(e={}){const{wrapper:t}={...(0,r.R)(),...e.components};return t?(0,o.jsx)(t,{...e,children:(0,o.jsx)(m,{...e})}):m(e)}},19365:(e,t,n)=>{n.d(t,{A:()=>i});n(96540);var o=n(18215);const r={tabItem:"tabItem_Ymn6"};var a=n(74848);function i(e){let{children:t,hidden:n,className:i}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,o.A)(r.tabItem,i),hidden:n,children:t})}},11470:(e,t,n)=>{n.d(t,{A:()=>j});var o=n(96540),r=n(18215),a=n(23104),i=n(56347),s=n(205),d=n(57485),l=n(31682),u=n(70679);function c(e){return o.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:n}=e;return(0,o.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:n,attributes:o,default:r}}=e;return{value:t,label:n,attributes:o,default:r}}))}(n);return function(e){const t=(0,l.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function h(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function p(e){let{queryString:t=!1,groupId:n}=e;const r=(0,i.W6)(),a=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,d.aZ)(a),(0,o.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(r.location.search);t.set(a,e),r.replace({...r.location,search:t.toString()})}),[a,r])]}function b(e){const{defaultValue:t,queryString:n=!1,groupId:r}=e,a=m(e),[i,d]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!h({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const o=n.find((e=>e.default))??n[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:t,tabValues:a}))),[l,c]=p({queryString:n,groupId:r}),[b,f]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[r,a]=(0,u.Dv)(n);return[r,(0,o.useCallback)((e=>{n&&a.set(e)}),[n,a])]}({groupId:r}),g=(()=>{const e=l??b;return h({value:e,tabValues:a})?e:null})();(0,s.A)((()=>{g&&d(g)}),[g]);return{selectedValue:i,selectValue:(0,o.useCallback)((e=>{if(!h({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);d(e),c(e),f(e)}),[c,f,a]),tabValues:a}}var f=n(92303);const g={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var M=n(74848);function x(e){let{className:t,block:n,selectedValue:o,selectValue:i,tabValues:s}=e;const d=[],{blockElementScrollPositionUntilNextRender:l}=(0,a.a_)(),u=e=>{const t=e.currentTarget,n=d.indexOf(t),r=s[n].value;r!==o&&(l(t),i(r))},c=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const n=d.indexOf(e.currentTarget)+1;t=d[n]??d[0];break}case"ArrowLeft":{const n=d.indexOf(e.currentTarget)-1;t=d[n]??d[d.length-1];break}}t?.focus()};return(0,M.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.A)("tabs",{"tabs--block":n},t),children:s.map((e=>{let{value:t,label:n,attributes:a}=e;return(0,M.jsx)("li",{role:"tab",tabIndex:o===t?0:-1,"aria-selected":o===t,ref:e=>d.push(e),onKeyDown:c,onClick:u,...a,className:(0,r.A)("tabs__item",g.tabItem,a?.className,{"tabs__item--active":o===t}),children:n??t},t)}))})}function v(e){let{lazy:t,children:n,selectedValue:r}=e;const a=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===r));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return(0,M.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==r})))})}function y(e){const t=b(e);return(0,M.jsxs)("div",{className:(0,r.A)("tabs-container",g.tabList),children:[(0,M.jsx)(x,{...t,...e}),(0,M.jsx)(v,{...t,...e})]})}function j(e){const t=(0,f.A)();return(0,M.jsx)(y,{...e,children:c(e.children)},String(t))}},89236:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(19365),r=(n(96540),n(74848));function a(e){return(0,r.jsx)(r.Fragment,{children:(0,r.jsx)(o.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(11470),r=(n(96540),n(74848));function a(e){return(0,r.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,r.jsx)(o.A,{...e})})}},70627:(e,t,n)=>{n.d(t,{A:()=>o});const o=n.p+"assets/images/edit_mod-8edd0021fea77e67b69b6cb394fbe28b.png"},28453:(e,t,n)=>{n.d(t,{R:()=>i,x:()=>s});var o=n(96540);const r={},a=o.createContext(r);function i(e){const t=o.useContext(a);return o.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function s(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:i(e.components),o.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/fb32500e.c21ec9c3.js b/Doc/assets/js/fb32500e.c21ec9c3.js new file mode 100644 index 00000000..bacc279f --- /dev/null +++ b/Doc/assets/js/fb32500e.c21ec9c3.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkmodio_docs=self.webpackChunkmodio_docs||[]).push([[8950],{44149:(e,t,n)=>{n.r(t),n.d(t,{assets:()=>u,contentTitle:()=>d,default:()=>h,frontMatter:()=>s,metadata:()=>l,toc:()=>c});var o=n(74848),r=n(28453),a=n(27064),i=n(89236);const s={id:"ue-edit-mods",title:"Edit an Existing Mod",slug:"/unreal/getting-started/edit-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/edit-mods.mdx"},d=void 0,l={id:"getting-started/ue-edit-mods",title:"Edit an Existing Mod",description:"Mod details can be edited in-game using SubmitModChangesAsync. This function allows you to edit multiple parameters with a single call. It takes a ModioModID of the mod to edit, a ModioEditModParams containing one or more parameters to be altered, and a callback that will contain an optional updated ModioModInfo object on success.",source:"@site/public/en-us/getting-started/edit-mods.mdx",sourceDirName:"getting-started",slug:"/unreal/getting-started/edit-mods",permalink:"/unreal/getting-started/edit-mods",draft:!1,unlisted:!1,editUrl:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/edit-mods.mdx",tags:[],version:"current",frontMatter:{id:"ue-edit-mods",title:"Edit an Existing Mod",slug:"/unreal/getting-started/edit-mods",custom_edit_url:"https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/edit-mods.mdx"},sidebar:"sidebar",previous:{title:"In-Game Mod Submission",permalink:"/unreal/getting-started/submit-mods"},next:{title:"Error Handling",permalink:"/unreal/getting-started/error-handling"}},u={},c=[];function m(e){const t={a:"a",admonition:"admonition",code:"code",img:"img",p:"p",pre:"pre",strong:"strong",...(0,r.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsxs)(t.p,{children:["Mod details can be edited in-game using ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#submitmodchangesasync-1",children:(0,o.jsx)(t.code,{children:"SubmitModChangesAsync"})}),". This function allows you to edit multiple parameters with a single call. It takes a ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#modiomodid",children:(0,o.jsx)(t.code,{children:"ModioModID"})})," of the mod to edit, a ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#modioeditmodparams",children:(0,o.jsx)(t.code,{children:"ModioEditModParams"})})," containing one or more parameters to be altered, and a callback that will contain an optional updated ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#modiomodinfo",children:(0,o.jsx)(t.code,{children:"ModioModInfo"})})," object on success."]}),"\n",(0,o.jsx)(t.admonition,{type:"note",children:(0,o.jsxs)(t.p,{children:["To update the mod file itself, use ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#submitnewmodfileformod",children:(0,o.jsx)(t.strong,{children:(0,o.jsx)(t.code,{children:"SubmitNewModFileForMod"})})}),". See ",(0,o.jsx)(t.strong,{children:(0,o.jsx)(t.a,{href:"submit-mods#submitting-a-file-for-a-mod",children:"Submitting a file for a mod"})})," for more information."]})}),"\n",(0,o.jsxs)(a.A,{"group-id":"languages",children:[(0,o.jsx)(i.A,{value:"blueprint",label:"Blueprint",children:(0,o.jsx)(t.p,{children:(0,o.jsx)(t.img,{alt:"edit_mod",src:n(33136).A+"",width:"1155",height:"480"})})}),(0,o.jsxs)(i.A,{value:"c++",label:"C++",default:!0,children:[(0,o.jsxs)(t.p,{children:["Note that it would be more appropriate to pass an ",(0,o.jsx)(t.a,{href:"/unreal/refdocs/#modioeditmodparams",children:(0,o.jsx)(t.code,{children:"FModioEditModParams"})})," with your desired parameters into ",(0,o.jsx)(t.code,{children:"UModioManagerSubsystem::EditMod()"}),". This example shows their creation within the function to illustrate their use."]}),(0,o.jsx)(t.pre,{children:(0,o.jsx)(t.code,{className:"language-cpp",children:'void UModioManagerSubsystem::EditMod(FModioModID ModID)\n{\n if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem())\n {\n \tFModioEditModParams EditParams;\n\n \t// Add one or more parameters to edit\n \tEditParams.Name = TEXT("My Edited Mod Name");\n \tEditParams.Summary = TEXT("My edited summary");\n \t\n \tSubsystem->SubmitModChangesAsync(ModID, EditParams, FOnGetModInfoDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnSubmitModChangesComplete));\n }\n}\n\nvoid UModioManagerSubsystem::OnSubmitModChangesComplete(FModioErrorCode ErrorCode, FModioOptionalModInfo UpdatedInfo)\n{\n if (!ErrorCode)\n {\n \t// Mod successfully updated. Can display new details from UpdatedInfo etc.\n }\n}\n\n'})})]})]})]})}function h(e={}){const{wrapper:t}={...(0,r.R)(),...e.components};return t?(0,o.jsx)(t,{...e,children:(0,o.jsx)(m,{...e})}):m(e)}},19365:(e,t,n)=>{n.d(t,{A:()=>i});n(96540);var o=n(18215);const r={tabItem:"tabItem_Ymn6"};var a=n(74848);function i(e){let{children:t,hidden:n,className:i}=e;return(0,a.jsx)("div",{role:"tabpanel",className:(0,o.A)(r.tabItem,i),hidden:n,children:t})}},11470:(e,t,n)=>{n.d(t,{A:()=>j});var o=n(96540),r=n(18215),a=n(23104),i=n(56347),s=n(205),d=n(57485),l=n(31682),u=n(70679);function c(e){return o.Children.toArray(e).filter((e=>"\n"!==e)).map((e=>{if(!e||(0,o.isValidElement)(e)&&function(e){const{props:t}=e;return!!t&&"object"==typeof t&&"value"in t}(e))return e;throw new Error(`Docusaurus error: Bad child <${"string"==typeof e.type?e.type:e.type.name}>: all children of the component should be , and every should have a unique "value" prop.`)}))?.filter(Boolean)??[]}function m(e){const{values:t,children:n}=e;return(0,o.useMemo)((()=>{const e=t??function(e){return c(e).map((e=>{let{props:{value:t,label:n,attributes:o,default:r}}=e;return{value:t,label:n,attributes:o,default:r}}))}(n);return function(e){const t=(0,l.X)(e,((e,t)=>e.value===t.value));if(t.length>0)throw new Error(`Docusaurus error: Duplicate values "${t.map((e=>e.value)).join(", ")}" found in . Every value needs to be unique.`)}(e),e}),[t,n])}function h(e){let{value:t,tabValues:n}=e;return n.some((e=>e.value===t))}function b(e){let{queryString:t=!1,groupId:n}=e;const r=(0,i.W6)(),a=function(e){let{queryString:t=!1,groupId:n}=e;if("string"==typeof t)return t;if(!1===t)return null;if(!0===t&&!n)throw new Error('Docusaurus error: The component groupId prop is required if queryString=true, because this value is used as the search param name. You can also provide an explicit value such as queryString="my-search-param".');return n??null}({queryString:t,groupId:n});return[(0,d.aZ)(a),(0,o.useCallback)((e=>{if(!a)return;const t=new URLSearchParams(r.location.search);t.set(a,e),r.replace({...r.location,search:t.toString()})}),[a,r])]}function p(e){const{defaultValue:t,queryString:n=!1,groupId:r}=e,a=m(e),[i,d]=(0,o.useState)((()=>function(e){let{defaultValue:t,tabValues:n}=e;if(0===n.length)throw new Error("Docusaurus error: the component requires at least one children component");if(t){if(!h({value:t,tabValues:n}))throw new Error(`Docusaurus error: The has a defaultValue "${t}" but none of its children has the corresponding value. Available values are: ${n.map((e=>e.value)).join(", ")}. If you intend to show no default tab, use defaultValue={null} instead.`);return t}const o=n.find((e=>e.default))??n[0];if(!o)throw new Error("Unexpected error: 0 tabValues");return o.value}({defaultValue:t,tabValues:a}))),[l,c]=b({queryString:n,groupId:r}),[p,f]=function(e){let{groupId:t}=e;const n=function(e){return e?`docusaurus.tab.${e}`:null}(t),[r,a]=(0,u.Dv)(n);return[r,(0,o.useCallback)((e=>{n&&a.set(e)}),[n,a])]}({groupId:r}),g=(()=>{const e=l??p;return h({value:e,tabValues:a})?e:null})();(0,s.A)((()=>{g&&d(g)}),[g]);return{selectedValue:i,selectValue:(0,o.useCallback)((e=>{if(!h({value:e,tabValues:a}))throw new Error(`Can't select invalid tab value=${e}`);d(e),c(e),f(e)}),[c,f,a]),tabValues:a}}var f=n(92303);const g={tabList:"tabList__CuJ",tabItem:"tabItem_LNqP"};var x=n(74848);function M(e){let{className:t,block:n,selectedValue:o,selectValue:i,tabValues:s}=e;const d=[],{blockElementScrollPositionUntilNextRender:l}=(0,a.a_)(),u=e=>{const t=e.currentTarget,n=d.indexOf(t),r=s[n].value;r!==o&&(l(t),i(r))},c=e=>{let t=null;switch(e.key){case"Enter":u(e);break;case"ArrowRight":{const n=d.indexOf(e.currentTarget)+1;t=d[n]??d[0];break}case"ArrowLeft":{const n=d.indexOf(e.currentTarget)-1;t=d[n]??d[d.length-1];break}}t?.focus()};return(0,x.jsx)("ul",{role:"tablist","aria-orientation":"horizontal",className:(0,r.A)("tabs",{"tabs--block":n},t),children:s.map((e=>{let{value:t,label:n,attributes:a}=e;return(0,x.jsx)("li",{role:"tab",tabIndex:o===t?0:-1,"aria-selected":o===t,ref:e=>d.push(e),onKeyDown:c,onClick:u,...a,className:(0,r.A)("tabs__item",g.tabItem,a?.className,{"tabs__item--active":o===t}),children:n??t},t)}))})}function v(e){let{lazy:t,children:n,selectedValue:r}=e;const a=(Array.isArray(n)?n:[n]).filter(Boolean);if(t){const e=a.find((e=>e.props.value===r));return e?(0,o.cloneElement)(e,{className:"margin-top--md"}):null}return(0,x.jsx)("div",{className:"margin-top--md",children:a.map(((e,t)=>(0,o.cloneElement)(e,{key:t,hidden:e.props.value!==r})))})}function y(e){const t=p(e);return(0,x.jsxs)("div",{className:(0,r.A)("tabs-container",g.tabList),children:[(0,x.jsx)(M,{...t,...e}),(0,x.jsx)(v,{...t,...e})]})}function j(e){const t=(0,f.A)();return(0,x.jsx)(y,{...e,children:c(e.children)},String(t))}},89236:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(19365),r=(n(96540),n(74848));function a(e){return(0,r.jsx)(r.Fragment,{children:(0,r.jsx)(o.A,{className:"tw-rounded-md",...e})})}},27064:(e,t,n)=>{n.d(t,{A:()=>a});var o=n(11470),r=(n(96540),n(74848));function a(e){return(0,r.jsx)("div",{className:"tw-border tw-border-solid tw-border-skyblue tw-rounded-md tw-bg-darkslategray tw-p-4 tw-mb-4",children:(0,r.jsx)(o.A,{...e})})}},33136:(e,t,n)=>{n.d(t,{A:()=>o});const o=n.p+"assets/images/edit_mod-8edd0021fea77e67b69b6cb394fbe28b.png"},28453:(e,t,n)=>{n.d(t,{R:()=>i,x:()=>s});var o=n(96540);const r={},a=o.createContext(r);function i(e){const t=o.useContext(a);return o.useMemo((function(){return"function"==typeof e?e(t):{...t,...e}}),[t,e])}function s(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:i(e.components),o.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/Doc/assets/js/runtime~main.232e3d14.js b/Doc/assets/js/runtime~main.232e3d14.js new file mode 100644 index 00000000..ebae73c4 --- /dev/null +++ b/Doc/assets/js/runtime~main.232e3d14.js @@ -0,0 +1 @@ +(()=>{"use strict";var e,a,t,r,c,f={},d={};function o(e){var a=d[e];if(void 0!==a)return a.exports;var t=d[e]={exports:{}};return f[e].call(t.exports,t,t.exports,o),t.exports}o.m=f,e=[],o.O=(a,t,r,c)=>{if(!t){var f=1/0;for(i=0;i=c)&&Object.keys(o.O).every((e=>o.O[e](t[b])))?t.splice(b--,1):(d=!1,c0&&e[i-1][2]>c;i--)e[i]=e[i-1];e[i]=[t,r,c]},o.n=e=>{var a=e&&e.__esModule?()=>e.default:()=>e;return o.d(a,{a:a}),a},t=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,o.t=function(e,r){if(1&r&&(e=this(e)),8&r)return e;if("object"==typeof e&&e){if(4&r&&e.__esModule)return e;if(16&r&&"function"==typeof e.then)return e}var c=Object.create(null);o.r(c);var f={};a=a||[null,t({}),t([]),t(t)];for(var d=2&r&&e;"object"==typeof d&&!~a.indexOf(d);d=t(d))Object.getOwnPropertyNames(d).forEach((a=>f[a]=()=>e[a]));return f.default=()=>e,o.d(c,f),c},o.d=(e,a)=>{for(var t in a)o.o(a,t)&&!o.o(e,t)&&Object.defineProperty(e,t,{enumerable:!0,get:a[t]})},o.f={},o.e=e=>Promise.all(Object.keys(o.f).reduce(((a,t)=>(o.f[t](e,a),a)),[])),o.u=e=>"assets/js/"+({433:"4b65d6a2",673:"0879cf16",881:"0ce884cc",957:"c141421f",1235:"a7456010",1462:"d99e1931",1493:"8ac087ef",1530:"732ed63c",1567:"22dd74f7",1697:"05ae0fb1",1991:"562207a5",2138:"1a4e3797",2381:"1da2407e",3124:"4f8ccd8c",3586:"8f59627d",4101:"b84ad1c9",4553:"68befa13",4944:"0c9c4b9e",5237:"2969dc70",5533:"4dd573c0",5633:"b2348a3c",5742:"aba21aa0",5753:"7b283257",5973:"433ca74c",6647:"9e4dfacd",6783:"65ba71d8",7098:"a7bd4aaa",7288:"b69370a5",7742:"4d3f9fe4",8156:"c5d35168",8335:"546be59f",8401:"17896441",8840:"cddf829a",8950:"fb32500e",9048:"a94703ab",9647:"5e95c892"}[e]||e)+"."+{416:"9b625bca",433:"3e711868",673:"dca6b1a0",881:"b95aa83e",957:"7bacecc8",1169:"fc010fc9",1176:"16f7cb82",1235:"a5521ac3",1245:"2ad1e165",1303:"812964c2",1331:"cdc427f8",1398:"5b1665cf",1462:"11845b15",1493:"4e5b794d",1530:"d993f088",1567:"ba359d28",1697:"2f8e1093",1946:"39e60b8a",1991:"a07c0f1f",2130:"30a4bfc0",2138:"b722f873",2237:"1e6df7a1",2376:"69b4a2e9",2381:"5245f7dc",2453:"81a3cb3e",2548:"31c1f5eb",2843:"a28e8e47",2925:"8c62ed1d",2983:"3778392d",3068:"e7d489d8",3124:"4c953df7",3586:"be8524c0",3626:"bc638be7",3706:"cfca602d",4101:"ddb378a0",4162:"93988957",4553:"ab7ee46b",4741:"06bee63e",4943:"c267289b",4944:"48f78565",5237:"6db03ace",5533:"3ed96b98",5633:"03af1dd0",5742:"0904d0f0",5753:"06f7fbec",5973:"5daf400e",6420:"52d71345",6647:"74a35171",6783:"3c23f95a",6788:"9b355e3d",6803:"e5a8a35f",7098:"6c02be9e",7106:"e9da2147",7288:"cd361d24",7426:"da722649",7742:"a288788f",8055:"af5b0b47",8156:"d3253ce3",8158:"54ebe1f4",8335:"aa883044",8337:"80c4cdb3",8401:"21dc7276",8478:"58361f63",8635:"8dc2e823",8810:"217c3214",8840:"6ee7539f",8869:"ad329b9e",8913:"bd2bcab1",8950:"c21ec9c3",9048:"3a76c072",9647:"6dcb10e4",9689:"2c1e089a"}[e]+".js",o.miniCssF=e=>{},o.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),o.o=(e,a)=>Object.prototype.hasOwnProperty.call(e,a),r={},c="modio-docs:",o.l=(e,a,t,f)=>{if(r[e])r[e].push(a);else{var d,b;if(void 0!==t)for(var n=document.getElementsByTagName("script"),i=0;i{d.onerror=d.onload=null,clearTimeout(s);var c=r[e];if(delete r[e],d.parentNode&&d.parentNode.removeChild(d),c&&c.forEach((e=>e(t))),a)return a(t)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:d}),12e4);d.onerror=l.bind(null,d.onerror),d.onload=l.bind(null,d.onload),b&&document.head.appendChild(d)}},o.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},(()=>{var e;o.g.importScripts&&(e=o.g.location+"");var a=o.g.document;if(!e&&a&&(a.currentScript&&(e=a.currentScript.src),!e)){var t=a.getElementsByTagName("script");if(t.length)for(var r=t.length-1;r>-1&&(!e||!/^http(s?):/.test(e));)e=t[r--].src}if(!e)throw new Error("Automatic publicPath is not supported in this browser");e=e.replace(/#.*$/,"").replace(/\?.*$/,"").replace(/\/[^\/]+$/,"/"),o.p=e+"../../"})(),o.gca=function(e){return e={17896441:"8401","4b65d6a2":"433","0879cf16":"673","0ce884cc":"881",c141421f:"957",a7456010:"1235",d99e1931:"1462","8ac087ef":"1493","732ed63c":"1530","22dd74f7":"1567","05ae0fb1":"1697","562207a5":"1991","1a4e3797":"2138","1da2407e":"2381","4f8ccd8c":"3124","8f59627d":"3586",b84ad1c9:"4101","68befa13":"4553","0c9c4b9e":"4944","2969dc70":"5237","4dd573c0":"5533",b2348a3c:"5633",aba21aa0:"5742","7b283257":"5753","433ca74c":"5973","9e4dfacd":"6647","65ba71d8":"6783",a7bd4aaa:"7098",b69370a5:"7288","4d3f9fe4":"7742",c5d35168:"8156","546be59f":"8335",cddf829a:"8840",fb32500e:"8950",a94703ab:"9048","5e95c892":"9647"}[e]||e,o.p+o.u(e)},(()=>{var e={5354:0,1869:0};o.f.j=(a,t)=>{var r=o.o(e,a)?e[a]:void 0;if(0!==r)if(r)t.push(r[2]);else if(/^(1869|5354)$/.test(a))e[a]=0;else{var c=new Promise(((t,c)=>r=e[a]=[t,c]));t.push(r[2]=c);var f=o.p+o.u(a),d=new Error;o.l(f,(t=>{if(o.o(e,a)&&(0!==(r=e[a])&&(e[a]=void 0),r)){var c=t&&("load"===t.type?"missing":t.type),f=t&&t.target&&t.target.src;d.message="Loading chunk "+a+" failed.\n("+c+": "+f+")",d.name="ChunkLoadError",d.type=c,d.request=f,r[1](d)}}),"chunk-"+a,a)}},o.O.j=a=>0===e[a];var a=(a,t)=>{var r,c,f=t[0],d=t[1],b=t[2],n=0;if(f.some((a=>0!==e[a]))){for(r in d)o.o(d,r)&&(o.m[r]=d[r]);if(b)var i=b(o)}for(a&&a(t);n{"use strict";var e,a,t,c,r,f={},o={};function d(e){var a=o[e];if(void 0!==a)return a.exports;var t=o[e]={exports:{}};return f[e].call(t.exports,t,t.exports,d),t.exports}d.m=f,e=[],d.O=(a,t,c,r)=>{if(!t){var f=1/0;for(i=0;i=r)&&Object.keys(d.O).every((e=>d.O[e](t[b])))?t.splice(b--,1):(o=!1,r0&&e[i-1][2]>r;i--)e[i]=e[i-1];e[i]=[t,c,r]},d.n=e=>{var a=e&&e.__esModule?()=>e.default:()=>e;return d.d(a,{a:a}),a},t=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,d.t=function(e,c){if(1&c&&(e=this(e)),8&c)return e;if("object"==typeof e&&e){if(4&c&&e.__esModule)return e;if(16&c&&"function"==typeof e.then)return e}var r=Object.create(null);d.r(r);var f={};a=a||[null,t({}),t([]),t(t)];for(var o=2&c&&e;"object"==typeof o&&!~a.indexOf(o);o=t(o))Object.getOwnPropertyNames(o).forEach((a=>f[a]=()=>e[a]));return f.default=()=>e,d.d(r,f),r},d.d=(e,a)=>{for(var t in a)d.o(a,t)&&!d.o(e,t)&&Object.defineProperty(e,t,{enumerable:!0,get:a[t]})},d.f={},d.e=e=>Promise.all(Object.keys(d.f).reduce(((a,t)=>(d.f[t](e,a),a)),[])),d.u=e=>"assets/js/"+({433:"4b65d6a2",673:"0879cf16",881:"0ce884cc",957:"c141421f",1235:"a7456010",1462:"d99e1931",1493:"8ac087ef",1530:"732ed63c",1567:"22dd74f7",1697:"05ae0fb1",1991:"562207a5",2138:"1a4e3797",2381:"1da2407e",3124:"4f8ccd8c",3586:"8f59627d",4101:"b84ad1c9",4553:"68befa13",4944:"0c9c4b9e",5237:"2969dc70",5533:"4dd573c0",5633:"b2348a3c",5742:"aba21aa0",5753:"7b283257",5973:"433ca74c",6647:"9e4dfacd",6783:"65ba71d8",7098:"a7bd4aaa",7288:"b69370a5",7742:"4d3f9fe4",8156:"c5d35168",8335:"546be59f",8401:"17896441",8840:"cddf829a",8950:"fb32500e",9048:"a94703ab",9647:"5e95c892"}[e]||e)+"."+{416:"9b625bca",433:"b96e2355",673:"dca6b1a0",881:"b95aa83e",957:"7bacecc8",1169:"fc010fc9",1176:"16f7cb82",1235:"a5521ac3",1245:"2ad1e165",1303:"812964c2",1331:"cdc427f8",1398:"5b1665cf",1462:"ad6d4183",1493:"27a161e3",1530:"5c782035",1567:"ba359d28",1697:"2f8e1093",1946:"39e60b8a",1991:"b60eadca",2130:"30a4bfc0",2138:"b722f873",2237:"1e6df7a1",2376:"69b4a2e9",2381:"9927ed81",2453:"81a3cb3e",2548:"31c1f5eb",2843:"a28e8e47",2925:"8c62ed1d",2983:"3778392d",3068:"e7d489d8",3124:"be71a477",3586:"f96d3965",3626:"bc638be7",3706:"cfca602d",4101:"2d53dc4b",4162:"93988957",4553:"ab7ee46b",4741:"06bee63e",4943:"c267289b",4944:"33716226",5237:"3fd40665",5533:"c971e0ef",5633:"612bac9c",5742:"0904d0f0",5753:"06f7fbec",5973:"ab83666c",6420:"52d71345",6647:"0af8d679",6783:"9f5de872",6788:"9b355e3d",6803:"e5a8a35f",7098:"6c02be9e",7106:"e9da2147",7288:"93b89783",7426:"da722649",7742:"a288788f",8055:"af5b0b47",8156:"34c70772",8158:"54ebe1f4",8335:"aab0b80d",8337:"80c4cdb3",8401:"21dc7276",8478:"58361f63",8635:"8dc2e823",8810:"217c3214",8840:"11002d8d",8869:"ad329b9e",8913:"bd2bcab1",8950:"b8d34aff",9048:"3a76c072",9647:"6dcb10e4",9689:"2c1e089a"}[e]+".js",d.miniCssF=e=>{},d.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),d.o=(e,a)=>Object.prototype.hasOwnProperty.call(e,a),c={},r="modio-docs:",d.l=(e,a,t,f)=>{if(c[e])c[e].push(a);else{var o,b;if(void 0!==t)for(var n=document.getElementsByTagName("script"),i=0;i{o.onerror=o.onload=null,clearTimeout(s);var r=c[e];if(delete c[e],o.parentNode&&o.parentNode.removeChild(o),r&&r.forEach((e=>e(t))),a)return a(t)},s=setTimeout(l.bind(null,void 0,{type:"timeout",target:o}),12e4);o.onerror=l.bind(null,o.onerror),o.onload=l.bind(null,o.onload),b&&document.head.appendChild(o)}},d.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},(()=>{var e;d.g.importScripts&&(e=d.g.location+"");var a=d.g.document;if(!e&&a&&(a.currentScript&&(e=a.currentScript.src),!e)){var t=a.getElementsByTagName("script");if(t.length)for(var c=t.length-1;c>-1&&(!e||!/^http(s?):/.test(e));)e=t[c--].src}if(!e)throw new Error("Automatic publicPath is not supported in this browser");e=e.replace(/#.*$/,"").replace(/\?.*$/,"").replace(/\/[^\/]+$/,"/"),d.p=e+"../../"})(),d.gca=function(e){return e={17896441:"8401","4b65d6a2":"433","0879cf16":"673","0ce884cc":"881",c141421f:"957",a7456010:"1235",d99e1931:"1462","8ac087ef":"1493","732ed63c":"1530","22dd74f7":"1567","05ae0fb1":"1697","562207a5":"1991","1a4e3797":"2138","1da2407e":"2381","4f8ccd8c":"3124","8f59627d":"3586",b84ad1c9:"4101","68befa13":"4553","0c9c4b9e":"4944","2969dc70":"5237","4dd573c0":"5533",b2348a3c:"5633",aba21aa0:"5742","7b283257":"5753","433ca74c":"5973","9e4dfacd":"6647","65ba71d8":"6783",a7bd4aaa:"7098",b69370a5:"7288","4d3f9fe4":"7742",c5d35168:"8156","546be59f":"8335",cddf829a:"8840",fb32500e:"8950",a94703ab:"9048","5e95c892":"9647"}[e]||e,d.p+d.u(e)},(()=>{var e={5354:0,1869:0};d.f.j=(a,t)=>{var c=d.o(e,a)?e[a]:void 0;if(0!==c)if(c)t.push(c[2]);else if(/^(1869|5354)$/.test(a))e[a]=0;else{var r=new Promise(((t,r)=>c=e[a]=[t,r]));t.push(c[2]=r);var f=d.p+d.u(a),o=new Error;d.l(f,(t=>{if(d.o(e,a)&&(0!==(c=e[a])&&(e[a]=void 0),c)){var r=t&&("load"===t.type?"missing":t.type),f=t&&t.target&&t.target.src;o.message="Loading chunk "+a+" failed.\n("+r+": "+f+")",o.name="ChunkLoadError",o.type=r,o.request=f,c[1](o)}}),"chunk-"+a,a)}},d.O.j=a=>0===e[a];var a=(a,t)=>{var c,r,f=t[0],o=t[1],b=t[2],n=0;if(f.some((a=>0!==e[a]))){for(c in o)d.o(o,c)&&(d.m[c]=o[c]);if(b)var i=b(d)}for(a&&a(t);nGetIdentityInterface(); + +OnlineIdentity->AddOnLoginCompleteDelegate_Handle(0, NativeLoginComplete); +OnlineIdentity->Login(0, FOnlineAccountCredentials()); + +// In the NativeLoginComplete delegate, however you set it up, if it was a successful login, you can then obtain the Server Auth Token as follows: +FString IdToken, ServerAuthToken; + +UserAccount.Get()->GetAuthAttribute(AUTH_ATTR_ID_TOKEN, IdToken); +UserAccount.Get()->GetAuthAttribute(AUTH_ATTR_AUTHORIZATION_CODE, ServerAuthToken); + +// Now perform Auth to the mod.io service +AuthParams.AuthToken = ServerAuthToken; +AuthParams.bUserHasAcceptedTerms = true; + +// Alternatively you could set AuthParams.AuthToken = IdToken and use EModioAuthenticationProvider::GoogleIDToken +ModioSubsystem->AuthenticateUserExternalAsync(AuthParams, EModioAuthenticationProvider::GoogleServerSideToken, FOnErrorOnlyDelegateFast::CreateUObject(this, OnAuthenticationComplete)); +``` + +## Single Sign-On (Pre-5.4) + +For Unreal Engine versions before 5.4, you need to make some engine-level modifications in order for Android SSO to work. You must be using a source build rather than an engine build. + +Open `Engine\Plugins\Online\OnlineSubsystemGoogle\Source\ThirdParty\Android\Java\GoogleLogin.java` and change the `init` method to uncomment `.requestServerAuthCode(serverClientId)`. + +Change the `getAuthTokenJsonStr` method to the following: + +```cpp +private String getAuthTokenJsonStr(GoogleSignInAccount acct) +{ + if (acct != null) + { + return "{\"access_token\":\""+ acct.getServerAuthCode() + "\"," + + "\"refresh_token\":\"androidInternal\"," + + "\"id_token\":\""+ acct.getIdToken() + "\"}"; + } + return ""; +} +``` + +Finally, Open OnlineSubsystemGoogle.uplugin and add Android to the `PlatformAllowList` array. + +### Configuration + +Once you have made the above changes, update your AndroidEngine.ini to contain the following: + +``` +[OnlineSubsystem] +DefaultPlatformService=GooglePlay +NativePlatformService=Google + +[OnlineSubsystemGoogle] +bEnabled=True +ClientId=your-client-id-here +ServerClientId=your-server-client-id-here + +[OnlineSubsystemGoogle.OnlineIdentityGoogle] ++ScopeFields="email" ++ScopeFields="profile" ++ScopeFields="https://www.googleapis.com/auth/userinfo.email" ++ScopeFields="https://www.googleapis.com/auth/userinfo.profile" +``` + +This will ensure that you are using OnlineSubsystemGoogle as the native platform OSS in order to perform SSO. + +Ensure that you replace the ClientId and ServerClientId with your client IDs from Google Cloud Console. The ClientId should be the Android OAuth Credential that you have linked to your title in Google Play Console. ServerClientId should be the Web Application OAuth Credential that you configured. + +### Getting the Token + +Once you have configured the above options, you should be able to obtain a server side token that you can use for auth with mod.io. + +```cpp +const IOnlineSubsystem* OnlineSubsystem = IOnlineSubsystem::GetByPlatform(); +const IOnlineIdentityPtr OnlineIdentity = OnlineSubsystem->GetIdentityInterface(); + +OnlineIdentity->AddOnLoginCompleteDelegate_Handle(0, NativeLoginComplete); +OnlineIdentity->Login(0, FOnlineAccountCredentials()); + +// In the NativeLoginComplete delegate, however you set it up, if it was a successful login, you can then obtain the Server Auth Token as follows: +const FString PlatformToken = OnlineIdentity->GetAuthToken(0); + +// Now perform Auth to the mod.io service +AuthParams.AuthToken = PlatformToken; +AuthParams.bUserHasAcceptedTerms = true; + +ModioSubsystem->AuthenticateUserExternalAsync(AuthParams, EModioAuthenticationProvider::GoogleServerSideToken, FOnErrorOnlyDelegateFast::CreateUObject(this, OnAuthenticationComplete)); +``` \ No newline at end of file diff --git a/Doc/doc_root/en-us/asset-support-for-mods.mdx b/Doc/doc_root/en-us/asset-support-for-mods.mdx new file mode 100644 index 00000000..75618445 --- /dev/null +++ b/Doc/doc_root/en-us/asset-support-for-mods.mdx @@ -0,0 +1,124 @@ +--- +id: ue-asset-support-for-mods +title: Asset Support for Mods +slug: /unreal/asset-support-for-mods/ +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/asset-support-for-mods +--- + +## Overview + +The main factor for supporting assets is dependency: only dependencies to the project's content are allowed, but not engine's dependency. When trying to package any asset that has engine dependency, it will fail. + +Some assets have internal dependencies which can't be bypassed without modifying the engine's source code. For example, when trying to create a directional light or point light in the level, you'll have an `EmptyActor` dependency to the engine - packaging of that asset and consequently the whole mod will fail. Therefore, you should generally avoid using any engine's assets dependency (except for very basic ones, such as dependency to `AActor` class), and instead create your own classes in your project and implement the necessary functionality there, and allow the mod to use those classes instead of the engine's ones. + +## Supported Asset Types + +### Allowed Actors In Levels + +These actors, including custom actors derived from them and saved in the project or mod, can be placed in levels without causing packaging issues in the mod: + +1. `AStaticMeshActor` +2. `APlayerStart` +3. `APawn` + - `ACharacter` + - `ADefaultPawn` + - All other derived actors +4. `ABrush` & `AVolume` + - `AKillZVolume` + - `ABlockingVolume` + - `AAudioVolume` + - `ACullDistanceVolume` + - `APostProcessVolume` + - All other derived actors +5. `ADecalActor` +6. `AInfo` + - `AVolumetricCloud` + - `AExponentialHeightFog` + - Other supported derived actors +7. `ATriggerBase` + - `ATriggerBox` + - `ATriggerCapsule` + - `ATriggerSphere` + - All other derived actors + +### Allowed Components in Actors + +These components, including custom components derived from them and saved in the project or mod, can be placed in actors without causing packaging issues in the mod: + +1. `USceneComponent` + - `UPrimitiveComponent` + - `UTextRenderComponent` + - `UArrowComponent` + - `UShapeComponent` (collisions) + - `UBoxComponent` + - `UCapsuleComponent` + - `USphereComponent` + - `UMeshComponent` + - `UStaticMeshComponent` + - `UWidgetComponent` + - All other derived components + - `UCameraComponent` + - `UChildActorComponent` + - `UText3DComponent` + - Other supported derived components +2. `UWidgetComponent` +3. `UCameraComponent` +4. `UDecalComponent` +5. `UMovementComponent` + - `UPawnMovementComponent` + - `UCharacterMovementComponent` + - All other derived components +- All other derived components + +## Unsupported Asset Types + +### Prohibited Actors in Levels + +These actors, including custom actors derived from them and saved in the project or mod, should not be placed in levels as they will likely cause packaging issues in the mod: + +1. `ALight` + - `ADirectionalLight` + - `APointLight` + - `ARectLight` + - `ASpotLight` +2. `AInfo` + - `ASkyLight` + - `ASkyAtmosphere` + - Other unsupported derived actors +3. `ASkyAtmosphere` +4. `ATextRenderActor` + - Other unsupported derived actors + +### Prohibited Components in Actors + +These components, including custom components derived from them and saved in the project or mod, should not be placed in actors as they will likely cause packaging issues in the mod: + +1. `USceneComponent` + - `ULightComponentBase` + - `UDirectionalLightComponent` + - `ULightComponent` + - `ULocalLightComponent` + - `UPointLightComponent` + - `URectLightComponent` + - `USkyLightComponent` + - `USpotLightComponent` + - `USynthComponent` + - `UAudioComponent` + - Other unsupported derived components + +## Asset Support in Packages (UPackage) + +Generally, all possible asset types are supported to be included in the plugin after the packaging. The tested ones are listed below, but any other assets without dependency on the engine are supported as well, such as manually created ones in the mod itself: + +- Actor +- Pawn +- Character +- Actor Component +- Scene Component +- Level +- Material +- Widget +- Texture +- Sound Wave +- Static Mesh +- Skeletal Mesh \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/browsing-mods.mdx b/Doc/doc_root/en-us/getting-started/browsing-mods.mdx new file mode 100644 index 00000000..b4e3b86c --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/browsing-mods.mdx @@ -0,0 +1,56 @@ +--- +id: ue-browsing-mods +title: Browsing Mods +slug: /unreal/getting-started/browsing-mods +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/browsing-mods.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +After [initializing the plugin](initialization) and [authenticating a user](user-authentication), you can query the available mods using [`ListAllModsAsync`](/unreal/refdocs/#listallmodsasync). + +`ListAllModsAsync` supports filtering by name, tag, author, mature content, and more using [`ModioFilterParams`](/unreal/refdocs/#modiofilterparams). You can sort results as specified by `ModioSortFieldType`, and request paginated or indexed results. By default, `ModioFilterParams` asks for the first 100 results (the maximum number returnable in a query) sorted by `ModioModID`. + + + +The primary way this is done is through [K2_ListAllModsAsync](/unreal/refdocs/#listallmodsasync). + +![list_all_mods](img/list_all_mods.png) + +To search for a specific query string, sort in a different order, or combine different filters, you can use a [ModioFilterParams](/unreal/refdocs/#modiofilterparams) object like this: + +![list_all_mods_filter](img/list_all_mods_filter.png) + + + + ```cpp +void UModioManagerSubsystem::ListAllMods() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + FModioFilterParams Filter; + // Build the filter by chaining together multiple calls + Filter.PagedResults(1, 5).IndexedResults(3, 5).WithTags("Multiplayer").SortBy(EModioSortFieldType::ID, EModioSortDirection::Descending); + + Subsystem->ListAllModsAsync(Filter, FOnListAllModsDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnListAllModsComplete)); + } +} + +void UModioManagerSubsystem::OnListAllModsComplete(FModioErrorCode ErrorCode, TOptional OptionalModList) +{ + // Ensure ListAllModsAsync was successful + if (!ErrorCode) + { + // ModList is guaranteed to be valid if there is no error + TArray ModInfoArray = OptionalModList.GetValue().GetRawList(); + + // Do something with ModInfoArray + + // You can use OptionalModList().GetValue().Paged related methods to make further paginated requests if required + } +} + + ``` + + \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/edit-mods.mdx b/Doc/doc_root/en-us/getting-started/edit-mods.mdx new file mode 100644 index 00000000..ad475ef4 --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/edit-mods.mdx @@ -0,0 +1,52 @@ +--- +id: ue-edit-mods +title: Edit an Existing Mod +slug: /unreal/getting-started/edit-mods +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/edit-mods.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +Mod details can be edited in-game using [`SubmitModChangesAsync`](/unreal/refdocs/#submitmodchangesasync-1). This function allows you to edit multiple parameters with a single call. It takes a [`ModioModID`](/unreal/refdocs/#modiomodid) of the mod to edit, a [`ModioEditModParams`](/unreal/refdocs/#modioeditmodparams) containing one or more parameters to be altered, and a callback that will contain an optional updated [`ModioModInfo`](/unreal/refdocs/#modiomodinfo) object on success. + +:::note +To update the mod file itself, use [**`SubmitNewModFileForMod`**](/unreal/refdocs/#submitnewmodfileformod). See **[Submitting a file for a mod](submit-mods#submitting-a-file-for-a-mod)** for more information. +::: + + + +![edit_mod](img/edit_mod.png) + + + + +Note that it would be more appropriate to pass an [`FModioEditModParams`](/unreal/refdocs/#modioeditmodparams) with your desired parameters into `UModioManagerSubsystem::EditMod()`. This example shows their creation within the function to illustrate their use. + + ```cpp +void UModioManagerSubsystem::EditMod(FModioModID ModID) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + FModioEditModParams EditParams; + + // Add one or more parameters to edit + EditParams.Name = TEXT("My Edited Mod Name"); + EditParams.Summary = TEXT("My edited summary"); + + Subsystem->SubmitModChangesAsync(ModID, EditParams, FOnGetModInfoDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnSubmitModChangesComplete)); + } +} + +void UModioManagerSubsystem::OnSubmitModChangesComplete(FModioErrorCode ErrorCode, FModioOptionalModInfo UpdatedInfo) +{ + if (!ErrorCode) + { + // Mod successfully updated. Can display new details from UpdatedInfo etc. + } +} + +``` + + + \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/error-handling.mdx b/Doc/doc_root/en-us/getting-started/error-handling.mdx new file mode 100644 index 00000000..0e8332cb --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/error-handling.mdx @@ -0,0 +1,68 @@ +--- +id: ue-error-handling +title: Error Handling +slug: /unreal/getting-started/error-handling +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/error-handling.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +:::note +Error handling in Blueprint is still undergoing improvement. Please let us know if you have any feedback or suggestions for improving it! +::: + +Many of the plugin's functions utilize the [`ModioErrorCode`](/unreal/refdocs/#modioerrorcode) type. For example, all [async](/unreal/getting-started/plugin-structure/#non-blocking-asynchronous-interface) functions take a `ModioErrorCode` as the first parameter for the function's delegate. This is essentially an opaque wrapper around a numeric error code with a category and an associated string message. + +### Checking for errors + +A "truthy" error code represents an error. In Blueprint, you can check if a `ModioErrorCode` represents a success or failure by using the `IsError` node. + + + + + +![is_error](img/is_error.png) + + + ```cpp +void UModioManagerSubsystem::MyCallback(FModioErrorCode ErrorCode) +{ + if (ErrorCode) + { + // Failure + } + else + { + // Success + } +} +``` + + + +### Inspecting `ModioErrorCode` more deeply + +Sometimes this information will be all that is required; just a simple 'success/fail' that you can handle. + +In many cases, however, you will want to perform some degree of inspection on an `ModioErrorCode` to determine specific information about that error - if nothing else, so that you can display a reason for the failure to the end user. + +#### Semantic queries + +In your application, there are many errors that will likely be handled the same way. For example, you probably don’t need to handle different network errors in different ways. The semantics of networking errors are largely "try the function again later". + +This is where [`ErrorCodeMatches`](/unreal/refdocs/#error-code-matches) comes in. It allows you to query if the error satisfies a particular condition, such as "does this code represent some kind of networking error", without needing to explicitly check the code against all the individual errors in the category. + +![error_code_matches_network_error](img/error_code_matches_network_error.png) + +By querying if an error meets a specific condition, you can focus on handling a family of errors (in this case, network transmission errors) without needing to deal with individual errors within that group. No more manually checking against individual HttpError values, just a single query. + +These semantic checks help you to consolidate your error handling into limited set of generic error handlers rather than dealing with each potential outcome individually. + +![error_code_matches_chaining](img/error_code_matches_chaining.png) + +### Parameter validation errors + +Some of the plugin functions may return errors that indicate a parameter or data validation failure. For these cases, the plugin parses the error response from the mod.io API and stores the information about which parameters failed validation until the next network request is performed. If a plugin function returns an error which matches `EModioErrorCondition::InvalidArgsError`, you can call `GetLastValidationError` in your callback to retrieve those errors and display appropriate feedback to the end user. + +![get_last_validation_error](img/get_last_validation_error.png) \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/getting-started.mdx b/Doc/doc_root/en-us/getting-started/getting-started.mdx new file mode 100644 index 00000000..18e70c48 --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/getting-started.mdx @@ -0,0 +1,23 @@ +--- +id: ue-getting-started +title: Getting Started +slug: /unreal/getting-started/ +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/getting-started.mdx +--- + +Below are a series of quick-start guides with code samples demonstrating the mod.io Unreal Engine plugin's core functionality. + +### Quick-Start Guides + +- [Plugin Structure & Concepts](plugin-structure) +- [Initialization & Teardown](initialization) +- [User Authentication](user-authentication) +- [Browsing Mods](browsing-mods) +- [Mod Subscriptions & Management](mod-subscriptions) +- [Error Handling](error-handling) +- [In-Game Mod Submission](submit-mods) +- [Edit an Existing Mod](edit-mods) +- [Monetization](monetization) +- [Temporary Mod Sets](temporary-mods) +- [Muting & Unmuting a User](mute-user) +- [Metrics Play Sessions](metrics-play-sessions) \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/initialization.mdx b/Doc/doc_root/en-us/getting-started/initialization.mdx new file mode 100644 index 00000000..6a0d72b5 --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/initialization.mdx @@ -0,0 +1,100 @@ +--- +id: ue-initialization +title: Initialization & Teardown +slug: /unreal/getting-started/initialization +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/initialization.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +## Best practices + +This guide describes the mod.io Unreal Engine plugin's basic event loop, intialization, and shutdown functions. Ideally, you should wrap these functions in a separate subsystem (e.g. `ModioManagerSubsystem`) which is responsible for managing the full initialization and authentication flow for your game. + +For more information, see our [User Authentication](user-authentication) quick-start guide. + +## Calling RunPendingHandlers +Before initializing the plugin, ensure [`RunPendingHandlers`](/unreal/refdocs/#run-pending-handlers) is being continually called. You can do this yourself in your project’s main loop, or by setting [`Use Background Thread`](/unreal/installation-and-setup/#using-a-background-thread) to true in the mod.io project settings. For best performance, `RunPendingHandlers` should be called at least once per frame. See [Maintaining the plugin event loop](/unreal/getting-started/plugin-structure/#maintaining-the-plugin-event-loop) for additional information. + +## Initialization +The mod.io Unreal Engine plugin is initialized by calling [`InitializeAsync`](/unreal/refdocs/#initializeasync), passing in a [`ModioInitializeOptions`](/unreal/refdocs/#modioinitializeoptions) with the relevant information for the current session, and a callback containing a [`ModioErrorCode`](/unreal/refdocs/#modioerrorcode) that will contain the result of the initialization on completion. + +Use [`GetProjectInitializeOptionsForSessionId`](/unreal/refdocs/#get-project-initialize-options-for-session-id) to initialize the plugin using the information entered in the [mod.io project settings](/unreal/installation-and-setup/#plugin-configuration). On Windows, [`GetDefaultSessionIdWindows`](/unreal/refdocs/#get-default-session-id-windows) can be used as the [Session ID](/unreal/getting-started/plugin-structure/#session-ids). Some platforms also require additional parameters added to `ModioInitializeOptions.ExtendedInitializationParameters`. Refer to the relevant [platform documentation](/platforms/) for more information. + + + + + ```cpp +void UModioManagerSubsystem::Init() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->InitializeAsync(UModioSDKLibrary::GetProjectInitializeOptionsForSessionId(UModioSDKLibrary::GetDefaultSessionIdWindows()), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnInitCallback)); + } + UE_LOG(LogModioGame, Log, TEXT("Initializing mod.io")); +} + +void UModioManagerSubsystem::OnInitCallback(FModioErrorCode ErrorCode) +{ + UE_LOG(LogModioGame, Log, TEXT("mod.io initialization complete with result: %s"), *ErrorCode.GetErrorMessage()); +} + +void UModioManagerSubsystem::Tick(float DeltaTime) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->RunPendingHandlers(); + } +} +``` + + + ![run_pending_handlers](img/run_pending_handlers.png) + + ![initasync_getoptions](img/initasync_getoptions.png) + + + + +:::note +* The error-handling in this sample has been omitted. See our [**Error Handling quick-start guide**](error-handling) for more information. +* `InitializeAsync` is a asynchronous function, therefore you *must* wait for the callback for confirmation that the initialization is complete. +::: + +## Shutdown + +The mod.io Unreal Engine plugin is shut down by calling [`ShutdownAsync`](/unreal/refdocs/#shutdownasync) in a similar fashion. + + + + + ```cpp +void UModioManagerSubsystem::Shutdown() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->ShutdownAsync(FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnShutdownComplete)); + } +} + +void UModioManagerSubsystem::OnShutdownComplete(FModioErrorCode ErrorCode) +{ + UE_LOG(LogModioGame, Log, TEXT("Shutdown complete with result: %s"), *ErrorCode.GetErrorMessage()); +} +``` + + + +![shutdownasync_runhandlers](img/shutdownasync_runhandlers.png) + + + + + + +:::note +* You *must* continue to call [**`RunPendingHandlers`**](/unreal/refdocs/#run-pending-handlers) while shutdown is in progress to allow intermediate handlers to complete. +* Any in-flight mod.io async calls will complete with an error code indicating cancellation. +* You *must not* call **`ShutdownAsync`** from within another completion handler/callback. This may cause the application to deadlock. +::: diff --git a/Doc/doc_root/en-us/getting-started/metrics-play-sessions.mdx b/Doc/doc_root/en-us/getting-started/metrics-play-sessions.mdx new file mode 100644 index 00000000..4b63ba6e --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/metrics-play-sessions.mdx @@ -0,0 +1,147 @@ +--- +id: ue-metrics-play-sessions +title: Metrics Play Sessions +slug: /unreal/getting-started/metrics-play-sessions +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/metrics-play-sessions.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +The mod.io Unreal Engine Plugin supports all of the mod.io metrics features, allowing you to start a metrics play sesion, keeping that session alive via a heartbeat (automatically called, or manually handled) and then ending that session. Metric sessions allow you to track which mods your players interact with most frequently. Visit https://docs.mod.io/metrics/ for an overview of the mod.io metrics system. + +:::note +Running metrics play sessions is a premium feature. If you are interested in mod.io premium features, please contact developers@mod.io. +::: + +The Metrics session functionality is all accessible through the `ModioSubsystem` class. + +Metrics based on the platform and portal, are transparently taken care of with no additional consideration needed when using the SDK. + +#### Initialization + +The mod.io metrics features are enabled as part of generating a Metrics Secret Key your API settings in your game dashboard, e.g. https://mod.io/g/game-name/admin/api-key. Once this key has been generated, you need to pass it in as an ExtendedInitializationParameters entry in your InitializeOptions when Initializing the mod.io SDK, e.g.: + + +```cpp +FModioInitializeOptions initializeOptions; +initializeOptions.ExtendedInitializationParameters.Add("MetricsSecretKey", "00000000-1111-2222-3333-444444444444"); +``` + +Failing to set up the Metrics Secret Key will result in a `SessionNotInitialized`` error being returned when using the metrics functionality. + +#### Starting a Metrics Session + +You can call [MetricsSessionStartAsync](/unreal/refdocs/#metricssessionstartasync) to start a new session tracking the usage of mods in the context of your game. You'll notice that `MetricsSessionStartAsync` takes a [MetricsSessionParams](/unreal/refdocs/#modiometricssessionparams) struct as its parameter. This contains an optional Session Id, as well as a required vector of mods to track. + +:::note +If a Session Id is not provided, a random one will be created for you. +::: + + + + + ![metrics_session_start_1](img/metrics_session_start_1.png) + +:::note +Use the `SetSessionId` node if you want to associate your own Id with this session. +::: + + ![metrics_session_start_2](img/metrics_session_start_2.png) + + + + +```cpp +FModioMetricsSessionParams Params = FModioMetricsSessionParams(ModIds); + +void UModioManager::MetricsSessionStart(const FModioMetricsSessionParams& Params) +{ + GEngine->GetEngineSubsystem()->MetricsSessionStartAsync( + Params, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionStartCallback)); +} +``` + + + + +The Metrics Session Params accepts an optional Session Id in the form of a `FModioGuid` which you may want to use to associate the new session with any supplementary telemetry you are gathering in your game. + +#### Metrics heartbeat + +To ensure that the session is kept alive, a heartbeat is required to be submitted at most every 5 minutes. We recommend doing this a bit earlier than the threshold to ensure you do not miss the window. + +There are two methods provided to control the behaviour of the heart beat, [MetricsSessionSendHeartbeatOnceAsync](/unreal/refdocs/#metricssessionsendheartbeatonceasync) which you can call at your desired precision, as well as a single call and forget [MetricsSessionSendHeartbeatAtIntervalAsync](/unreal/refdocs/#metricssessionsendheartbeatatintervalasync) with a desired interval. + +Calling `MetricsSessionSendHeartbeatOnceAsync` will submit a single heartbeat, and return an error code if something went wrong. If no error has occured, the heartbeat has been successfully sent. + + + + + + ![metrics_session_heartbeat](img/metrics_session_heartbeat.png) + + + + +```cpp +void UModioManager::MetricsSessionHeartbeat() +{ + GEngine->GetEngineSubsystem()->MetricsSessionSendHeartbeatOnceAsync( + FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionHeartbeatCallback)); +} +``` + + + + +Calling `MetricsSessionSendHeartbeatAtIntervalAsync` requires a parameter with the desired interval frequency in seconds. An error code will be returned if something went wrong, otherwise you will receive a false-y error if the interval loop has been shut down successfully (such as ending a session). + + + + + + ![metrics_session_heartbeat_interval](img/metrics_session_heartbeat_interval.png) + + + + +```cpp +const FModioUnsigned64 IntervalSeconds(150); + +void UModioManager::MetricsSessionHeartbeat(const FModioUnsigned64& IntervalSeconds) +{ + GEngine->GetEngineSubsystem()->MetricsSessionSendHeartbeatAtIntervalAsync( + IntervalSeconds, + FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionHeartbeatCallback)); +} +``` + + + + +#### Ending a Metrics Session + +To complete a session, for example when finishing a match, or quitting out of your game, you can call [MetricsSessionEndAsync](/unreal/refdocs/#metricssessionendasync). +As with the other calls, you will receive an error if anything has gone wrong, otherwise the operation successfully completed. + + + + + + ![metrics_session_end](img/metrics_session_end.png) + + + + + +```cpp +void UModioManager::MetricsSessionHeartbeat() +{ + GEngine->GetEngineSubsystem()->MetricsSessionEndAsync( + FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManager::MetricsSessionHeartbeatCallback)); +} +``` + + + \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/monetization.mdx b/Doc/doc_root/en-us/getting-started/monetization.mdx new file mode 100644 index 00000000..8e8270aa --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/monetization.mdx @@ -0,0 +1,245 @@ +--- +id: ue-monetization +title: Monetization +slug: /unreal/getting-started/monetization +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/monetization.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +The mod.io Unreal Engine Plugin supports a range of monetization features, allowing you to sell a per-game virtual currency to your players that they can use to purchase mods, with a share of the revenue split between creators and your studio. An overview of the mod.io monetization system is available [here](/monetization/). + +Every platform requires specific setup for monetization features to work, particularly with respect to the virtual currency configuration and API calls. The following documentation is generically applicable. Platform-specific information is available in the relevant [platform documentation section](/platforms/). + +## Initialization + +The mod.io monetization features are enabled during the onboarding process on your [game profile](https://mod.io/g). + +Ensure that you have set the appropriate `Portal` when [initializing the plugin](initialization). For instance, on Steam you must initialize with `EModioPortal::Steam` to redeem entitlements for Steam. + +## Getting the user's wallet + +On startup, you can make a call to `UModioSubsystem::GetUserWalletBalanceAsync` to get the balance of the current user's wallet. If no wallet exists for the user, one will be automatically created for them. This call returns the user's wallet balance for the current game. + +The only time you need to make this call is on start-up. We recommend that you cache the value of this result in your game code rather than making consistent calls to `UModioSubsystem::GetUserWalletBalanceAsync`, and update your local state from the return values of other calls that affect wallet balance. + + + + + ![get_user_wallet](img/get_user_wallet.png) + + + + + ```cpp +void UModioManagerSubsystem::GetUserWallet() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->GetUserWalletBalanceAsync(FOnGetUserWalletBalanceDelegate::CreateUObject(this, &UModioManagerSubsystem::OnGetUserWalletComplete)); + } +} + +void UModioManagerSubsystem::OnGetUserWalletComplete(FModioErrorCode ErrorCode, FModioOptionalUInt64 WalletBalance) +{ + if (!ErrorCode) + { + // Wallet balance successfully retrieved + } +} +``` + + + + +## Querying mods + +As part `UModioSubsystem::ListAllModsAsync`, you can include an additional filter for whether you list paid mods. By default, only free mods are shown. Set `RevenueType` on the `ModioFilterParams` object passed to `UModioSubsystem::ListAllModsAsync` to include free and paid content, or just paid content. All mods returned will have a `Price` property, indicating the virtual currency price that must be paid to purchase that mod. + +Filtering for Paid/Unpaid content is not currently exposed to Blueprint. + +## Purchasing mods + +Call `UModioSubsystem::PurchaseModAsync` to purchase a specified mod. `PurchaseModAsync` takes two parameters: the `ModioModID` of the mod to purchase, and the `ExpectedPrice`, which is the price displayed to the user from `UModioSubsystem::ListAllModsAsync`. You must include this parameter for safety so the user is not charged more or less than the price displayed to them, in case the price of the mod has changed between the call to `ListAllModsAsync` and purchase time. + +Once a mod is purchased, it is automatically subscribed to for the user. + +You should validate that the user has enough virtual currency to make the purchase by comparing it to the balance you received from `UModioSubsystem::GetUserWalletBalanceAsync`. This is purely for user experience (e.g. for graying out the purchase button in the UI, or upselling the user a virtual currency pack); `UModioSubsystem::PurchaseModAsync` will return an error if the user has insufficient funds in their wallet. + +An updated wallet balance (with the purchase amount subtracted) is returned in the callback of `UModioSubsystem::PurchaseModAsync`. + + + + + ![purchase_mod](img/purchase_mod.png) + + + + + ```cpp +void UModioManagerSubsystem::PurchaseMod(FModioModID ModId, FModioUnsigned64 ExpectedPrice) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->PurchaseModAsync(ModId, ExpectedPrice, FOnPurchaseModDelegate::CreateUObject(this, &UModioManagerSubsystem::OnPurchaseModComplete)); + } +} + +void UModioManagerSubsystem::OnPurchaseModComplete(FModioErrorCode ErrorCode, FModioOptionalTransactionRecord Transaction) +{ + if (!ErrorCode) + { + // Mod purchase successful + } +} +``` + + + + +## Showing user purchases + +While purchased mods are automatically subscribed and installed at purchase time, the user can freely unsubscribe and uninstall purchased mods and they and they will remain owned and purchased by the user. They must re-subscribe to the mod to have it re-installed. Use `UModioSubsystem::FetchUserPurchasesAsync` to fetch an updated list of a user's purchased mods from the server. After a successful call, you can then display the user's purchased mods with `UModioSubsystem::QueryUserPurchasedMods`, allowing re-subscription as desired. + + + + + ![show_user_purchases](img/show_user_purchases.png) + + + + + ```cpp +void UModioManagerSubsystem::FetchUserPurchases() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->FetchUserPurchasesAsync(FOnFetchUserPurchasesDelegate::CreateUObject(this, &UModioManagerSubsystem::OnFetchUserPurchasesComplete)); + } +} + +void UModioManagerSubsystem::OnFetchUserPurchasesComplete(FModioErrorCode ErrorCode) +{ + if (!ErrorCode) + { + // Purchases Successfully Fetched + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + // We can now access the list of purchased mods directly + TMap PurchasedMods = Subsystem->QueryUserPurchasedMods(); + } + } +} +``` + + + + +## Getting a user delegation token + +User delegation tokens can be used by a backend server for S2S (Server to Server) transactions. You can get one for the current user by calling `UModioSubsystem::GetUserDelegationTokenAsync`, the callback for which contains the token as an `FString`. + + + + + + ![get_user_delegation_token](img/get_user_delegation_token.png) + + + + + ```cpp +void UModioManagerSubsystem::GetUserDelegationToken() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->GetUserDelegationTokenAsync(FOnGetUserDelegationTokenDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnGetUserDelegationTokenCallback)); + } +} + +void UModioManagerSubsystem::OnGetUserDelegationTokenCallback(FModioErrorCode ErrorCode, FString UserDelegationToken) +{ + if (!ErrorCode) + { + // Successfully got User Delegation Token + } +} +``` + + + + + +## Listing and Purchasing Premium Currency + +The Monetization premium features mod.io includes the ability to show the relevant platform store (where possible), which allows the user to purchase virtual currency, top up their wallet, and purchase Premium UGC all without leaving the game. + +This functionality is exposed in one of two ways. + +### Displaying the Native Platform Store + +The recommended method is to simply call `UModioUISubsystem::RequestShowTokenPurchaseUIWithHandler` which will attempt to show the appropriate Platform Native storefront via Unreal's `IOnlineSubsystem`. This function will return an `EModioOpenStoreResult` indicating whether call succeeded or not. + +`FailedInactive` indicates that the Monetization feature is disabled in the base Mod.io plugin settings. + +`FailedUnsupportedPlatform` means that the current platform has not implemented its native storefront in the Unreal Engine Online Subsystem, and so you will have to implement your own method for getting and displaying purchasable items. An example/template of this can be found in the ComponentUI reference design, see below for more information. + +`FailedUnknown` is a catch-all for other error types, such as inactive online subsystems, network issues, etc, and you should check the relevant logs for any pertinent information. + +`Success` means that the platform store UI is being displayed, and you can then listen for the `Callback` to handle any resulting purchases. This `Callback` only indicates whether the Store was opened and closed successfully, not whether the user made a purchase or not. There is also `UModioUISubsystem::RequestShowTokenPurchaseUI` for instances where a Callback is not needed. + + + + + + ![ShowPlatformStoreUI](img/ShowPlatformStoreUI.png) + + + + +### Displaying Component UI Store Offers + +In cases where native store display is not supported, there is a provided template UI component for displaying and listing product offers within the Component UI itself. + +Found at `ModioComponentUI/Content/UI/Templates/Default/WBP_ModioDefaultTokenPackBrowser` this Widget provides a framework and example of querying a set of product offers, displaying them, and purchasing them through the platform store. + +NOTE: With either the Native or Component UI methods, the refreshing and consumption of purchased Entitlements, as well as the updating of the user's Wallet Balance, is left up to you the developer due to platform limitations. + +## Consuming User Entitlements + +User Entitlements are virtual currency packs/products that the user can purchase through their relevant platform store. These Entitlements can then be consumed; in the case of virtual currency, adding balance to their mod.io wallet for use purchasing UGC. + +Once a user has purchased a virtual currency pack on a given store those entitlements must be consumed before they can be used. To do so you can call `UModioSubsystem::RefreshUserEntitlementsAsync` which takes in `FModioEntitlementParams` and a `FOnRefreshUserEntitlementsDelegateFast` as a Callback, returning an `FModioErrorCode` and `TOptional` indicating the status of the user’s entitlements. + +`FModioEntitlementParams` is to be populated with the Delegation Token received from `UModioSubsystem::GetUserDelegationTokenAsync`. + + + + +```cpp +// Get the relevant platform authentication token prior to making this call. +void UModioManager::RefreshUserEntitlements(const FString& PlatformAuthToken) +{ + if (GEngine->GetEngineSubsystem()) + { + FModioEntitlementParams Params; + Params.ExtendedParams.Add("auth_code", PlatformAuthToken); + GEngine->GetEngineSubsystem()->RefreshUserEntitlementsAsync(Params, FOnRefreshUserEntitlementsDelegateFast::CreateUObject(this, &UModioManager::OnRefreshUserEntitlementsCallback)); + } +} + +void UModioManager::OnRefreshUserEntitlementsCallback(FModioErrorCode ErrorCode, TOptional ConsumptionStatusList) +{ + if (ErrorCode == false) + { + // Successfully refreshed entitlements. + } +} +``` + + + + +After entitlements are refreshed successfully any classes properly implementing the `ModioUIWalletBalanceUpdatedEventReceiver` interface will receive the updated balance. + diff --git a/Doc/doc_root/en-us/getting-started/mute-user.mdx b/Doc/doc_root/en-us/getting-started/mute-user.mdx new file mode 100644 index 00000000..78497d3a --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/mute-user.mdx @@ -0,0 +1,124 @@ +--- +id: ue-mute-user +title: Muting and Unmuting a User +slug: /unreal/getting-started/mute-user +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/mute-user.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +Users have the ability to disable updates from other user’s mods. This will prevent mod.io from returning mods authored by the muted user. There are three actions available to take: mute a user, unmute a user, and list muted users + +:::note +To perform any of these actions, the muting user must be authenticated. +::: + +## Mute a user + +To mute a user, call [`MuteUserAsync`](/unreal/refdocs/#muteuserasync) with the corresponding `ModioUserID` and a callback. + + + + + +![mute_user](img/mute_user.png) + + + + + + ```cpp + +void UModioManagerSubsystem::MuteAUser(FModioUserID UserID) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->MuteUserAsync(UserID, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnMuteUserComplete)); + } +} + +void UModioManagerSubsystem::OnMuteUserComplete(FModioErrorCode ErrorCode) +{ + if (!ErrorCode) + { + // User successfully muted + } +} + +``` + + + + +## Unmute a user + +To perform the inverse operation call [`UnmuteUserAsync`](/unreal/refdocs/#unmuteuserasync) with the corresponding `ModioUserID` and a callback. + + + + + +![unmute_user](img/unmute_user.png) + + + + + + ```cpp + +void UModioManagerSubsystem::UnmuteAUser(FModioUserID UserID) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->UnmuteUserAsync(UserID, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnUnmuteUserComplete)); + } +} + +void UModioManagerSubsystem::OnUnmuteUserComplete(FModioErrorCode ErrorCode) +{ + if (!ErrorCode) + { + // User successfully unmuted + } +} +``` + + + + +## List muted users + +[`GetMutedUsersAsync`](/unreal/refdocs/#getmutedusersasync) returns a list of users previously muted by an authenticated user. + + + + +![get_muted_users](img/get_muted_users.png) + + + + + + ```cpp + +void UModioManagerSubsystem::ListMutedUsers() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->GetMutedUsersAsync(FOnMuteUsersDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnListMutedUsersComplete)); + } +} + +void UModioManagerSubsystem::OnListMutedUsersComplete(FModioErrorCode ErrorCode, FModioOptionalUserList MutedUsers) +{ + if (!ErrorCode) + { + // List of muted users successfully retrieved + } +} + +``` + + + \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/plugin-structure.mdx b/Doc/doc_root/en-us/getting-started/plugin-structure.mdx new file mode 100644 index 00000000..19bdd258 --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/plugin-structure.mdx @@ -0,0 +1,136 @@ +--- +id: ue-plugin-structure +title: Plugin Structure & Concepts +slug: /unreal/getting-started/plugin-structure/ +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/plugin-structure.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +## `ModioSubsystem` +The mod.io Unreal Engine Plugin consumes our [Native SDK](/cppsdk/) internally, and its public API is similar. The Native SDK exposes a number of free functions within the `Modio` namespace, and this plugin wraps those functions in a [Subsystem](https://dev.epicgames.com/documentation/en-us/unreal-engine/programming-subsystems-in-unreal-engine). + +The lifetime of the [`ModioSubsystem`](/unreal/refdocs/#modiosubsystem) is automatically managed by the engine and provides a convenient way to access the plugin's functionality. This is broadly grouped into the following categories: + +* Plugin management (initialization, teardown, event loop) +* User authentication +* Mod browsing and querying +* Mod management (subscription, unsubscription, and installation) + +## Value objects +All data returned by the plugin is presented using a small set of `USTRUCTs`, containing information such as: + +* Details for mods that are available for installation +* Progress information about mods being installed +* Details and load paths for installed mods + +As structs, these are value types, so if you want to hold onto them once you’ve shut down the plugin you can do so. They do not expose any methods that "talk back" to the `ModioSubsystem`, so their methods are safe to call. This allows you, for instance, to initialize the plugin, query the installed mods, and retain that list even after shutting down the plugin and its event loop. + +## UTF-8 guarantees + +The plugin uses UTF8 for all strings, as does the mod.io REST API. + +## Thread-safety guarantees + +The mod.io plugin is thread-safe with the exception of [`RunPendingHandlers`](/unreal/refdocs/#run-pending-handlers). If you are calling `RunPendingHandlers` manually, you must always call it on the same thread. + +:::note +The plugin event loop, any internal event handlers, and all callbacks you provide to the mod.io plugin will be run on the thread invoking `RunPendingHandlers`. +::: + +## Non-blocking, asynchronous interface + +The plugin communicates with the mod.io servers, the filesystem on the device you’re using, and platform-provided services for authentication. These may not return results immediately, so many functions provided by the `ModioSubsystem` are non-blocking and asynchronous. + +:::note +All async methods in the public API end with the suffix `Async`. +::: + +### Callback conventions + +All of these asynchronous methods take a delegate as an argument, which will be invoked **exactly once** with the results of the requested operation. Every async callback takes a [`ModioErrorCode`](/unreal/refdocs/#modioerrorcode) as its first parameter, with any optional results wrapped in `TOptional` so that you can easily check if a result is valid. + +Async functions should not be considered complete until their callback is invoked. For example, although `InitializeAsync` returns immediately, you must wait for its callback (with a successful `ModioErrorCode`) before you are able to call any other functions from the `ModioSubsystem`. + +Any return values provided to your callback are passed-by-value; the plugin doesn’t expect you to have to call `release` or otherwise free up resources given to you. + +:::note +Even if the plugin is shut down while asynchronous operations are in-flight, your delegates will still be invoked **exactly once**. In this instance, the `ModioErrorCode` the delegate receives will indicate a cancellation state, and you should check for this as part of your error handling in your delegates. +::: + +### Maintaining the plugin event loop + +In order to provide a non-blocking implementation, the plugin operates an internal event loop. This event loop will only ever run on the thread which calls [`RunPendingHandlers`](/unreal/refdocs/#run-pending-handlers). `RunPendingHandlers` is the function we provide to tick the internal event loop and process any pending work the plugin needs to perform. + +You should either enable the [configuration setting](/unreal/installation-and-setup/#plugin-configuration) to allow the plugin to manage a background thread which automatically calls `RunPendingHandlers`, or invoke it on tick in the game thread yourself. + +:::note +This means that if you stop calling `RunPendingHandlers`, any pending asynchronous API methods you have called will not complete and their associated callbacks will not be invoked, nor will the internal data allocated for those operations be freed. +::: + +## Error handling + +Many of the plugin's functions utilize the [`ModioErrorCode`](/unreal/refdocs/#modioerrorcode) type. For example, all [async](#non-blocking-asynchronous-interface) functions take a `ModioErrorCode` as the first parameter for the function's delegate. This is essentially an opaque wrapper around a numeric error code with a category and an associated string message. + +The plugin doesn’t attempt to predict what your error-handling logic or requirements are. Instead, we return the error code to you so you can decide what to do. For instance, if you call a function and receive an error code matching a network error condition, do you want to close down the plugin? Retry again according to custom back-off logic? That decision is left to the consuming application. + +For more details on the error codes and how to inspect their values, please see our [Error Handling quick-start guide](/unreal/getting-started/error-handling). + +## User sessions + +The mod.io plugin runs on a per-platform-user basis. If you are using the plugin on a platform that requires user switching support, you must call [`ShutdownAsync`](/unreal/refdocs/#shutdownasync) and re-initialize the plugin by calling [`InitializeAsync`](/unreal/refdocs/#initializeasync) with a different [Session ID](#session-ids). + +Generating a stable platform-specific Session ID for each user will ensure that an incoming user who has already authenticated with mod.io on the current device and game won’t need to re-authenticate unless their authentication token has expired. + +### Session IDs +Session IDs are chosen by the developer. They can be whatever you like as long as they are deterministic and stable. Session IDs create a scope or a local profile for the current user to live in so that a single system can support multiple authenticated users side-by-side without requiring de-authentication of the previous user. Internally, the Session ID is used to create a folder containing the authentication information and cached profile of the authenticated user. + +On console platforms, we suggest that the Session ID should be a string representation of the platform-provided user ID. This gives the best experience when it comes to things like user switching. For example, a game on Xbox using a sanitized string representation of the Xbox Live ID as the Session ID would have a folder structure in persistent storage like the following: + +``` +/mod.io/// +/mod.io/// +``` + +With this example, you can detect the user associated with the current controller on game start and initialize the mod.io plugin with the Session ID set to the stable string representation of that user's Xbox Live ID. As a result, the user's authentication status will be maintained from the previous session on that device. + +In the case of a Windows title with user-provided profile names, the same folder structure would be more like the following: + +``` +%USERDATA%/mod.io//ProfileName1/ +%USERDATA%/mod.io//ProfileName2/ +%USERDATA%/mod.io//ProfileName3/ +``` + +This allows multiple players, for example siblings, to each have their own session living in the same Windows account. + +## Mod data directory + +The plugin stores mods in a game-specific directory in the following directory by default: + +| Windows | Linux | macOS | iOS | +| --- | --- | --- | --- | +| `${FolderID_Public}/mod.io` | `${USER_HOME}/mod.io` | `${USER_HOME}/Library/Application Support/mod.io` | `${APP-DIRECTORY}/Documents/mod.io` | + +However, this value can be overridden in one of two ways: + +### Globally for a system account + + On first run of a game using the plugin, `${FOLDERID_LocalAppData}/mod.io/globalsettings.json` will be created. + + This JSON object contains a `RootLocalStoragePath` element - changing the string here to a valid path on disk will globally redirect the mod installation directory for **ALL** games using the mod.io Unreal Engine Plugin or Native SDK for the current system account. + + :::warning + Changing this value while the SDK is initialized is not supported and behaviour is undefined. + ::: + +### Per-game for a system account + + Per-game settings are stored in `${FOLDERID_LocalAppData}/mod.io/${GameId}/${mod.io user string}/user.json`. + + Adding a `RootLocalStoragePath` element to this file will redirect the mod installation directory for this specific game only for the current system account. Removing this value will cause the game to revert back to the global value in `globalsettings.json`. + +:::note +In Linux and macOS, mods and cached data binds to a single user. Other client have their own instance in their home directory. +::: \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/submit-mods.mdx b/Doc/doc_root/en-us/getting-started/submit-mods.mdx new file mode 100644 index 00000000..7275eb0e --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/submit-mods.mdx @@ -0,0 +1,89 @@ +--- +id: ue-submit-mods +title: In-Game Mod Submission +slug: /unreal/getting-started/submit-mods +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/submit-mods.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +Submitting a mod from inside your game and making it visible to other players involves two steps: + +* Submission of the mod +* Submission of the mod’s data (aka "the mod file") + +These steps are outlined below. Mods can also be edited after submission, as [detailed here](edit-mods) + +## Submitting a new mod + +To submit a mod, you must first create a mod handle using [`GetModCreationHandle`](/unreal/refdocs/#getmodcreationhandle), and use that handle when calling [`SubmitNewModAsync`](/unreal/refdocs/#submitnewmodasync). Note that the newly created mod will remain hidden until a mod file is added in the next step. + + + +![submit_new_mod](img/submit_new_mod.png) + + + + ```cpp +void UModioManagerSubsystem::SubmitNewMod() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + FModioModCreationHandle Handle = Subsystem->GetModCreationHandle(); + + FModioCreateModParams Params; + Params.Name = TEXT("My Amazing Mod"); + Params.Description = TEXT("This mod does amazing things"); + Params.PathToLogoFile = TEXT("C:\\path\\to\\image.png"); + + Subsystem->SubmitNewModAsync(Handle, Params, FOnSubmitNewModDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnSubmitNewModComplete)); + } +} + +void UModioManagerSubsystem::OnSubmitNewModComplete(FModioErrorCode ErrorCode, TOptional ModId) +{ + if (!ErrorCode) + { + // Mod was successfully submitted + // We can now call SubmitNewModFileForMod with this ModId + } +} +``` + + + +## Submitting a file for a mod + +Once you have successfully submitted a mod, you can submit a mod file for that mod using [`SubmitNewModFileForMod`](/unreal/refdocs/#submitnewmodfileformod). When you submit a mod file, you pass a [`ModioCreateModFileParams`](/unreal/refdocs/#modiocreatemodfileparams) containing the directory containing all the files that you want to submit. The plugin will compress this folder into a .zip file and upload it as the active version of the mod. + +In the future, if the mod is updated and requires a new mod file, `SubmitNewModFileForMod` can be called again. The most recent mod file uploaded by `SubmitNewModFileForMod` will be set as the active version. + +:::note +There is no callback for `SubmitNewModFileForMod`; you’ll be notified of the completed upload by your [**mod management callback**](mod-subscriptions#installation-management). +::: + + + + +After the callback for submitting a mod has completed, you can get the Mod Id to use for file submission. + +![submit_new_mod_file](img/submit_new_mod_file.png) + + + + + ```cpp +void UModioManagerSubsystem::SubmitNewModFile(FModioModID ModId) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + FModioCreateModFileParams Params; + Params.PathToModRootDirectory = TEXT("C:\\path\\to\\mod-folder"); + Subsystem->SubmitNewModFileForMod(ModId, Params); + } +} +``` + + + \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/subscriptions.mdx b/Doc/doc_root/en-us/getting-started/subscriptions.mdx new file mode 100644 index 00000000..9d28b755 --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/subscriptions.mdx @@ -0,0 +1,253 @@ +--- +id: ue-mod-subscriptions +title: Mod Subscriptions & Management +slug: /unreal/getting-started/mod-subscriptions +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/mod-subscriptions.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +So you’ve [shown the user some mods](browsing-mods), and they've picked one they’d like to install. How do you begin the installation process? Once the mod is installed, how do you know what files to load into memory? + +## Mod subscriptions + +Users indicate they want to install a mod by **subscribing** to it. Subscriptions are stored on the mod.io servers and are associated with a user’s mod.io account, so subscribed mods for a given game will be installed to all devices where a user logs into mod.io for that game. Similarly, when a user **unsubscribes** from a mod, that mod will be uninstalled from every device they’re logged into mod.io with for that game. + +Subscriptions are managed with calls to either [`SubscribeToModAsync`](/unreal/refdocs/#subscribetomodasync) or [`UnsubscribeFromModAsync`](/unreal/refdocs/#unsubscribefrommodasync) containing the desired `ModioModID` and a delegate to receive the status of the request. + +:::note +To subscribe or unsubscribe from a mod, [**`EnableModManagement`**](/unreal/refdocs/#enablemodmanagement) *must* be called beforehand. +::: + +`SubscribeToModAsync` also takes a bool indicating whether or not to to subscribe to any and all dependencies for the given `ModioModID`. + +:::note +When dependencies are included in the `SubscribeToModAsync` call, they will *not* download automatically. **Only the primary mod specified by `ModioModID` will download automatically.** To download all subscribed content including dependencies, call [`FetchExternalUpdatesAsync`](/unreal/refdocs/#fetchexternalupdatesasync) after `SubscribeToModAsync` successfully completes. +::: + + + + +![subscribe_to_mod](img/subscribe_to_mod.png) + +![unsubscribe_from_mod](img/unsubscribe_from_mod.png) + + + + + ```cpp +void UModioManagerSubsystem::SubscribeToMod(FModioModID ModToSubscribeTo, bool IncludeDependencies) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->SubscribeToModAsync(ModId, IncludeDependencies, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnSubscribeToModComplete, ModId)); + } +} + +void UModioManagerSubsystem::OnSubscribeToModComplete(FModioErrorCode ErrorCode, FModioModID ModId) +{ + if (!ErrorCode) + { + // Indicate success to your user. This ModId's files will begin installing. + // Call FetchExternalUpdatesAsync if dependencies were included + } +} + +void UModioManagerSubsystem::UnsubscribeFromMod(FModioModID ModId) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->UnsubscribeFromModAsync(ModId, FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnUnsubscribeFromModComplete, ModId)); + } +} + +void UModioManagerSubsystem::OnUnsubscribeFromModComplete(FModioErrorCode ErrorCode, FModioModId ModId) +{ + if (!ErrorCode) + { + // Indicate success to your user. This ModID's files will begin uninstalling. + } +} +``` + + + + +## External subscription changes + +Because mod.io's services are available via our website, users can manage their subscriptions outside of your application. This means that we need to be able to query the server for any external subscription changes. + +Call [`FetchExternalUpdatesAsync`](/unreal/refdocs/#fetchexternalupdatesasync) to synchronise the server state with the plugin’s local subscriptions. Any required installations or uninstallations based on the updated user subscriptions will be processed automatically. + + + + +![fetch_external_updates](img/fetch_external_updates.png) + + + + + + ```cpp + +void UModioManagerSubsystem::FetchExternalUpdates() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->FetchExternalUpdatesAsync(FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnFetchExternalUpdatesComplete)); + } +} + +void UModioManagerSubsystem::OnFetchExternalUpdatesComplete(FModioErrorCode ErrorCode) +{ + // error handling etc. +} + +``` + + + +:::warning +Call [`FetchExternalUpdatesAsync`](/unreal/refdocs/#fetchexternalupdatesasync) sparingly, only when required to ensure that the local state is up-to-date such as on game start-up or based on user input (i.e. as a button on your UI). It should **NOT** be called on tick. +::: + +## Checking the user subscription list + +To see which mods the user has subscribed to, call [`QueryUserSubscriptions`](/unreal/refdocs/#query-user-subscriptions). This retrieves a `TMap` of [`ModioModCollectionEntry`](/unreal/refdocs/#modiomodcollectionentry) objects, one for each subscribed mod. Each `ModioModCollectionEntry` object contains the mod’s state, profile information, ID and more. + +:::note +This collection includes mods that are still in the process of being installed! Make sure to check the result of [`GetModState`](/unreal/refdocs/#get-mod-state) before attempting to load files from the mods in this collection. Alternatively, use [`QueryUserInstallations`](/unreal/refdocs/#query-user-installations) as described in [**Retrieving mod directory paths for loading**](#retrieving-mod-directory-paths-for-loading). +::: + + + + +![query_user_subscriptions](img/query_user_subscriptions.png) + + + + + + ```cpp +void UModioManagerSubsystem::QueryUserSubscriptions() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + TMap SubscribedMods = Subsystem->QueryUserSubscriptions(); + // Do something with SubscribedMods e.g. display on a UI + } +} +``` + + + +## Installation management + +So a subscription marks a mod as requiring installation, and an unsubscription indicates uninstallation, but how do you actually control when the plugin **does** those things? After all, you don’t want a mod to be uninstalled after your main program has loaded those files into memory, locking them from deletion. Likewise, you probably don’t want to be using networking or processor resources during gameplay for downloading mods. To give you control over when these processes occur without forcing you to shut down the plugin, you can call [`EnableModManagement`](/unreal/refdocs/#enablemodmanagement) and [`DisableModManagement`](/unreal/refdocs/#disable-mod-management). + +To notify your users when a mod is finished installing or updating, [`EnableModManagement`](/unreal/refdocs/#enablemodmanagement) asks you to provide it with a callback. This callback will be invoked **every time** a [`ModioModManagementEvent`](/unreal/refdocs/#modiomodmanagementevent) occurs i.e. whenever a mod is installed, updated, uninstalled, or uploaded by the plugin’s internal event loop. This behavior persists until a corresponding call to [`DisableModManagement`](/unreal/refdocs/#disable-mod-management) or [`ShutdownAsync`](/unreal/refdocs/#shutdownasync) is made. + +:::note +[`EnableModManagement`](/unreal/refdocs/#enablemodmanagement) is not an async function. It does not end with the `*Async` suffix. Its callback operates differently to asynchronous result callbacks used elsewhere in the mod.io plugin. +::: + + + +![enable_mod_management](img/enable_mod_management.png) + + + + ```cpp +void UModioManagerSubsystem::EnableModManagement() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->EnableModManagement(FOnModManagementDelegateFast::CreateUObject(this, &UModioManagerSubsystem::ModManagementCallback)); + } +} + +void UModioManagerSubsystem::ModManagementCallback(FModioModManagementEvent ModManagementEvent) +{ + if (ModManagementEvent.Status) + { + // error handling + } + switch(ModManagementEvent.Event) + { + case EModioModManagementEventType::BeginInstall: + case EModioModManagementEventType::BeginUninstall: + case EModioModManagementEventType::BeginUpdate: + case EModioModManagementEventType::BeginUpload: + UE_LOG(LogModioGame, Log, TEXT("Begin processing Mod %s"), *ModManagementEvent.ID.ToString()); + break; + case EModioModManagementEventType::Installed: + UE_LOG(LogModioGame, Log, TEXT("Received an Installed event for Mod %s"), *ModManagementEvent.ID.ToString()); + break; + case EModioModManagementEventType::Uninstalled: + UE_LOG(LogModioGame, Log, TEXT("Received an Uninstalled event for Mod %s"), *ModManagementEvent.ID.ToString()); + break; + case EModioModManagementEventType::Updated: + UE_LOG(LogModioGame, Log, TEXT("Received an Updated event for Mod %s"), *ModManagementEvent.ID.ToString()); + break; + case EModioModManagementEventType::Uploaded: + UE_LOG(LogModioGame, Log, TEXT("Received an Uploaded event for Mod %s"), *ModManagementEvent.ID.ToString()); + break; + default:; + } +} +``` + + + +While mod management is enabled, the plugin assumes that it has the ability to make changes to the filesystem, including deleting mods that the user has no longer subscribed to. As a result you should make sure that you don’t have any open handles to files inside the mod directories when you call [`EnableModManagement`](/unreal/refdocs/#enablemodmanagement). + +When you want to be able to freely open files in the mod directories, call [`DisableModManagement`](/unreal/refdocs/#disable-mod-management). The plugin will finish the current operation but will not continue any others. + +Call [`IsModManagementBusy`](/unreal/refdocs/#is-mod-management-busy) to see if mod management is currently processing a mod. + + + + + ```cpp +void UModioManagerSubsystem::DisableModManagement() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + if (!Subsystem->IsModManagementBusy()) + { + Subsystem->DisableModManagement(); + } + } +} +``` + + + +### Retrieving mod directory paths for loading + +So now we have the user picking mods and marking them for installation, we’re enabling mod management at a point where we don’t mind the plugin changing the filesystem, and mods are being installed by the plugin. We now need to know where they are on disk, otherwise you can’t load them into your game! + +The easiest way to do this is by using [`QueryUserInstallations`](/unreal/refdocs/#query-user-installations). This function returns a `TMap` of [`ModioModCollectionEntry`](/unreal/refdocs/#modiomodcollectionentry) objects that can be queried for folder paths to use for loading a mod's files into your game. `QueryUserInstallations` also allows you to specify whether or not to include outdated mods. + + + + ```cpp +void UModioManagerSubsystem::GetInstalledMods() +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + TMap InstalledMods = Subsystem->QueryUserInstallations(false); + + // Do something with each installed mod, ie adding paths/loading the content appropriately + } +} +``` + + + +![query_user_installations](img/query_user_installations.png) + + + + \ No newline at end of file diff --git a/Doc/doc_root/en-us/getting-started/temporary-mods.mdx b/Doc/doc_root/en-us/getting-started/temporary-mods.mdx new file mode 100644 index 00000000..c8b24768 --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/temporary-mods.mdx @@ -0,0 +1,58 @@ +--- +id: ue-temporary-mods +title: Temporary Mod Sets +slug: /unreal/getting-started/temporary-mods +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/temporary-mods.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +In some situations, you may want mods to only exist on a temporary basis. For instance, in multiplayer environments where you don't want to subscribe a user to a piece of content. Temporary mod sets allow management of these transient pieces of content separately from subscriptions. Temp mod sets do not require authentication, however mod management must still be enabled. + +Temp mods are downloaded to a separate folder from subscriptions, and are not updated or handled when you call `FetchExternalUpdatesAsync`. That means you can prioritize download and installation of temp mods outside of the regular subscription flow. + +To use temp mods, you can start a `TempModSet` by calling `InitTempModSet` and passing a list of `ModioModIDs` to be downloaded and extracted. At any time while a `TempModSet` is open, you can call [`AddToTempModSet`](/unreal/refdocs/#addtotempmodset) to add mods to the set. These will be instantly downloaded and extracted. If you no longer need a mod, you can call [`RemoveFromTempModSet`](/unreal/refdocs/#removefromtempmodset) which will remove the file. Once you have finished with a `TempModSet`, call [`CloseTempModSet`](/unreal/refdocs/#closetempmodset) which will delete all temporary mods. Temporary mods are also deleted the next time you re-initialize the mod.io plugin. + +Like regular mods, temp mods can be queried using [`QueryTempModSet`](/unreal/refdocs/#querytempmodset) to get a `ModioModCollectionEntry` with an installation path. + +### Installing Temporary Mods + + + + +![temp_mod_init](img/temp_mod_init.png) + + + + ```cpp +Modio::EnableModManagement([](Modio::ModManagementEvent ModEvent) +{ + if (ModEvent.Status && ModEvent.Event == Modio::ModManagementEvent::EventType::Installed) + { + std::cout << "Mod with ID: " << ModEvent.ID << " is installed" << std::endl; + } + else + { + std::cout << "Mod with ID: " << ModEvent.ID << " failed to install: " << ModEvent.Status.message() << std::endl; + } +}); + +std::vector ModIds = {8, 4, 5}; + +Modio::InitTempModSet(ModIds); + +while(Modio::IsModManagementBusy()) +{ + Modio::RunPendingHandlers(); +} +``` + + + + +This call will start a `TempModSet` and install mods with IDs 8, 4 and 5. + +:::note +If you add a mod to `TempModSet` that is already subscribed, it will not be downloaded; the player will already have that content. If you try to unsubscribe from it while it's in `TempModSet`, the plugin will wait for it to be removed from `TempModSet` before processing the unsubscribe. +::: diff --git a/Doc/doc_root/en-us/getting-started/user-authentication.mdx b/Doc/doc_root/en-us/getting-started/user-authentication.mdx new file mode 100644 index 00000000..e0cb600e --- /dev/null +++ b/Doc/doc_root/en-us/getting-started/user-authentication.mdx @@ -0,0 +1,230 @@ +--- +id: ue-user-authentication +title: User Authentication +slug: /unreal/getting-started/user-authentication +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/getting-started/user-authentication.mdx +--- + +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +## Best practices +We recommend creating a new game-instance based subsystem (e.g. `ModioManagerSubsystem`) that depends on `ModioSubsystem`, and is responsible for managing the full initialization and authentication flow for your game. + +### Recommended flow +- Create a subsystem to manage mod.io initialization and authentication. +- Perform [Online Subsystem (OSS) login](#using-the-unreal-online-subsystem) in your subsystem's `Initialize` function to ensure you have a valid platform auth token ready when its time to authenticate to mod.io using [single sign-on (SSO)](#single-sign-on-authentication). + - Log in with the online identity provider. + - On success, get a linked account token for the current user. + - On success, get an authentication token from the linked account or from the online identity. +- [Initialize](initialization) the mod.io plugin with the required environment parameters for the current platform. +- Display the mod.io [Terms of Service](/terms/). Ensure they are accepted by the user. +- Authenticate the user using [SSO](#single-sign-on-authentication). + +:::note +If the Online Subsystem login fails at any point in the above process, fall back to simply initializing the plugin and offer email authentication as an alternative. +::: + +## Terms & user consent + +Users must accept the mod.io Terms of Service before using any mod.io services. The Terms of Service can be fetched from mod.io using [`GetTermsOfUseAsync`](/unreal/refdocs/#gettermsofuseasync). Before attempting authentication, ensure these terms are displayed to and accepted by the user. + +See our full [Terms & User Consent](/terms/) documentation for further details. + +## Authentication methods + +The mod.io Unreal Engine plugin provides two ways for users to create an account to use the service: email authentication, and single sign-on (SSO) through an external authentication partner. + +:::tip +For the best user experience, we recommend using SSO wherever possible. +::: + +### Email authentication + +Users can authenticate to mod.io with just an email address. This involves: + - Requesting a one-time authentication code be sent from mod.io to an email address provided by the user. + - Submitting the received code to mod.io. + +#### Requesting an authentication code + +Request a one-time authentication code by calling [`RequestEmailAuthCodeAsync`](/unreal/refdocs/#requestemailauthcodeasync). Pass in a user-provided email address as the first argument. + + + + ```cpp +void UModioManagerSubsystem::RequestEmailAuthCode(const FString &EmailAddress) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->RequestEmailAuthCodeAsync(FModioEmailAddress(*EmailAddress), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnRequestEmailAuthCodeComplete)); + } +} + +void UModioManagerSubsystem::OnRequestEmailAuthCodeComplete(FModioErrorCode ErrorCode) +{ + UE_LOG(LogModioGame, Log, TEXT("Email auth code request complete with result: %s"), *ErrorCode.GetErrorMessage()); +} +``` + + +![request_email_auth_code](img/request_email_auth_code.png) + + + +When your callback is invoked, inspect the `ModioErrorCode` for a successful result. On success, prompt the user to check their emails and enter the authentication code they have received into your game. + +#### Submitting an authentication code + +To finalize the user's authentication, submit the code they have entered to mod.io by calling [`AuthenticateUserEmailAsync`](/unreal/refdocs/#authenticateuseremailasync). Pass the user's code in as the first argument. + + + + + ```cpp +void UModioManagerSubsystem::AuthenticateUserEmail(const FString& AuthCode) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->AuthenticateUserEmailAsync(FModioEmailAuthCode(*AuthCode), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::OnAuthenticateUserEmailComplete)); + } +} + +void UModioManagerSubsystem::OnAuthenticateUserEmailComplete(FModioErrorCode ErrorCode) +{ + UE_LOG(LogModioGame, Log, TEXT("Email auth code submission complete with result: %s"), *ErrorCode.GetErrorMessage()); +} +``` + + +![authenticate_user_email](img/authenticate_user_email.png) + + + +When your callback is invoked, inspect the `ModioErrorCode` for a successful result. On success, user authentication is complete. + +### Single sign-on authentication + +The mod.io Unreal Engine plugin features single sign-on (SSO) authentication from a number of [external providers](/unreal/refdocs/#EModioAuthenticationProvider). These include: + +- [Xbox Live](/platforms/gdk/#authentication) +- [Steam](/platforms/steam/authentication) +- [GOG Galaxy](/platforms/gog/authentication) +- itch.io +- [Nintendo Switch](/platforms/switch/#authentication) +- Discord +- [PlayStation™Network](/platforms/playstation/#authentication) +- [Epic Online Services](/platforms/epic/authentication) +- Oculus +- OpenID +- [Google](/platforms/google/authentication) + +:::note +OpenID is a premium feature. If you are interested in mod.io premium features, please contact developers@mod.io. +::: + +Each platform has their own requirements and prerequisites for performing SSO. Platform-specific authentication can be found in the respective [platform documentation](/platforms/). SSO is performed by calling [`AuthenticateUserExternalAsync`](/unreal/refdocs/#authenticateuserexternalasync). This function takes a [`ModioAuthenticationParams`](/unreal/refdocs/#modioauthenticationparams) containing provider-specific parameters, and a [`ModioAuthenticationProvider`](/unreal/refdocs/#EModioAuthenticationProvider) indicating the provider you're using. + +[Unreal Engine's Online Subsystem](#using-the-unreal-online-subsystem) provides a convenient way to get a token for `ModioAuthenticationParams.AuthToken`. [`GetDefaultAuthProviderForCurrentPlatform`](/unreal/refdocs/#get-default-auth-provider-for-current-platform) can be used to get the default provider for the current platform. + +:::note +The plugin will automatically URL encode parameters (such as the auth token) when making the request. +::: + + + + + ```cpp +void UModioManagerSubsystem::AuthenticateWithSSO(const FModioAuthenticationParams& AuthParams) +{ + if (UModioSubsystem* Subsystem = GEngine->GetEngineSubsystem()) + { + Subsystem->AuthenticateUserExternalAsync(AuthParams, UModioPlatformHelpersLibrary::GetDefaultAuthProviderForCurrentPlatform(), FOnErrorOnlyDelegateFast::CreateUObject(this, &UModioManagerSubsystem::AuthenticateUserExternalComplete)); + } +} + +void UModioManagerSubsystem::AuthenticateUserExternalComplete(FModioErrorCode ErrorCode) +{ + UE_LOG(LogModioGame, Log, TEXT("SSO complete with result: %s"), *ErrorCode.GetErrorMessage()); +} +``` + + +![authenticate_user_email](img/authenticate_user_external.png) + + + +## Using the Unreal Online Subsystem + +Since Unreal Engine 5.2, most of the authentication methods supported by mod.io are abstracted by Unreal's [Online Subsystem](https://dev.epicgames.com/documentation/en-us/unreal-engine/API/Plugins/OnlineSubsystem/IOnlineSubsystem). Below are some code snippets demonstrating the flow to get a linked account token from Unreal's Online Subsystem. + +- Create a subsystem to manage mod.io initialization and authentication that depends on the `ModioSubsystem`. During your subsystem's initialization, perform an Online Subsystem (OSS) login. This guarantees you will have a valid platform auth token ready when needed for mod.io SSO authentication. + + + + ```cpp +const IOnlineSubsystem* OnlineSubsystem = IOnlineSubsystem::GetByPlatform(); +if (OnlineSubsystem != nullptr && OnlineSubsystem->GetIdentityInterface() != nullptr) +{ + // Get the Identity interface from our OnlineSubsystem and perform login + const IOnlineIdentityPtr OnlineIdentity = OnlineSubsystem->GetIdentityInterface(); + OnlineIdentity->AddOnLoginCompleteDelegate_Handle( + 0, FOnLoginCompleteDelegate::CreateLambda( + [this](int32 LocalUserNum, bool bWasSuccessful, const FUniqueNetId& UserId, const FString& Error) { + this->OnSubsystemLoginComplete(LocalUserNum, bWasSuccessful, UserId, Error); + })); + OnlineIdentity->Login(0, FOnlineAccountCredentials()); +} +else +{ + // If we don't have a valid OnlineSubsystem, just initialize the mod.io plugin + InitializeModio(); +} +``` + + + +- After a successful OSS login, request the linked account token. + + + + ```cpp +void UModioManagerSubsystem::OnSubsystemLoginComplete(int32 LocalUserNum, bool bWasSuccessful, const FUniqueNetId& UserId, const FString& Error) +{ + if (bWasSuccessful) + { + auto OnlineIdentity = IOnlineSubsystem::GetByPlatform()->GetIdentityInterface(); + OnlineIdentity->GetLinkedAccountAuthToken(0, FString(), IOnlineIdentity::FOnGetLinkedAccountAuthTokenCompleteDelegate::CreateUObject(this, &UModioManagerSubsystem::OnGetLinkedAccountTokenComplete)); + } +} +``` + + + +- On success, the callback received from `GetLinkedAccountAuthToken` will contain a token that you can use to perform SSO. This will contain either a byte array `TokenData` (e.g. for Steam SSO) or `TokenString` for a plaintext string. These need to be handled appropriately. + + + + ```cpp +void UModioManagerSubsystem::OnGetLinkedAccountTokenComplete(int LocalUserNum, bool bWasSuccessful, const FExternalAuthToken& AuthToken) +{ + if (AuthToken.IsValid()) + { + if (AuthToken.HasTokenData()) + { + AuthTokenString = FBase64::Encode(AuthToken.TokenData); + } + else if (AuthToken.HasTokenString()) + { + AuthTokenString = AuthToken.TokenString; + } + } +} +``` + + + +## Token Lifetime & Re-Authentication + +By default, tokens issued via email token exchange have a lifetime of 1 year. You can verify that a user has been successfully authenticated with [`VerifyUserAuthenticationAsync`](/unreal/refdocs/#verifyuserauthenticationasync). + +If a user is not authenticated for any reason (e.g. their token has been invalidated by changing their password, or their profile data has been removed from their device) then you should prompt them to re-authenticate. diff --git a/Doc/doc_root/en-us/index.mdx b/Doc/doc_root/en-us/index.mdx new file mode 100644 index 00000000..27c5db62 --- /dev/null +++ b/Doc/doc_root/en-us/index.mdx @@ -0,0 +1,43 @@ +--- +id: ue-index +title: Overview +slug: /unreal/ +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/index.mdx +--- + +The mod.io Unreal Engine plugin enables game developers to easily integrate mods into their Unreal Engine 5 games. + +The plugin provides a C++ and Blueprint interface around the [mod.io SDK](/cppsdk/) to connect to the [mod.io REST API](https://docs.mod.io/restapiref/). Also included is a set of UI components to easily create a custom user interface that matches your game's style, plus a comprehensive template UI to get you started. + +You can download the [mod.io Unreal Engine plugin from GitHub](https://github.com/modio/modio-ue). Once you have, follow the [installation guide](installation-and-setup/) for installing and configuring the plugin. A series of [quick-start guides](getting-started/) are also available to familiarize yourself with the plugin's core features. + +## Engine & platform compatability + +The mod.io plugin is generally maintained to support the 3 most recent versions of Unreal Engine. If you are using an older version of the engine, you can access the last supported release of the engine from our [releases page](https://github.com/modio/modio-ue/releases) on GitHub. + +### Engine compatability + +|Engine Version | Last Release | +|---|---| +|UE4.26 | 2023.11 | +|UE4.27 | 2023.11 | +|UE5.0 | 2023.11 | +|UE5.1 | 2024.6 | +|UE5.2 | Current | +|UE5.3 | Current | +|UE5.4 | Current | + +### Platform compatability + +The mod.io Unreal Engine plugin supports Windows, Linux, Android and macOS as part of the public release. For access to Windows (GDK), Xbox, PlayStation®4, PlayStation®5 or Switch, follow the [instructions here](/platforms/console-sdks/). + +## Features + +* C++ and Blueprint support +* UGC browsing and filtering +* Automatic downloads and updates +* Customizable component-based user interface +* Permissive MIT/BSL-license +* Async delegate-based interface +* Non-blocking IO +* SSO authentication for Steam, Epic Games Store, console platform and custom implementations \ No newline at end of file diff --git a/Doc/doc_root/en-us/installation-and-setup.mdx b/Doc/doc_root/en-us/installation-and-setup.mdx new file mode 100644 index 00000000..361d74ed --- /dev/null +++ b/Doc/doc_root/en-us/installation-and-setup.mdx @@ -0,0 +1,78 @@ +--- +id: ue-installation-and-setup +title: Installation & Setup +slug: /unreal/installation-and-setup/ +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/installation-and-setup.mdx +--- + +## Adding plugin files to your project + +Download the Unreal Engine plugin from the [mod.io GitHub repository](https://github.com/modio/modio-ue) using one of the following methods: + +### As a git submodule + +- In the directory with your `.uproject` file, add and download the mod.io plugin as a git submodule with the command: + + `git submodule add https://github.com/modio/modio-ue.git Plugins/Modio` + +- Next, initialize our submodules and dependencies: + + `git submodule update --init --recursive` + +### In a non-git project, or without submodules + +- Grab the latest release zip from the [releases section](https://github.com/modio/modio-ue/releases) on GitHub and extract the contents to your project’s `Plugins/Modio` directory + +![get_latest_release](img/get_latest_release.png) ![get_latest_release2](img/get_latest_release2.png) + +:::note +GitHub's automatically generated zips will not work! They do not contain our submodule dependencies. Ensure you download the zip file from the releases page as highlighted above. +::: + +## Enabling the plugin + +Start the editor and enable the plugin in the Unreal Engine plugin settings. + +![plugin_window](img/plugin_window.png) + + + +## Plugin configuration + +The plugin requires some initial configuration to behave correctly. You can access these settings in the Project Settings window: + +![plugin_settings](img/plugin_settings.png) + +The settings have the following parameters: + +| Property | Description | +| --- | --- | +| Game Id | Your mod.io-provided Game Id for the target environment | +| API Key | Your mod.io-provided API key for the target environment | +| Environment | Your target environment, `Live` or `Test` | +| Log Level | The default logging level to use. Messages with a lower log level will be silently discarded. | +| Portal | The default portal to use. This usually corresponds to the store your game will be distributed through. | +| Use Background Thread | Enables or disables the use of a [background thread](#using-a-background-thread) for the plugin's work. | + +:::note +The Test environment is available for usage in limited circumstances only. **[All games should be set up on the Live environment](/unreal/installation-and-setup/#using-the-live-environment)**. Set your game to **hidden** to restrict access during the testing phase. If you require access to Test, please contact us. +::: + +## Using a background thread + +[`RunPendingHandlers`](/unreal/refdocs/#run-pending-handlers) is responsible for running any pending plugin work, including invoking callbacks passed to asynchronous operations. With `Use Background Thread` set to `true`, the mod.io plugin will automatically spin up a background thread and continually call `RunPendingHandlers` for you. This decouples the frequency of the plugin’s work from your game's main thread, and improves performance of the plugin overall. + +Calling methods from the [`ModioSubsystem`](/unreal/refdocs/#modiosubsystem) will marshall all callbacks back to the game thread. All other plugin call guarantees remain the same. + +## Using the `Live` environment + +All games should be set up on the `Live` environment for full access to the mod.io REST API's capabilities. + +If your game or it's modding capabilities are not yet publicly available, set your game to **hidden** on the `Live` environment. This allows you to restrict access to specific accounts and perform QA against the production environment without exposing your title to the public. + +Once you are ready, change the state from **hidden** to **public** for a full release of your game’s modding capabilities. + +## Next Steps + +* Our [Getting Started Guide](/unreal/getting-started/) page contains a series of quick-start guides with code samples demonstrating the plugin's core functionality. +* A sample project demonstrating basic mod.io functionality is available [here](https://go.mod.io/ue5-sample). diff --git a/Doc/doc_root/en-us/marketplace.mdx b/Doc/doc_root/en-us/marketplace.mdx new file mode 100644 index 00000000..1ef7977d --- /dev/null +++ b/Doc/doc_root/en-us/marketplace.mdx @@ -0,0 +1,25 @@ +--- +id: ue-marketplace +title: Marketplace +slug: /unreal/marketplace/ +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/marketplace.mdx +--- + +## Setup + +The mod.io Unreal Engine Plugin supports in-game integration of the mod.io Marketplace - exposing the data and functions required to build a UGC store in-game. + +The mod.io monetization features are need to be enabled through the web UI and, if selling the virtual currency through a platform store, will need a service to consume the virtual currency entitlements and apply them to player's mod.io account. + +## Implementation +Marketplace functionality is accessible via the mod.io Subsystem along with all other functionality and as Blueprint nodes. Functions specific to monetization include: + +- `GetUserWalletBalanceAsync` +- `FetchUserPurchasesAsync` +- `PurchaseModAsync` + +## References + +* [Monetization Introduction](/monetization/) +* [SDK Monetization Guide](/cppsdk/getting-started/#monetization) +* [Purchase Server Implementation](/web-services/marketplace/overview/) diff --git a/Doc/doc_root/en-us/mod-creation-tool-documentation.mdx b/Doc/doc_root/en-us/mod-creation-tool-documentation.mdx new file mode 100644 index 00000000..6b42e57e --- /dev/null +++ b/Doc/doc_root/en-us/mod-creation-tool-documentation.mdx @@ -0,0 +1,174 @@ +--- +id: ue-mod-creation-tool +title: Mod Creation Tool +slug: /unreal/mod-creation-tool/ +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/mod-creation-tool-documentation.mdx +--- + +## Mod Creation & Upload Tool + +The mod.io Unreal Engine's plugin includes an editor module which provides an embedded mod.io - Content Creation and Upload Tool. + +### Editor-specific plugin modules + +| Module Name | Description | Module Type | +| --- | --- | --- | +| ModioEditor | Editor details customization and asset factories for UE content creation and upload tool classes | Editor | + +### Opening the tool +Click the mod.io icon in Unreal Engine to show a drop down menu, click the **Create & Upload** menu. Once opened, it may require mod.io authentication as below. + +![how_to_open_tool](img/Tool/how_to_open_tool.jpg) + +### Authenticating + +Authentication may be required either being a first time user of the module or not logged in previously. If you are already logged in, you will automatically be redirected to creating or editing mods. + +The user will be authenticated if not already logged in to mod.io account. + +**Email:** +Enter your email in the text box shown below and click Login, a verification code will be sent to the submitted email address. + +![auth_enter_email](img/Tool/auth_enter_email.jpg) + +#### Verification Code + +**Verification Code:** Enter the code received in your email and click Authenticate. + +![auth_enter_code](img/Tool/auth_enter_code.jpg) + +If everything goes correctly, you will be authenticated successfully. + +### Create or Edit Mods +After successfull authentication, you can either **Create Mod** or **Edit Mods**. + +![create_or_upload_mod](img/Tool/create_or_upload_mod.jpg) + +#### Create Mod + +**Create Mod** lets you create a new mod. + +![create_mod](img/Tool/create_mod.jpg) + +#### Edit Mods + +**Edit Mods** lets you edit existing mods and change any specific values or mod files on them. + +![edit_mods](img/Tool/edit_mods.jpg) + +### Create a Mod +Creating a mod is made easy directly from within Unreal Engine described in the +process below. + +#### Create Mod + +Click **Create Mod** and fill the following required fields: + +**Create Mod Properties** + +| Fields | Description | +| --- | --- | +| Path to Logo File | Browse to select your **.png** file for logo | +| Name | Name of the mod | +| Summary | A brief summary of the mod | + +![create_mod_properties](img/Tool/create_mod_properties.jpg) + +Once filled all the required fields, click **Submit** to create the mod. + +![create_mod_properties_submit](img/Tool/create_mod_properties_submit.jpg) + +**TIP**: You may get an error dialog if the game doesn't support your platform or the game is locked. + +![error_platform_not_supported](img/Tool/error_platform_not_supported.jpg) +![error_game_locked](img/Tool/error_game_locked.jpg) + +Once everything goes successful you will be asked if you also want to upload a mod file right away. + +![mod_created_modfile_prompt](img/Tool/mod_created_modfile_prompt.jpg) + +If **Yes** is selected you will be redirected to **STEP 6**. + +### Edit existing mods +Editing existing mods is made easy in the same way as creating a mod directly from within Unreal Engine described in the process below. + +#### Edit Mods + +By clicking **Edit Mods**, a list of mods will be shown to edit. You can select a mod by clicking on it. + +**Browse Mods** + +| Name | Description | +| --- | --- | +| Your mod name | Your mod description | + +![browse_mods](img/Tool/browse_mods.jpg) + +Once a mod is selected, click **Edit Mod** to edit the mod. + +![browse_mods_edit_mods](img/Tool/browse_mods_edit_mods.jpg) + +**Edit Mod Properties** + +| Field | Description | +| --- | --- | +| Name | Your mod name | +| Summary | Your mod summary | +| Homepage URL | Your mod homepage url | + +![edit_mod_properties](img/Tool/edit_mod_properties.jpg) + +If you want to change the mod properties, make desirable changes to above fields and click **Submit**. + +![edit_mod_properties_submit](img/Tool/edit_mod_properties_submit.jpg) + +Then click **Edit Files** to see list of existing mod files. + +![edit_mod_properties_edit_files](img/Tool/edit_mod_properties_edit_files.jpg) + +**Modfile** + +| Name | Platform | Version | Status | +| --- | --- | --- | --- | +| Your mod file name | Your mod platform | Your mod file version | Your mod file status | + +![browse_modfile](img/Tool/browse_modfile.jpg) + +### Uploading a new modfile +To upload a mod file, a workspace directory path is mandatory which is described in the process below. + +Click **New Modfile** + +![browse_modfile_new_modfile](img/Tool/browse_modfile_new_modfile.jpg) + +You may need to select **Create mod for PC** + +![create_mod_for_pc](img/Tool/create_mod_for_pc.jpg) + +Fill the required fields: + +TIP: You may get an error dialog if the game doesn't support your platform or the game is locked. + +**Upload Mod File** + +| Field | Description | +| --- | --- | +| Version | Your mod file version | +| Changelog | Your mod file platform | +| Set as Active Release | Whether to set as Active Release | +| Path to Mod Root Directory | Path to the workspace directory | + +![upload_modfile](img/Tool/upload_modfile.jpg) + +Once all fields are filled out, click **Submit** + +![upload_modfile_submit](img/Tool/upload_modfile_submit.jpg) + +Once submitted, the progress bar will be displayed about the workspace directory being zipped and uploaded with all other information provided for the new mod file. + +![upload_modfile_submit_status](img/Tool/upload_modfile_submit_status.jpg) + +### Mod file upload +**TIP**: You may get an error dialog if the game doesn't support your platform or the game is locked or any error that occurs due to internet or any other circumstances. + +![modfile_successfully_uploaded](img/Tool/modfile_successfully_uploaded.jpg) \ No newline at end of file diff --git a/Doc/doc_root/en-us/profiling.mdx b/Doc/doc_root/en-us/profiling.mdx new file mode 100644 index 00000000..fc334b42 --- /dev/null +++ b/Doc/doc_root/en-us/profiling.mdx @@ -0,0 +1,14 @@ +--- +id: ue-profiling +title: Profiling +slug: /unreal/profiling/ +custom_edit_url: https://github.com/modio/modio-ue4-internal/blob/develop/Plugins/Modio/Doc/doc_root/en-us/profiling.mdx +--- + +The mod.io UE plugin includes some basic profiling and metrics that can be viewed and exposed via Unreal’s `stat` commands and via Unreal Insights. Profiling via the `MODIO_ENABLE_PROFILING` and `MODIO_UNREAL_PROFILING_SUPPORT` macros are enabled by default in the mod.io plugin for non-shipping builds. + +## Showing Captured Data + +The plugin includes two stat groups that can be shown: `Modio` (accessed via `stat modio`) and ModioScoped (async operations profiled for access via Unreal Insights). +* `Modio` includes some basic counters, such as the length of various background queues that process API requests and downloads. +* `ModioScoped` captures performance insights for a majority of the async operations that the plugin performs. This data is not suitable for showing in realtime via a `stat` command, however it is captured when creating stat profiles for use in Unreal Insights. \ No newline at end of file diff --git a/Doc/img/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions.png b/Doc/img/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions.png index 48104bbc..a6a3f56a 100644 Binary files a/Doc/img/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions.png and b/Doc/img/nd_img_ModioCommonTypesLibrary_MakeInitializeOptions.png differ diff --git a/Doc/img/nd_img_ModioCommonTypesLibrary_NotEqualTo.png b/Doc/img/nd_img_ModioCommonTypesLibrary_NotEqualTo.png index 4d50cf21..634a30da 100644 Binary files a/Doc/img/nd_img_ModioCommonTypesLibrary_NotEqualTo.png and b/Doc/img/nd_img_ModioCommonTypesLibrary_NotEqualTo.png differ diff --git a/Doc/img/nd_img_ModioCommonTypesLibrary_SetBackgroundThread.png b/Doc/img/nd_img_ModioCommonTypesLibrary_SetBackgroundThread.png new file mode 100644 index 00000000..a32f2528 Binary files /dev/null and b/Doc/img/nd_img_ModioCommonTypesLibrary_SetBackgroundThread.png differ diff --git a/Doc/img/nd_img_ModioCommonTypesLibrary_SetPortal.png b/Doc/img/nd_img_ModioCommonTypesLibrary_SetPortal.png index 92f2537c..8462dd45 100644 Binary files a/Doc/img/nd_img_ModioCommonTypesLibrary_SetPortal.png and b/Doc/img/nd_img_ModioCommonTypesLibrary_SetPortal.png differ diff --git a/Doc/img/nd_img_ModioCommonTypesLibrary_SetSessionId.png b/Doc/img/nd_img_ModioCommonTypesLibrary_SetSessionId.png index 5cd2f8db..159bf985 100644 Binary files a/Doc/img/nd_img_ModioCommonTypesLibrary_SetSessionId.png and b/Doc/img/nd_img_ModioCommonTypesLibrary_SetSessionId.png differ diff --git a/Doc/img/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier.png b/Doc/img/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier.png index c545e096..4010a428 100644 Binary files a/Doc/img/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier.png and b/Doc/img/nd_img_ModioCommonTypesLibrary_SetSessionIdentifier.png differ diff --git a/Doc/img/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob.png b/Doc/img/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob.png index 2481e581..c41da116 100644 Binary files a/Doc/img/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob.png and b/Doc/img/nd_img_ModioCreateModLibrary_SetModFileMetadataBlob.png differ diff --git a/Doc/img/nd_img_ModioCreateModLibrary_SetVersionString.png b/Doc/img/nd_img_ModioCreateModLibrary_SetVersionString.png index 30c26472..01f959c8 100644 Binary files a/Doc/img/nd_img_ModioCreateModLibrary_SetVersionString.png and b/Doc/img/nd_img_ModioCreateModLibrary_SetVersionString.png differ diff --git a/Doc/img/nd_img_ModioErrorCodeLibrary_GetMessage.png b/Doc/img/nd_img_ModioErrorCodeLibrary_GetMessage.png index 06387798..0c172019 100644 Binary files a/Doc/img/nd_img_ModioErrorCodeLibrary_GetMessage.png and b/Doc/img/nd_img_ModioErrorCodeLibrary_GetMessage.png differ diff --git a/Doc/img/nd_img_ModioErrorCodeLibrary_GetValue.png b/Doc/img/nd_img_ModioErrorCodeLibrary_GetValue.png index 3e51285c..ade4391b 100644 Binary files a/Doc/img/nd_img_ModioErrorCodeLibrary_GetValue.png and b/Doc/img/nd_img_ModioErrorCodeLibrary_GetValue.png differ diff --git a/Doc/img/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync.png b/Doc/img/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync.png index e32ec8c9..75b0e4e7 100644 Binary files a/Doc/img/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync.png and b/Doc/img/nd_img_ModioExampleLibrary_ListUserSubscriptionAsync.png differ diff --git a/Doc/img/nd_img_ModioImageLibrary_GetState.png b/Doc/img/nd_img_ModioImageLibrary_GetState.png index 99e0cd72..3fbff1bc 100644 Binary files a/Doc/img/nd_img_ModioImageLibrary_GetState.png and b/Doc/img/nd_img_ModioImageLibrary_GetState.png differ diff --git a/Doc/img/nd_img_ModioImageLibrary_GetTexture.png b/Doc/img/nd_img_ModioImageLibrary_GetTexture.png index 180b2495..da6a565c 100644 Binary files a/Doc/img/nd_img_ModioImageLibrary_GetTexture.png and b/Doc/img/nd_img_ModioImageLibrary_GetTexture.png differ diff --git a/Doc/img/nd_img_ModioModCollectionLibrary_GetID.png b/Doc/img/nd_img_ModioModCollectionLibrary_GetID.png index 2e32ebd4..6bfe6c96 100644 Binary files a/Doc/img/nd_img_ModioModCollectionLibrary_GetID.png and b/Doc/img/nd_img_ModioModCollectionLibrary_GetID.png differ diff --git a/Doc/img/nd_img_ModioModProgressInfoLibrary_GetCurrentState.png b/Doc/img/nd_img_ModioModProgressInfoLibrary_GetCurrentState.png index 4700131c..1ec27ddd 100644 Binary files a/Doc/img/nd_img_ModioModProgressInfoLibrary_GetCurrentState.png and b/Doc/img/nd_img_ModioModProgressInfoLibrary_GetCurrentState.png differ diff --git a/Doc/img/nd_img_ModioModProgressInfoLibrary_GetTotalProgress.png b/Doc/img/nd_img_ModioModProgressInfoLibrary_GetTotalProgress.png index 6b588d10..3ee23c47 100644 Binary files a/Doc/img/nd_img_ModioModProgressInfoLibrary_GetTotalProgress.png and b/Doc/img/nd_img_ModioModProgressInfoLibrary_GetTotalProgress.png differ diff --git a/Doc/img/nd_img_ModioPresetFilterParamsLibrary_ToFilterParams.png b/Doc/img/nd_img_ModioPresetFilterParamsLibrary_ToFilterParams.png index f1030075..e98377e1 100644 Binary files a/Doc/img/nd_img_ModioPresetFilterParamsLibrary_ToFilterParams.png and b/Doc/img/nd_img_ModioPresetFilterParamsLibrary_ToFilterParams.png differ diff --git a/Doc/img/nd_img_ModioSDKLibrary_GetMonetizationPurchaseCategory.png b/Doc/img/nd_img_ModioSDKLibrary_GetMonetizationPurchaseCategory.png new file mode 100644 index 00000000..888e1e4c Binary files /dev/null and b/Doc/img/nd_img_ModioSDKLibrary_GetMonetizationPurchaseCategory.png differ diff --git a/Doc/img/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory.png b/Doc/img/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory.png index 91de5136..d1dcc3ea 100644 Binary files a/Doc/img/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory.png and b/Doc/img/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFileForModFromMemory.png differ diff --git a/Doc/img/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync.png b/Doc/img/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync.png index 7a0a604b..4747bac8 100644 Binary files a/Doc/img/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync.png and b/Doc/img/nd_img_ModioSubmissionExtensionLibrary_K2_SubmitNewModFromMemoryAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_CloseTempModSet.png b/Doc/img/nd_img_ModioSubsystem_K2_CloseTempModSet.png index 489c0c56..db66c41a 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_CloseTempModSet.png and b/Doc/img/nd_img_ModioSubsystem_K2_CloseTempModSet.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync.png index 6cc30a31..df1e4049 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_FetchExternalUpdatesAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync.png index 9519f93a..8e219a39 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_FetchUserPurchasesAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync.png index 8175f93c..b5a91d2d 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_GetModMediaGalleryImageAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync.png index 678e320f..a0ab878c 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_GetUserWalletBalanceAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_InitTempModSet.png b/Doc/img/nd_img_ModioSubsystem_K2_InitTempModSet.png index a9e4b009..70d24793 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_InitTempModSet.png and b/Doc/img/nd_img_ModioSubsystem_K2_InitTempModSet.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_ListAllModsAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_ListAllModsAsync.png index a6b764ca..7efcf565 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_ListAllModsAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_ListAllModsAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync.png index 37f1cc60..918f2734 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_MetricsSessionSendHeartbeatAtIntervalAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_PurchaseModAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_PurchaseModAsync.png index 7c9b9ce3..e9e01a82 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_PurchaseModAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_PurchaseModAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_QueryTempModSet.png b/Doc/img/nd_img_ModioSubsystem_K2_QueryTempModSet.png index df294550..c53cb9a3 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_QueryTempModSet.png and b/Doc/img/nd_img_ModioSubsystem_K2_QueryTempModSet.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_RemoveFromTempModSet.png b/Doc/img/nd_img_ModioSubsystem_K2_RemoveFromTempModSet.png index 688c8dd8..a3c289e6 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_RemoveFromTempModSet.png and b/Doc/img/nd_img_ModioSubsystem_K2_RemoveFromTempModSet.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_ReportContentAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_ReportContentAsync.png index 5696e671..df199f0e 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_ReportContentAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_ReportContentAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_SubmitNewModAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_SubmitNewModAsync.png index 34251e69..62e6f599 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_SubmitNewModAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_SubmitNewModAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_SubscribeToModAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_SubscribeToModAsync.png index 587b01f6..389e6852 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_SubscribeToModAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_SubscribeToModAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_UnmuteUserAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_UnmuteUserAsync.png index 1175a92d..d4c73efe 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_UnmuteUserAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_UnmuteUserAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync.png b/Doc/img/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync.png index 2c07b9ba..59f5c69b 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync.png and b/Doc/img/nd_img_ModioSubsystem_K2_UnsubscribeFromModAsync.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_PrioritizeTransferForMod.png b/Doc/img/nd_img_ModioSubsystem_PrioritizeTransferForMod.png index 71ac0375..850ec592 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_PrioritizeTransferForMod.png and b/Doc/img/nd_img_ModioSubsystem_PrioritizeTransferForMod.png differ diff --git a/Doc/img/nd_img_ModioSubsystem_QueryUserInstallations.png b/Doc/img/nd_img_ModioSubsystem_QueryUserInstallations.png index f2b7df79..c83e96c2 100644 Binary files a/Doc/img/nd_img_ModioSubsystem_QueryUserInstallations.png and b/Doc/img/nd_img_ModioSubsystem_QueryUserInstallations.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_GetDescription.png b/Doc/img/nd_img_ModioTokenPackLibrary_GetDescription.png new file mode 100644 index 00000000..829e21b8 Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_GetDescription.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_GetDisplayPrice.png b/Doc/img/nd_img_ModioTokenPackLibrary_GetDisplayPrice.png new file mode 100644 index 00000000..aaf91c54 Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_GetDisplayPrice.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_GetFields.png b/Doc/img/nd_img_ModioTokenPackLibrary_GetFields.png new file mode 100644 index 00000000..26cb91dd Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_GetFields.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_GetId.png b/Doc/img/nd_img_ModioTokenPackLibrary_GetId.png new file mode 100644 index 00000000..b7b6e525 Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_GetId.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_GetLongDescription.png b/Doc/img/nd_img_ModioTokenPackLibrary_GetLongDescription.png new file mode 100644 index 00000000..dc20e222 Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_GetLongDescription.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_GetModioId.png b/Doc/img/nd_img_ModioTokenPackLibrary_GetModioId.png new file mode 100644 index 00000000..f498d840 Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_GetModioId.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_GetNumericPrice.png b/Doc/img/nd_img_ModioTokenPackLibrary_GetNumericPrice.png new file mode 100644 index 00000000..aeaed84b Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_GetNumericPrice.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_GetRegularPrice.png b/Doc/img/nd_img_ModioTokenPackLibrary_GetRegularPrice.png new file mode 100644 index 00000000..b409259b Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_GetRegularPrice.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_GetTitle.png b/Doc/img/nd_img_ModioTokenPackLibrary_GetTitle.png new file mode 100644 index 00000000..3dd1efe4 Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_GetTitle.png differ diff --git a/Doc/img/nd_img_ModioTokenPackLibrary_IsPurchasable.png b/Doc/img/nd_img_ModioTokenPackLibrary_IsPurchasable.png new file mode 100644 index 00000000..e1eb4cce Binary files /dev/null and b/Doc/img/nd_img_ModioTokenPackLibrary_IsPurchasable.png differ diff --git a/Doc/img/nd_img_ModioUnsigned64Library_Divide.png b/Doc/img/nd_img_ModioUnsigned64Library_Divide.png index 498e7b3c..06a9f7df 100644 Binary files a/Doc/img/nd_img_ModioUnsigned64Library_Divide.png and b/Doc/img/nd_img_ModioUnsigned64Library_Divide.png differ diff --git a/Doc/img/nd_img_ModioUnsigned64Library_DivideToFloat.png b/Doc/img/nd_img_ModioUnsigned64Library_DivideToFloat.png index 2a4e511d..3282e9e7 100644 Binary files a/Doc/img/nd_img_ModioUnsigned64Library_DivideToFloat.png and b/Doc/img/nd_img_ModioUnsigned64Library_DivideToFloat.png differ diff --git a/Doc/img/nd_img_ModioUnsigned64Library_GreaterThan.png b/Doc/img/nd_img_ModioUnsigned64Library_GreaterThan.png index fa79cde6..aa57f583 100644 Binary files a/Doc/img/nd_img_ModioUnsigned64Library_GreaterThan.png and b/Doc/img/nd_img_ModioUnsigned64Library_GreaterThan.png differ diff --git a/Doc/img/nd_img_ModioUnsigned64Library_LessThan.png b/Doc/img/nd_img_ModioUnsigned64Library_LessThan.png index 668f76f0..c9883f55 100644 Binary files a/Doc/img/nd_img_ModioUnsigned64Library_LessThan.png and b/Doc/img/nd_img_ModioUnsigned64Library_LessThan.png differ diff --git a/Doc/img/nd_img_ModioUnsigned64Library_MakeFromComponents.png b/Doc/img/nd_img_ModioUnsigned64Library_MakeFromComponents.png index 4ed4c572..fef40741 100644 Binary files a/Doc/img/nd_img_ModioUnsigned64Library_MakeFromComponents.png and b/Doc/img/nd_img_ModioUnsigned64Library_MakeFromComponents.png differ diff --git a/Doc/img/nd_img_ModioUnsigned64Library_NotEqualTo.png b/Doc/img/nd_img_ModioUnsigned64Library_NotEqualTo.png index 4d50cf21..634a30da 100644 Binary files a/Doc/img/nd_img_ModioUnsigned64Library_NotEqualTo.png and b/Doc/img/nd_img_ModioUnsigned64Library_NotEqualTo.png differ diff --git a/Doc/img/nd_img_ModioUnsigned64Library_Percentage_Unsigned64.png b/Doc/img/nd_img_ModioUnsigned64Library_Percentage_Unsigned64.png index 0a971ab9..b9bec54b 100644 Binary files a/Doc/img/nd_img_ModioUnsigned64Library_Percentage_Unsigned64.png and b/Doc/img/nd_img_ModioUnsigned64Library_Percentage_Unsigned64.png differ diff --git a/Doc/img/nd_img_ModioUnsigned64Library_Subtract.png b/Doc/img/nd_img_ModioUnsigned64Library_Subtract.png index e66f5f61..cb6f0684 100644 Binary files a/Doc/img/nd_img_ModioUnsigned64Library_Subtract.png and b/Doc/img/nd_img_ModioUnsigned64Library_Subtract.png differ diff --git a/Doc/index.html b/Doc/index.html index 7e70d7f3..369267f9 100644 --- a/Doc/index.html +++ b/Doc/index.html @@ -4,7 +4,7 @@ - + diff --git a/Modio.uplugin b/Modio.uplugin index 5bf3a6ea..1d0c5674 100644 --- a/Modio.uplugin +++ b/Modio.uplugin @@ -65,6 +65,10 @@ { "Name": "PluginBrowser", "Enabled": true + }, + { + "Name": "OnlineSubsystem", + "Enabled": true } ] } \ No newline at end of file diff --git a/Source/Modio/Modio.Build.cs b/Source/Modio/Modio.Build.cs index 4b9d2132..e1237c20 100644 --- a/Source/Modio/Modio.Build.cs +++ b/Source/Modio/Modio.Build.cs @@ -415,7 +415,8 @@ private void AddCommonHeaderPaths(string GeneratedHeaderPath) Path.Combine(ModuleDirectory, "../ThirdParty/concurrentqueue"), Path.Combine(GeneratedHeaderPath, "Private"), - Path.Combine(GeneratedHeaderPath, "Public") + Path.Combine(GeneratedHeaderPath, "Public"), + GeneratedHeaderPath }); @@ -428,7 +429,7 @@ private void AddCommonDependencyModules() }); PrivateDependencyModuleNames.AddRange(new string[] { - "Projects", "CoreUObject", "RHI", "RenderCore", "HTTP" + "Projects", "CoreUObject", "RHI", "RenderCore", "HTTP", "OnlineSubsystem" }); if (Target.bBuildEditor) @@ -513,14 +514,15 @@ private void AddCommonDefinitions() string VersionString = File.ReadAllText(VersionFilePath); VersionString = VersionString.Trim(); PrivateDefinitions.Add(string.Format("MODIO_COMMIT_HASH=\"UNREAL-{0}\"", VersionString)); + // Add dependency on version file so if it is changed we trigger a rebuild + ExternalDependencies.Add(VersionFilePath); } else { PrivateDefinitions.Add("MODIO_COMMIT_HASH=\"UNREAL-DEV\""); } - // Add dependency on version file so if it is changed we trigger a rebuild - ExternalDependencies.Add(VersionFilePath); + } private bool DoesPlatformSupportProfiling(UnrealTargetPlatform Platform) diff --git a/Source/Modio/Private/Libraries/ModioCommonTypesLibrary.cpp b/Source/Modio/Private/Libraries/ModioCommonTypesLibrary.cpp index 976ceb18..490f2a9e 100644 --- a/Source/Modio/Private/Libraries/ModioCommonTypesLibrary.cpp +++ b/Source/Modio/Private/Libraries/ModioCommonTypesLibrary.cpp @@ -74,13 +74,14 @@ FString UModioCommonTypesLibrary::Conv_UserIDToString(FModioUserID UserID) FModioInitializeOptions UModioCommonTypesLibrary::MakeInitializeOptions(int64 GameId, const FString& APIKey, EModioEnvironment GameEnvironment, - EModioPortal PortalInUse) + EModioPortal PortalInUse, bool bUseBackgroundThread) { FModioInitializeOptions Options; Options.GameId = FModioGameID(GameId); Options.ApiKey = FModioApiKey(APIKey); Options.GameEnvironment = GameEnvironment; Options.PortalInUse = PortalInUse; + Options.bUseBackgroundThread = bUseBackgroundThread; return Options; } @@ -115,8 +116,15 @@ FModioInitializeOptions UModioCommonTypesLibrary::SetPortal(const FModioInitiali return DuplicateOptions; } +FModioInitializeOptions UModioCommonTypesLibrary::SetBackgroundThread(const FModioInitializeOptions& Options, bool bUseBackgroundThread) +{ + FModioInitializeOptions DuplicateOptions {Options}; + DuplicateOptions.bUseBackgroundThread = bUseBackgroundThread; + return DuplicateOptions; +} + FModioInitializeOptions UModioCommonTypesLibrary::SetSessionIdentifier(const FModioInitializeOptions& Options, - const FString& SessionIdentifier) + const FString& SessionIdentifier) { FModioInitializeOptions DuplicateOptions {Options}; DuplicateOptions.LocalSessionIdentifier = SessionIdentifier; diff --git a/Source/Modio/Private/Libraries/ModioSDKLibrary.cpp b/Source/Modio/Private/Libraries/ModioSDKLibrary.cpp index 358742e2..07a6bfeb 100644 --- a/Source/Modio/Private/Libraries/ModioSDKLibrary.cpp +++ b/Source/Modio/Private/Libraries/ModioSDKLibrary.cpp @@ -15,6 +15,7 @@ #include "ModioSettings.h" #include "Internal/ModioConvert.h" #include "Misc/EngineVersionComparison.h" +#include "Libraries/ModioPlatformHelpersLibrary.h" FModioGameID UModioSDKLibrary::GetProjectGameId() @@ -297,3 +298,16 @@ EModioLanguage UModioSDKLibrary::GetLanguageCodeFromString(FString LanguageCode) } return EModioLanguage::English; } + +FString UModioSDKLibrary::GetMonetizationPurchaseCategory(EModioPortal Portal) +{ + const UModioSettings* Settings = GetDefault(); + + if (!Settings->PlatformIdentifiers.Contains(Portal)) + { + UE_LOG(LogTemp, Warning, TEXT("Could not find current platform in Platform identifiers. Platform: %d"), (int)Portal); + return FString(); + } + + return Settings->PlatformIdentifiers[Portal]; +} diff --git a/Source/Modio/Private/ModioSubsystem.cpp b/Source/Modio/Private/ModioSubsystem.cpp index 1bbb9d08..31e3e3fa 100644 --- a/Source/Modio/Private/ModioSubsystem.cpp +++ b/Source/Modio/Private/ModioSubsystem.cpp @@ -172,6 +172,8 @@ void UModioSubsystem::InitializeAsync(const FModioInitializeOptions& Options, FO BackgroundThread = MakeUnique(this); } + CachedInitializeOptions = Options; + #if WITH_EDITOR if (const UModioSettings* Settings = GetMutableDefault()) { diff --git a/Source/Modio/Private/Types/ModioCommonTypes.cpp b/Source/Modio/Private/Types/ModioCommonTypes.cpp index 5171aa2c..eaf52ed8 100644 --- a/Source/Modio/Private/Types/ModioCommonTypes.cpp +++ b/Source/Modio/Private/Types/ModioCommonTypes.cpp @@ -70,4 +70,11 @@ FModioGuid FModioGuid::InvalidGuid() FModioGuid FModioGuid::GenerateGuid() { return FModioGuid(ToUnreal(*Modio::Guid::GenerateGuid())); -} \ No newline at end of file +} + +FModioTokenPackID::FModioTokenPackID() {} + +uint32 GetTypeHash(FModioTokenPackID ModioTokenPackId) +{ + return FCrc::MemCrc32(&ModioTokenPackId, sizeof(FModioTokenPackID)); +} diff --git a/Source/Modio/Private/Types/ModioTokenPackListUImpl.cpp b/Source/Modio/Private/Types/ModioTokenPackListUImpl.cpp new file mode 100644 index 00000000..50a504d2 --- /dev/null +++ b/Source/Modio/Private/Types/ModioTokenPackListUImpl.cpp @@ -0,0 +1,27 @@ +/* + * Copyright (C) 2024 mod.io Pty Ltd. + * + * This file is part of the mod.io UE4 Plugin. + * + * Distributed under the MIT License. (See accompanying file LICENSE or + * view online at ) + * + */ + +#include "Types/ModioTokenPackList.h" + +#include "Internal/Convert/List.h" +#include "ModioSDK.h" + +FModioTokenPackList::FModioTokenPackList(const FModioPagedResult& PagedResult, TArray&& TokenPackList) + : FModioPagedResult(PagedResult), + FModioList(MoveTemp(TokenPackList)) +{} + +FModioTokenPackList::FModioTokenPackList(TArray TokenPackList) + : FModioList(MoveTemp(TokenPackList)) +{} + +FModioOptionalTokenPackList::FModioOptionalTokenPackList(TOptional&& TokenPackList) + : Internal(MoveTemp(TokenPackList)) +{} diff --git a/Source/Modio/Public/Libraries/ModioCommonTypesLibrary.h b/Source/Modio/Public/Libraries/ModioCommonTypesLibrary.h index 4b099d46..46327ae5 100644 --- a/Source/Modio/Public/Libraries/ModioCommonTypesLibrary.h +++ b/Source/Modio/Public/Libraries/ModioCommonTypesLibrary.h @@ -23,16 +23,17 @@ class MODIO_API UModioCommonTypesLibrary : public UBlueprintFunctionLibrary GENERATED_BODY() public: /** - * @brief Create a game id from a integer, should only be used in conjunction with InitializeAsync - * @param GameId - a positive integer that maps to your game + * @brief Create a game id from a integer. Should only be used in conjunction with + * [`InitializeAsync`](#initializeasync) + * @param GameId A positive integer that maps to your game */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities", meta = (NativeMakeFunc)) static FModioGameID MakeGameId(int64 GameId); /** - * @brief Converts a game id to a string so you can debug output it - * @param GameId - Input Game Id - * @return InvalidGameId if invalid + * @brief Converts a game id to a string for debugging purposes + * @param GameId Input Game Id + * @return Empty if invalid */ UFUNCTION(BlueprintPure, meta = (DisplayName = "ToString (ModioGameId)", CompactNodeTitle = "->", BlueprintAutocast), @@ -40,27 +41,29 @@ class MODIO_API UModioCommonTypesLibrary : public UBlueprintFunctionLibrary static FString Conv_GameIDToString(FModioGameID GameId); /** - * @brief Creates an AuthenticationParams object - * @param AuthToken - Authentication provider-supplied OAuth token - * @param EmailAddress - User email address, can be left blank - * @param bHasAcceptedTOS - Has the user been shown the TOS and accepted the TOS? - * @return The constructed FModioAuthenticationParams object for use with <> + * @brief Creates a `ModioAuthenticationParams` object + * @param AuthToken Authentication provider-supplied OAuth token + * @param EmailAddress User email address, can be left blank + * @param bHasAcceptedTOS Has the user been shown the Terms of Service and accepted the Terms of Service? + * @return The constructed `ModioAuthenticationParams` object for use with + * [`AuthenticateUserExternalAsync`](#authenticateuserexternalasync) */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities", meta = (NativeMakeFunc)) static FModioAuthenticationParams MakeAuthParams(const FString AuthToken, const FString EmailAddress, const bool bHasAcceptedTOS); /** - * @brief Create a ApiKey id from a string, should only be used in conjunction with InitializeAsync - * @param ApiKey - a api key from your settings panel on mod.io + * @brief Create an ApiKey id from a string. Should only be used in conjunction with + * [`InitializeAsync`](#initializeasync) + * @param ApiKey The api key from your settings panel on mod.io */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities", meta = (NativeMakeFunc)) static FModioApiKey MakeApiKey(const FString ApiKey); /** - * @brief Converts a ApiKey string so you can debug output it + * @brief Converts a `ModioApiKey` to a string for debugging purposes * @param ApiKey Input Api Key - * @return InvalidApiKey if invalid + * @return Empty if invalid */ UFUNCTION(BlueprintPure, meta = (DisplayName = "ToString (ModioApiKey)", CompactNodeTitle = "->", BlueprintAutocast), @@ -68,18 +71,18 @@ class MODIO_API UModioCommonTypesLibrary : public UBlueprintFunctionLibrary static FString Conv_ApiKeyToString(FModioApiKey ApiKey); /** - * @brief Converts a ModId string so you can debug output it + * @brief Converts a `ModioModID` to a string for debugging purposes * @param ModID Input Mod ID - * @return InvalidModID if invalid + * @return Empty if invalid */ UFUNCTION(BlueprintPure, meta = (DisplayName = "ToString (ModioModID)", CompactNodeTitle = "->", BlueprintAutocast), Category = "mod.io|Utilities|String") static FString Conv_ModIDToString(FModioModID ModID); /** - * @brief Converts a FFileMetadataID to string so you can debug output it + * @brief Converts an `FileMetadataID` to a string for debugging purposes * @param FileMetadataID Input File metadata ID - * @return InvalidFileMetadataID if invalid + * @return Empty if invalid */ UFUNCTION(BlueprintPure, meta = (DisplayName = "ToString (FileMetadataID)", CompactNodeTitle = "->", BlueprintAutocast), @@ -87,9 +90,9 @@ class MODIO_API UModioCommonTypesLibrary : public UBlueprintFunctionLibrary static FString Conv_FileMetadataIDToString(FModioFileMetadataID FileMetadataID); /** - * @brief Converts a UserID string so you can debug output it - * @param UserID - - * @return InvalidUserID if invalid + * @brief Converts a `ModioUserID` to a string for debugging purposes + * @param UserID User ID to convert + * @return Empty if invalid */ UFUNCTION(BlueprintPure, meta = (DisplayName = "ToString (ModioUserID)", CompactNodeTitle = "->", BlueprintAutocast), @@ -97,9 +100,9 @@ class MODIO_API UModioCommonTypesLibrary : public UBlueprintFunctionLibrary static FString Conv_UserIDToString(FModioUserID UserID); /** - * @brief Converts a EmailAddress string so you can debug output it - * @param EmailAddress - - * @return InvalidEmailAddress if invalid + * @brief Converts a `ModioEmailAddress` string for debugging purposes + * @param EmailAddress Email address to convert + * @return Empty if invalid */ UFUNCTION(BlueprintPure, meta = (DisplayName = "ToString (ModioEmailAddress)", CompactNodeTitle = "->", BlueprintAutocast), @@ -107,9 +110,9 @@ class MODIO_API UModioCommonTypesLibrary : public UBlueprintFunctionLibrary static FString Conv_EmailAddressToString(FModioEmailAddress EmailAddress); /** - * @brief Converts a EmailAuthCode string so you can debug output it - * @param EmailAuthCode - - * @return InvalidEmailAuthCode if invalid + * @brief Converts a `ModioEmailAuthCode` string for debugging purposes + * @param EmailAuthCode Email auth code to convert + * @return Empty if invalid */ UFUNCTION(BlueprintPure, meta = (DisplayName = "ToString (ModioEmailAuthCode)", CompactNodeTitle = "->", BlueprintAutocast), @@ -117,7 +120,7 @@ class MODIO_API UModioCommonTypesLibrary : public UBlueprintFunctionLibrary static FString Conv_EmailAuthCodeToString(FModioEmailAuthCode EmailAuthCode); /** - * @brief Make a email address from a string + * @brief Make an email address from a string * @param Email Input string to convert * @return FModioEmailAddress object */ @@ -127,8 +130,8 @@ class MODIO_API UModioCommonTypesLibrary : public UBlueprintFunctionLibrary static FModioEmailAddress Conv_StringToEmailAddress(const FString& Email); /** - * @brief Make a email auth code from a string - * @param GameId Input string + * @brief Make an email auth code from a string + * @param AuthCode Input string to convert * @return Auth code for submission */ UFUNCTION(BlueprintPure, @@ -137,91 +140,110 @@ class MODIO_API UModioCommonTypesLibrary : public UBlueprintFunctionLibrary static FModioEmailAuthCode Conv_StringToEmailAuthCode(const FString& AuthCode); /** - * @brief Make initialization options, should only be used in conjunction with InitializeAsync - * @param GameId - a positive integer that maps to your game - * @param APIKey - APIKey available at https://.test.mod.io/edit/api or - * https://.mod.io/edit/api - * @param GameEnvironment - If your environment is setup on test or production + * @brief Make [`ModioInitializeOptions`](#modioinitializeoptions). Should only be used in conjunction with + * [`InitializeAsync`](#initializeasync). + * @param GameId - A positive integer that maps to your game. This can be found in the admin section of your game's + * page at https://mod.io/ + * @param APIKey - The API key for your game. This can be found in the admin section of your game's page at + * https://mod.io/ + * @param GameEnvironment - The environment your game has been set up on: test or live. + * @param PortalInUse The [`EModioPortal`](#EModioPortal) representing the store or service your game is being + * @param bUseBackgroundThread - Whether to run the mod.io SDK on a background thread. Defaults to `true`. + * distributed through. Defaults to `EModioPortal::None`. */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities", meta = (NativeMakeFunc)) static FModioInitializeOptions MakeInitializeOptions(int64 GameId, const FString& APIKey, - EModioEnvironment GameEnvironment, - EModioPortal PortalInUse = EModioPortal::None); + EModioEnvironment GameEnvironment, + EModioPortal PortalInUse = EModioPortal::None, bool bUseBackgroundThread = true); /** - * @brief Changes the game id for the provided set of initialization options - * @param Options - The template initialization options + * @brief Changes the game id for the provided set of [`ModioInitializeOptions`](#modioinitializeoptions) + * @param Options - The template [`ModioInitializeOptions`](#modioinitializeoptions) * @param GameId - The new game id to use - * @return New Initialization Options object with the game id set to the desired value + * @return New [`ModioInitializeOptions`](#modioinitializeoptions) object with the game id set to the desired value */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities") static FModioInitializeOptions SetGameId(const FModioInitializeOptions& Options, int64 GameId); /** - * @brief Changes the API key for the provided set of initialization options - * @param Options - The template initialization options + * @brief Changes the API key for the provided set of [`ModioInitializeOptions`](#modioinitializeoptions) + * @param Options - The template [`ModioInitializeOptions`](#modioinitializeoptions) * @param APIKey - The new API key to use - * @return New Initialization Options object with the API key set to the desired value + * @return New [`ModioInitializeOptions`](#modioinitializeoptions) object with the API key set to the desired value */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities", meta = (DisplayName = "Set API Key")) static FModioInitializeOptions SetAPIKey(const FModioInitializeOptions& Options, const FString& APIKey); /** - * @brief Changes the game environment for the provided set of initialization options - * @param Options - The template initialization options + * @brief Changes the game environment for the provided set of [`ModioInitializeOptions`](#modioinitializeoptions) + * @param Options - The template [`ModioInitializeOptions`](#modioinitializeoptions) * @param GameEnvironment - The new environment to use - * @return New Initialization Options object with the game environment set to the desired value + * @return New [`ModioInitializeOptions`](#modioinitializeoptions) object with the game environment set to the + * desired value */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities") static FModioInitializeOptions SetGameEnvironment(const FModioInitializeOptions& Options, EModioEnvironment GameEnvironment); /** - * @brief Changes the portal for the provided set of initialization options - * @param Options - The template initialization options + * @brief Changes the portal for the provided set of [`ModioInitializeOptions`](#modioinitializeoptions) + * @param Options - The template [`ModioInitializeOptions`](#modioinitializeoptions) * @param PortalToUse - The new portal to use - * @return New Initialization Options object with the portal set to the desired value + * @return New [`ModioInitializeOptions`](#modioinitializeoptions) object with the portal set to the desired value */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities") static FModioInitializeOptions SetPortal(const FModioInitializeOptions& Options, EModioPortal PortalToUse); /** - * @brief Changes the session identifier for the provided set of initialization options - * @param Options - The template initialization options + * @brief Changes the background thread usage for the provided set of [`ModioInitializeOptions`](#modioinitializeoptions) + * @param Options - The template [`ModioInitializeOptions`](#modioinitializeoptions) + * @param bUseBackgroundThread - Whether to use a background thread + * @return New [`ModioInitializeOptions`](#modioinitializeoptions) object with the background thread usage set to the + * desired value + */ + UFUNCTION(BlueprintPure, category = "mod.io|Utilities") + static FModioInitializeOptions SetBackgroundThread(const FModioInitializeOptions& Options, bool bUseBackgroundThread); + + /** + * @brief Changes the session identifier for the provided set of [`ModioInitializeOptions`](#modioinitializeoptions) + * @param Options - The template [`ModioInitializeOptions`](#modioinitializeoptions) * @param SessionIdentifier - The new session id to use - * @return New Initialization Options object with the session identifier set to the desired value + * @return New [`ModioInitializeOptions`](#modioinitializeoptions) object with the session identifier set to the + *desired value **/ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities") static FModioInitializeOptions SetSessionIdentifier(const FModioInitializeOptions& Options, const FString& SessionIdentifier); /** - * @brief Sets extended initialization parameters for the provided set of initialization options - * @param Options - The template initialization options + * @brief Sets extended initialization parameters for the provided set of + * [`ModioInitializeOptions`](#modioinitializeoptions) + * @param Options - The template [`ModioInitializeOptions`](#modioinitializeoptions) * @param ExtendedParameters - The new extended parameters to use (will overwrite existing values) - * @return New Initialization Options object with the extended parameters set to the desired value + * @return New [`ModioInitializeOptions`](#modioinitializeoptions) object with the extended parameters set to the + * desired value */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities") static FModioInitializeOptions SetExtendedInitializationParameters( const FModioInitializeOptions& Options, const TMap& ExtendedParameters); /** - * @brief Retrieves the raw underlying value from a FModioModID. FModioModIDs are intended as opaque types, so use - * with care. - * @param In - the FModioModID to retrieve the value for + * @brief Retrieves the raw underlying value from an `ModioModID`. `ModioModID`s are intended as opaque types, so + * use with care. + * @param In The `ModioModID` to retrieve the value for * @return The underlying value */ UFUNCTION(BlueprintPure, BlueprintCallable, Category = "mod.io|Utilities") static int64 GetRawValueFromModID(const FModioModID& In); /** - * @brief Compares two ModioModIDs, returning true if equal + * @brief Compares two mod IDs to check whether they're equal */ UFUNCTION(BlueprintPure, BlueprintCallable, Category = "mod.io|Utilities", meta = (CompactNodeTitle = "==", Keywords = "== equal", DisplayName = "ModioModID == ModioModID")) static bool EqualTo(const FModioModID& A, const FModioModID& B); /** - * @brief Compares two ModioModIDs, returning true if not equal + * @brief Compares two mod IDs to check whether they're not equal */ UFUNCTION(BlueprintPure, BlueprintCallable, Category = "mod.io|Utilities", meta = (CompactNodeTitle = "!=", Keywords = "!= not equal", DisplayName = "ModioModID != ModioModID")) diff --git a/Source/Modio/Public/Libraries/ModioCreateModLibrary.h b/Source/Modio/Public/Libraries/ModioCreateModLibrary.h index 38b6eb96..f7567f3a 100644 --- a/Source/Modio/Public/Libraries/ModioCreateModLibrary.h +++ b/Source/Modio/Public/Libraries/ModioCreateModLibrary.h @@ -29,7 +29,7 @@ class MODIO_API UModioCreateModLibrary : public UBlueprintFunctionLibrary public: UFUNCTION(BlueprintCallable, Category = "mod.io|Create Mod Params", meta = (DeprecatedProperty, - DeprecationMessage = "Deprecated as of 2023.6 release. Please use the <> instead.")) + DeprecationMessage = "Deprecated as of 2023.6 release. Please use the `SetInitialVisibility` instead.")) static void SetInitialVisibility_DEPRECATED(UPARAM(ref) FModioCreateModParams& In, bool InitialVisibility); UFUNCTION(BlueprintCallable, Category = "mod.io|Create Mod Params") diff --git a/Source/Modio/Public/Libraries/ModioEditModLibrary.h b/Source/Modio/Public/Libraries/ModioEditModLibrary.h index 428e48a2..23e8b6c5 100644 --- a/Source/Modio/Public/Libraries/ModioEditModLibrary.h +++ b/Source/Modio/Public/Libraries/ModioEditModLibrary.h @@ -13,8 +13,8 @@ #include "CoreMinimal.h" #include "Kismet/BlueprintFunctionLibrary.h" #include "Types/ModioCommonTypes.h" -#include "Types/ModioModInfo.h" #include "Types/ModioEditModParams.h" +#include "Types/ModioModInfo.h" #include "ModioEditModLibrary.generated.h" @@ -36,8 +36,7 @@ class MODIO_API UModioEditModLibrary : public UBlueprintFunctionLibrary static void SetNamePath(UPARAM(ref) FModioEditModParams& In, FString NamePath); UFUNCTION(BlueprintCallable, Category = "mod.io|Edit Mod Params", - meta = (DeprecatedProperty, - DeprecationMessage = "Deprecated as of 2023.6 release. Please use the <> instead.")) + meta = (DeprecatedProperty, DeprecationMessage = "Deprecated as of 2023.6 release. Please use `SetVisibility` with `EModioObjectVisibilityFlags Visibility` instead.")) static void SetVisibility_DEPRECATED(UPARAM(ref) FModioEditModParams& In, bool Visibility); UFUNCTION(BlueprintCallable, Category = "mod.io|Edit Mod Params") diff --git a/Source/Modio/Public/Libraries/ModioErrorCodeLibrary.h b/Source/Modio/Public/Libraries/ModioErrorCodeLibrary.h index 8cbde3fd..b5f7dab1 100644 --- a/Source/Modio/Public/Libraries/ModioErrorCodeLibrary.h +++ b/Source/Modio/Public/Libraries/ModioErrorCodeLibrary.h @@ -22,17 +22,15 @@ class MODIO_API UModioErrorCodeLibrary : public UBlueprintFunctionLibrary GENERATED_BODY() public: /** - * Checks if a error code contains a error - * @param Error - - * @return true if the error code is a error + * @brief Checks if an error code contains a error + * @return true if the error code is an error */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities|Error", meta = (CompactNodeTitle = "Error", BlueprintAutocast)) static bool IsError(const FModioErrorCode& Error); /** - * Checks if a error code contains a error - * @param Error - - * @return true if the error code is a error + * @brief Checks if an error code contains a error + * @return true if the error code is an error */ UFUNCTION(BlueprintCallable, Category = "mod.io|Utilities|Error", meta = (DisplayName = "IsError", ExpandBoolAsExecs = "ReturnValue")) @@ -40,23 +38,21 @@ class MODIO_API UModioErrorCodeLibrary : public UBlueprintFunctionLibrary /** - * Get underlying error code - * @param Error - - * @return 0 if there is no error + * @brief Get underlying error code for an FModioErrorCode. + * @return The underlying error code. 0 represents no error. */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities|Error") static int32 GetValue(const FModioErrorCode& Error); /** - * Get the textual representation of the error - * @param Error - - * @return + * @brief Get the textual representation of the error + * @return An `FString` message describing the error */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities|Error") static FString GetMessage(const FModioErrorCode& Error); /** - * Helper method to reconstruct a mod.io error passed via code that cannot reference mod.io types + * @brief Helper method to reconstruct a mod.io error passed via code that cannot reference mod.io types * @param Value The numeric value of the code * @param Category The category ID (populated by native code) */ diff --git a/Source/Modio/Public/Libraries/ModioErrorConditionLibrary.h b/Source/Modio/Public/Libraries/ModioErrorConditionLibrary.h index e87aee38..c425ac48 100644 --- a/Source/Modio/Public/Libraries/ModioErrorConditionLibrary.h +++ b/Source/Modio/Public/Libraries/ModioErrorConditionLibrary.h @@ -26,7 +26,7 @@ class MODIO_API UModioErrorConditionLibrary : public UBlueprintFunctionLibrary * @brief Checks if the passed-in ErrorCode matches the specified error condition * @param ErrorCode The code to check * @param Condition The error condition to check against - * @return true if the code matches the condition + * @return True if the code matches the condition */ UFUNCTION(BlueprintCallable, Category = "mod.io|Error Handling") static bool ErrorCodeMatches(FModioErrorCode ErrorCode, EModioErrorCondition Condition); diff --git a/Source/Modio/Public/Libraries/ModioImageLibrary.h b/Source/Modio/Public/Libraries/ModioImageLibrary.h index fe545b91..c7643fec 100644 --- a/Source/Modio/Public/Libraries/ModioImageLibrary.h +++ b/Source/Modio/Public/Libraries/ModioImageLibrary.h @@ -11,12 +11,11 @@ #pragma once #include "Kismet/BlueprintFunctionLibrary.h" #include "Types/ModioCommonTypes.h" -#include "Types/ModioImageWrapper.h" #include "Types/ModioImageState.h" +#include "Types/ModioImageWrapper.h" #include "ModioImageLibrary.generated.h" - DECLARE_DYNAMIC_DELEGATE_OneParam(FOnLoadImageDelegate, class UTexture2DDynamic*, Texture); UCLASS() @@ -39,14 +38,14 @@ class MODIO_API UModioImageLibrary : public UBlueprintFunctionLibrary /** * If a logo size is EModioLogoSize::Original, then the size of the Logo returned, * else, the thumbnail size a logo is returned - * @param Logo If null and EModioLogoSize::Original is passed, then (0, 0) is returned + * @param Logo If null and `EModioLogoSize::Original` is passed, then (0, 0) is returned * @param LogoSize The size of the logo we want to return * * @return Dimensions of the logo if displayed in a 1:1 pixel ratio */ UFUNCTION(BlueprintPure, Category = "mod.io|Image") static FVector2D GetLogoSize(class UTexture* Logo, EModioLogoSize LogoSize); - + /** * If a avatar size is EModioAvatarSize::Original, then the size of the Logo returned, * else, the thumbnail size a avatar is returned diff --git a/Source/Modio/Public/Libraries/ModioModCollectionLibrary.h b/Source/Modio/Public/Libraries/ModioModCollectionLibrary.h index 8b83324e..7f68daf5 100644 --- a/Source/Modio/Public/Libraries/ModioModCollectionLibrary.h +++ b/Source/Modio/Public/Libraries/ModioModCollectionLibrary.h @@ -34,9 +34,8 @@ class MODIO_API UModioModCollectionLibrary : public UBlueprintFunctionLibrary static const FModioModInfo& GetModProfile(const FModioModCollectionEntry& Entry); /** - * @return Path to the mod's installation folder on disk - * NOTE: If the mod is not yet installed this path may not yet exist. Check - * xref:GetModState[] before trying to load files in this location + * @return Path to the mod's installation folder on disk. If the mod is not yet installed this path may not yet exist. Check + * `GetModState` before trying to load files in this location **/ UFUNCTION(BlueprintPure, Category = "mod.io|Mods") static const FString GetPath(const FModioModCollectionEntry& Entry); diff --git a/Source/Modio/Public/Libraries/ModioModProgressInfoLibrary.h b/Source/Modio/Public/Libraries/ModioModProgressInfoLibrary.h index c73de779..8c13427c 100644 --- a/Source/Modio/Public/Libraries/ModioModProgressInfoLibrary.h +++ b/Source/Modio/Public/Libraries/ModioModProgressInfoLibrary.h @@ -27,7 +27,7 @@ class MODIO_API UModioModProgressInfoLibrary : public UBlueprintFunctionLibrary public: /** * @brief Returns a EModioModProgressState indicating which state the mod operation is in - * @param Info the progress struct to query + * @param Info The progress struct to query */ UFUNCTION(BlueprintPure, Category = "ModProgressInfo") static EModioModProgressState GetCurrentState(const FModioModProgressInfo& Info) @@ -52,7 +52,7 @@ class MODIO_API UModioModProgressInfoLibrary : public UBlueprintFunctionLibrary * @brief Retrieves the total amount of progress required for the specified state. * @param Info the progress struct to query * @param State which state to query total progress for - * @return Modio::FileSize for total progress in bytes + * @return `Modio::FileSize` for total progress in bytes */ UFUNCTION(BlueprintPure, Category = "ModProgressInfo") static FModioUnsigned64 GetTotalProgress(const FModioModProgressInfo& Info, EModioModProgressState State) diff --git a/Source/Modio/Public/Libraries/ModioSDKLibrary.h b/Source/Modio/Public/Libraries/ModioSDKLibrary.h index 34020e65..203b528c 100644 --- a/Source/Modio/Public/Libraries/ModioSDKLibrary.h +++ b/Source/Modio/Public/Libraries/ModioSDKLibrary.h @@ -11,6 +11,7 @@ #pragma once #include "Kismet/BlueprintFunctionLibrary.h" +#include "Types/ModioAuthenticationParams.h" #include "Types/ModioCommonTypes.h" #include "Types/ModioInitializeOptions.h" @@ -22,61 +23,61 @@ class UModioSDKLibrary : public UBlueprintFunctionLibrary GENERATED_BODY() public: /** - * Get the game id specified in the mod.io project settings + * @brief Get the game id specified in the mod.io project settings */ UFUNCTION(BlueprintPure, category = "mod.io|Utilities") static MODIO_API FModioGameID GetProjectGameId(); /** - * Get the api key specified in the mod.io project settings + * @brief Get the api key specified in the mod.io project settings */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities") static MODIO_API FModioApiKey GetProjectApiKey(); /** - * Get the environment specified in the mod.io project settings + * @brief Get the environment specified in the mod.io project settings */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities") static MODIO_API EModioEnvironment GetProjectEnvironment(); /** - * Get the options needed to initialize the mod.io SDK specified in the project settings + * @brief Get the options needed to initialize the mod.io SDK specified in the project settings * @deprecated Use GetProjectInitializeOptionsForSessionId() instead */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities", meta = (DeprecatedFunction, DeprecationMessage = "Use GetProjectInitializeOptionsForSessionId() instead")) static MODIO_API FModioInitializeOptions GetProjectInitializeOptions(); /** - * Get the options needed to initialize the mod.io SDK specified in the project settings - * @param The LocalSessionIdentifier option to initialize project with + * @brief Get the options needed to initialize the mod.io SDK specified in the project settings + * @param SessionId The LocalSessionIdentifier option to initialize project with */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities") static MODIO_API FModioInitializeOptions GetProjectInitializeOptionsForSessionId(const FString SessionId); /** - * Does a basic validation if the email address supplied has a valid form - * @return true if the email address has a valid format + * @brief Does a basic validation if the email address supplied has a valid form + * @return True if the email address has a valid format */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities") static MODIO_API bool IsValidEmailAddressFormat(const FString& String); /** - * Checks if the string has the same format as the mod.io security code - * @return true if the security code has a valid format + * @brief Checks if the string has the same format as the mod.io security code + * @return True if the security code has a valid format */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities") static MODIO_API bool IsValidSecurityCodeFormat(const FString& String); /** - * Get desired file size unit based on the size of the file + * @brief Get desired file size unit based on the size of the file * @return the desired file size unit */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities") static MODIO_API EFileSizeUnit GetDesiredFileSizeUnit(int64 FileSize); /** - * Converts a filesize to a human readable string with the appropriate unit + * @brief Converts a filesize to a human readable string with the appropriate unit * * @param FileSize - Filesize in bytes * @param MaxDecimals - Maximum amount of decimals to display of the filesize @@ -107,8 +108,10 @@ class UModioSDKLibrary : public UBlueprintFunctionLibrary Category = "mod.io|Utilities|Text") static MODIO_API FText Conv_Int64ToText(int64 Value, bool bAlwaysSign = false, bool bUseGrouping = true, int32 MinimumIntegralDigits = 1, int32 MaximumIntegralDigits = 324); - - /** @brief Dividend/Divisor and return the floating point result with no checks **/ + /** + * @brief Calculates percentage using two `int64` params + * @return The floating point result of Dividend/Divisor with no checks + */ UFUNCTION(BlueprintPure, meta = (DisplayName = "Get Percent (integer64/integer64)", CompactNodeTitle = "Percent", Keywords = "/ % percent pct"), @@ -116,7 +119,7 @@ class UModioSDKLibrary : public UBlueprintFunctionLibrary static MODIO_API float Pct_Int64Int64(int64 Dividend, int64 Divisor); - /** + /** * @brief Sets the correct decimals depending on the file size or speed * * @param Input value of bytes per second as FText @@ -141,8 +144,8 @@ class UModioSDKLibrary : public UBlueprintFunctionLibrary static MODIO_API FString GetShortenedNumberAsString(int64 Number); /** - * @brief Get Session Id for Windows for initialization of the SDK - * return empty string if you are not on Windows + * @brief Get Session Id for Windows for initialization of the SDK + * @return The windows session id, or an empty string if you are not on Windows */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities") static MODIO_API FString GetDefaultSessionIdWindows(); @@ -162,4 +165,11 @@ class UModioSDKLibrary : public UBlueprintFunctionLibrary */ UFUNCTION(BlueprintPure, Category = "mod.io|Utilities") static MODIO_API EModioLanguage GetLanguageCodeFromString(FString LanguageCode); + + /** + * @brief Get the purchase category to pass to the store overlay UI for a given portal. + * @return string of ID for the purchase category for a given Portal in use, returns empty if the given key was not found/not defined + */ + UFUNCTION(BlueprintPure, category = "mod.io|Utilities") + static MODIO_API FString GetMonetizationPurchaseCategory(EModioPortal Portal); }; diff --git a/Source/Modio/Public/Loc/ModioEnumLocalizationHelpers.h b/Source/Modio/Public/Loc/ModioEnumLocalizationHelpers.h index 52150bbb..635f9def 100644 --- a/Source/Modio/Public/Loc/ModioEnumLocalizationHelpers.h +++ b/Source/Modio/Public/Loc/ModioEnumLocalizationHelpers.h @@ -46,10 +46,11 @@ class MODIO_API UModioUIEnumLocalizationLibrary : public UBlueprintFunctionLibra // I'd prefer to do this such that you can pass in the enum value rather than the name but don't really want to deal // with a whole custom K2Node to properly support generic enum values - /// @brief Returns the string table FText for a given enum value's FName - *Only works with enums registered via - /// ModioUI::RegisterEnumAsLocalizable* - /// @param EnumName The Name from a given enum value - /// @return localized Text for the specified enum value, or dummy FText if not found + /** + * @brief Returns the string table `FText` for a given enum value's `FName`. Only works with enums registered via `ModioUI::RegisterEnumAsLocalizable`. + * @param EnumName The Name from a given enum value + * @return localized Text for the specified enum value, or dummy FText if not found + */ UFUNCTION(BlueprintCallable, BlueprintPure, Category = "mod.io|UI|Localization") static FText GetLocalizedTextForEnumByName(const FName& EnumName) { @@ -69,11 +70,13 @@ class MODIO_API UModioUIEnumLocalizationLibrary : public UBlueprintFunctionLibra /** * Converts a Unsigned64 filesize to a human readable string with the appropriate unit * - * @param FileSize - Filesize in bytes - * @param MaxDecimals - Maximum amount of decimals to display of the filesize - * @param Unit - If Largest, then it tries to display the size in the largest unit that will have a integral + * @param FileSize Filesize in bytes + * @param MinDecimals Minimum number of decimals to display for the filesize + * @param MaxDecimals Maximum number of decimals to display for the filesize + * @param Unit If `Largest`, it tries to display the size in the largest unit that will have a integral * part > 0, else it displays the filesize in the specified unit - * @return A text formatted from your specifications + * @param bIncludeUnitName Whether or not to include the unit name + * @return An `FText` formatted with your specifications */ UFUNCTION(BlueprintPure, Category = "mod.io|Text", meta = (DisplayName = "FileSizeToText (Unsigned64)", CompactNodeTitle = "FileSizeUnsigned64>Text")) @@ -124,9 +127,11 @@ class MODIO_API UModioUILocalizationLibrary : public UBlueprintFunctionLibrary { GENERATED_BODY() public: - /// @brief Returns the string table FText for a given string key - /// @param StringKey The key to look up in the table - /// @return localized Text for the specified key, or StringKey if not found + /** + * @brief Returns the string table `FText` for a given string key + * @param StringKey The key to look up in the table + * @return Localized Text for the specified key, or `StringKey` if not found + */ UFUNCTION(BlueprintCallable, BlueprintPure, Category = "mod.io|UI|Localization") static FText GetLocalizedTextFromDefaultTableByKey(const FString& StringKey) { diff --git a/Source/Modio/Public/ModioSettings.h b/Source/Modio/Public/ModioSettings.h index 56f069d2..7e68960b 100644 --- a/Source/Modio/Public/ModioSettings.h +++ b/Source/Modio/Public/ModioSettings.h @@ -10,6 +10,7 @@ #pragma once +#include "Types/ModioAuthenticationParams.h" #include "Types/ModioCommonTypes.h" #include "Types/ModioFilterParams.h" #include "Types/ModioInitializeOptions.h" @@ -84,6 +85,13 @@ class MODIO_API UModioSettings : public UObject UPROPERTY(EditDefaultsOnly, config, Category = "Feature Flags", meta=(DisplayName="Enable Mod Downvoting in UI")) bool bEnableModDownvoteFeature; + /** + * If you intend to sell Premium Currency through the relevant platform store, this Map is used to invoke the platform-native storefront on the given platform(s). + * See the monetization documentation for more info. + */ + UPROPERTY(EditDefaultsOnly, config, Category = "Monetization", meta = (EditCondition="bEnableMonetizationFeature", EditConditionHides="true")) + TMap PlatformIdentifiers; + private: #if WITH_EDITOR diff --git a/Source/Modio/Public/ModioSubsystem.h b/Source/Modio/Public/ModioSubsystem.h index 84f316fd..efa40601 100644 --- a/Source/Modio/Public/ModioSubsystem.h +++ b/Source/Modio/Public/ModioSubsystem.h @@ -127,6 +127,8 @@ DECLARE_DYNAMIC_DELEGATE_TwoParams(FOnGetUserDelegationTokenDelegate, FModioErro DECLARE_DYNAMIC_DELEGATE_TwoParams(FOnRefreshUserEntitlementsDelegate, FModioErrorCode, ErrorCode, FModioOptionalEntitlementConsumptionStatusList, Entitlements); +DECLARE_DYNAMIC_DELEGATE_TwoParams(FOnPlatformCheckoutDelegate, bool, bSuccess, FString, Message); + class UModioSubsystem; class FModioBackgroundThread : public FRunnable @@ -154,9 +156,9 @@ class FModioBackgroundThread : public FRunnable }; /** - * @brief Thin wrapper around the mod.io SDK. This mostly wraps all the functions available in modio/ModioSDK.h that's - * the public header of the mod.io SDK. It converts mod.io SDK types to a more unreal friendly types and caches some - * expensive operations so that conversions don't need to be done multiple times + * @brief `ModioSubsystem` is a thin wrapper around the mod.io SDK, wrapping all the functions available in the SDK's + * public header `modio/ModioSDK.h`. This subsystem also converts mod.io SDK types to unreal-friendly types and caches + * some expensive operations. */ UCLASS(MinimalAPI) class UModioSubsystem : public UEngineSubsystem @@ -164,8 +166,15 @@ class UModioSubsystem : public UEngineSubsystem GENERATED_BODY() protected: + TUniquePtr BackgroundThread; + friend class UModioUISubsystem; + EModioPortal GetCurrentPortal() const + { + return CachedInitializeOptions.PortalInUse; + } + #if WITH_EDITOR /// @brief Internal method used for emitting a warning during PIE if the Plugin was initialized during that PIE /// session @@ -176,7 +185,14 @@ class UModioSubsystem : public UEngineSubsystem private: bool bUseBackgroundThread = false; + /** + * @brief Internal cache of the options used to initialize this session. Specific properties + * are exposed as protected methods and used by friend classes for additional functionality. + */ + FModioInitializeOptions CachedInitializeOptions; + public: + FOnUserProfileUpdatedDelegate OnUserProfileUpdatedDelegate; /** Begin USubsystem interface */ @@ -185,7 +201,7 @@ class UModioSubsystem : public UEngineSubsystem virtual bool ShouldCreateSubsystem(UObject* Outer) const override; /** End USubsystem interface */ - /// @brief Bind to a function that returns the appropriate EModioLanguage for the game. The Terms of Use will be + /// @brief Bind to a function that returns the appropriate `EModioLanguage` for the game. The Terms of Use will be /// displayed in this language during authentication. FGetCurrentLanguageDelegate GetCurrentLanguage; @@ -196,33 +212,32 @@ class UModioSubsystem : public UEngineSubsystem MODIO_API void KillBackgroundThread(); /** - * @brief Initializes the SDK for the given user. Loads the state of mods installed on the system as well as the + * @brief Initializes the SDK for the given user. Loads the state of all mods installed on the system as well as the * set of mods the specified user has installed on this device * @param InitializeOptions Parameters to the function packed as a struct where all members needs to be initialized * for the call to succeed - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory FilesystemError|Couldn't create the user data or common data folders - * @errorcategory ConfigurationError|InitializeOptions contains an invalid value - inspect ec.value() to determine - * what was incorrect - * @errorcategory SDKAlreadyInitialized|SDK already initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `FilesystemError`|Couldn't create the user data or common data folders + * @errorcategory `ConfigurationError`|`InitializeOptions` contains an invalid value. Inspect `ec.value()` to + * determine what was incorrect + * @errorcategory `SDKAlreadyInitialized`|SDK already initialized */ MODIO_API void InitializeAsync(const FModioInitializeOptions& InitializeOptions, FOnErrorOnlyDelegateFast OnInitComplete); /** - * @brief Sets the global logging level - messages with a log level below the specified value will not be displayed + * @brief Sets the global logging level. * @note This is a transient function, and won't be stored to disk. So at the next engine start, the log level will - * be restored to the one in your FModioRuntimeSettings - * @param UnrealLogLevel Value indicating which priority of messages should be included in the log output + * be restored to the one in your `ModioRuntimeSettings` + * @param UnrealLogLevel Determines which messages to include in the log output. Messages with a log level below the + * specified value will not be displayed. */ UFUNCTION(BlueprintCallable, Category = "mod.io") MODIO_API void SetLogLevel(EModioLogLevel UnrealLogLevel); /** - * @brief Runs any pending SDK work on the calling thread, including invoking any callbacks passed to asynchronous + * @brief Runs any pending mod.io work on the calling thread and invokes any callbacks passed to asynchronous * operations. - * NOTE: This should be called while xref:InitializeAsync[] and xref:ShutdownAsync[] are running, - as they both utilize the internal event loop for functionality. */ UFUNCTION(BlueprintCallable, Category = "mod.io") MODIO_API void RunPendingHandlers(); @@ -230,17 +245,17 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Cancels any running internal operations, frees SDK resources, and invokes any pending callbacks with an *OperationCanceled error category. This function will NOT block while the deinitialization occurs. - * @param OnShutdownComplete Callback invoked when the plugin is shut down and calling <> is no - * longer required + * @param OnShutdownComplete Callback invoked when the plugin is shut down and calling + *[`RunPendingHandlers`](#run-pending-handlers) is no longer required * @requires initialized-sdk - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `SDKNotInitialized`|SDK not initialized **/ MODIO_API void ShutdownAsync(FOnErrorOnlyDelegateFast OnShutdownComplete); /** * @brief If the last request to the mod.io servers returned a validation failure, this function returns extended * information describing the fields that failed validation. - * @return Collection of FModioValidationError objects, or empty collection if there were no validation failures + * @return Collection of `ModioValidationError` objects, or empty collection if there were no validation failures * @requires initialized-sdk */ UFUNCTION(BlueprintCallable, Category = "mod.io") @@ -255,29 +270,29 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ MODIO_API void SubscribeToModAsync(FModioModID ModToSubscribeTo, bool IncludeDependencies, FOnErrorOnlyDelegateFast OnSubscribeComplete); /** * @brief Sends a request to the mod.io server to remove the specified mod from the user's list of subscriptions. - * If no other local users are subscribed to the specified mod this function will also mark the mod for + * If no other local users are subscribed to the specified mod, this function will also mark the mod for * uninstallation by the SDK. * @param ModToUnsubscribeFrom Mod ID of the mod requiring unsubscription. - * @param OnUnsubscribeComplete Callback invoked when the unsubscription request is completed. + * @param OnUnsubscribeComplete Callback invoked when the unsubscription request is complete. * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ MODIO_API void UnsubscribeFromModAsync(FModioModID ModToUnsubscribeFrom, FOnErrorOnlyDelegateFast OnUnsubscribeComplete); @@ -290,41 +305,44 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user **/ MODIO_API void FetchExternalUpdatesAsync(FOnErrorOnlyDelegateFast OnFetchDone); /** * @brief Retrieve a list of updates between the users local mod state, and the server-side state. This allows you - *to identify which mods will be modified by the next call to <> in order to perform any - *content management (such as unloading files) that might be required. + * to identify which mods will be modified by the next call to + * [`FetchExternalUpdatesAsync`](#fetchexternalupdatesasync) in order to perform any content management (such as + * unloading files) that might be required. * @param OnPreviewDone Callback invoked when the external state has been retrieved. It contains a dictionary with - *ModID as keys and changes as values. Empty when there are no differences between local and the mod.io API service + * ModID as keys and change maps as values. Empty when there are no differences between local and the mod.io API + * service * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user **/ MODIO_API void PreviewExternalUpdatesAsync(FOnPreviewExternalUpdatesDelegateFast OnPreviewDone); /** * @brief Enables the automatic management of installed mods on the system based on the user's subscriptions. - * @param Callback This callback handler will be invoked with a ModManagementEvent for each mod operation performed - * by the SDK - * @return FModioErrorCode indicating if mod management was enabled successfully - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory ModManagementAlreadyEnabled|Mod management was already enabled. The mod management callback has + * @param Callback This callback handler will be invoked with a + * [`ModioModManagementEvent`](#modiomodmanagementevent) for each mod operation performed by the SDK + * @return An error code indicating success or failure of enabling mod management. Note that this is independent of + * error codes for mod management events. Inspect the `Callback` for information on each mod management event. + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementAlreadyEnabled`|Mod management was already enabled. The mod management callback has * not been changed. */ MODIO_API FModioErrorCode EnableModManagement(FOnModManagementDelegateFast Callback); /** * @brief Disables automatic installation or uninstallation of mods based on the user's subscriptions. Allows - * currently processing installation to complete; will cancel any pending operations when called. + * currently processing installation to complete. Will cancel any pending operations when called. **/ UFUNCTION(BlueprintCallable, Category = "mod.io|Mod Management") MODIO_API void DisableModManagement(); @@ -344,14 +362,14 @@ class UModioSubsystem : public UEngineSubsystem * successful or if the mod was already being processed * @requires initialized-sdk * @requires authenticated-user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid or not present in the list of pending operations + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid or not present in the list of pending operations */ UFUNCTION(BlueprintCallable, Category = "mod.io|Mod Management") MODIO_API FModioErrorCode PrioritizeTransferForMod(FModioModID ModToPrioritize); /** * @brief Provides progress information for a mod installation or update operation if one is currently in progress. - * @return Optional ModProgressInfo object containing information regarding the progress of the installation + * @return Optional `ModioModProgressInfo` object containing information regarding the progress of the installation * operation. */ MODIO_API TOptional QueryCurrentModUpdate(); @@ -359,73 +377,73 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Fetches the local view of the user's subscribed mods, including mods that are subscribed but not * yet installed - * @return TMap providing information about the subscribed + * @return `TMap` providing information about the subscribed * mods */ UFUNCTION(BlueprintPure, Category = "mod.io|User") MODIO_API const TMap& QueryUserSubscriptions(); /** - * @brief Fetches the subset of the user's subscribed mods that are installed and therefore ready for loading + * @brief Fetches the subset of the user's subscribed mods that are installed and ready for loading * @param bIncludeOutdatedMods Include subscribed mods that are installed but have an updated version on the server * that has not yet been installed - * @return TMap providing information about the subscribed + * @return `TMap` providing information about the subscribed * mods **/ UFUNCTION(BlueprintPure, Category = "mod.io|User") MODIO_API TMap QueryUserInstallations(bool bIncludeOutdatedMods); /** - * @brief Fetches the currently authenticated mod.io user profile if there is one associated with the current - * platform user - * @return Optional FModioUser object containing profile information + * @brief Fetches the currently authenticated mod.io user profile if there is one. + * @return Optional [`ModioUser`](#modiouser) object containing profile information **/ MODIO_API TOptional QueryUserProfile(); /** - * @brief Fetches all mods installed on the system such that a consuming application can present the information in - * a UI in order to free up space by uninstalling mods - * @return TMap using Mod IDs as keys and ModCollectionEntry objects providing information about mods installed - * on the system regardless of which user installed them + * @brief Fetches all mods installed on the system, including those installed by other users. + * @return A `TMap` of all mods installed on the system, including those + * installed by other users. */ UFUNCTION(BlueprintPure, Category = "mod.io|Mod Management") MODIO_API TMap QuerySystemInstallations(); /** - * @brief Provides a list of mods for the current game, that match the parameters specified in the filter - * @param Filter FModioFilterParams object containing any filters that should be applied to the query - * @param Callback Callback invoked with a status code and an optional ModInfoList providing mod profiles + * @brief Provides a list of mods for the current game that match the parameters specified in the filter + * @param Filter [`ModioFilterParams`](#modiofilterparams) object containing any filters that should be applied to + * the query + * @param Callback Callback invoked with a status code and an optional `ModioModInfoList` providing mod profiles * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized */ MODIO_API void ListAllModsAsync(const FModioFilterParams& Filter, FOnListAllModsDelegateFast Callback); /** - * @brief Provides a list of games for the current user, that match the parameters specified in the filter - * @param Filter FModioFilterParams object containing any filters that should be applied to the query - * @param Callback Callback invoked with a status code and an optional ModInfoList providing mod profiles + * @brief Provides a list of games for the current user that match the parameters specified in the filter + * @param Filter [`ModioFilterParams`](#modiofilterparams) object containing any filters that should be applied to + * the query + * @param Callback Callback invoked with a status code and an optional `ModioModInfoList` providing mod profiles * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `SDKNotInitialized`|SDK not initialized */ MODIO_API void ListUserGamesAsync(const FModioFilterParams& Filter, FOnListUserGamesDelegateFast Callback); /** * @brief Fetches detailed information about the specified game * @param GameID Game ID of the game data to fetch - * @param Callback Callback providing a status code and an optional FModioGameInfo object with the game's extended + * @param Callback Callback providing a status code and an optional `ModioGameInfo` object with the game's extended * information * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory EntityNotFoundError|Specified game does not exist or was deleted - * @errorcategory InvalidArgsError|The supplied game ID is invalid + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `EntityNotFoundError`|Specified game does not exist or was deleted + * @errorcategory `InvalidArgsError`|The supplied game ID is invalid **/ MODIO_API void GetGameInfoAsync(FModioGameID GameID, FOnGetGameInfoDelegateFast Callback); @@ -433,14 +451,14 @@ class UModioSubsystem : public UEngineSubsystem * @brief Fetches detailed information about the specified mod, including description and file metadata for the * most recent release * @param ModId Mod ID of the mod to fetch data - * @param Callback Callback providing a status code and an optional FModioModInfo object with the mod's extended + * @param Callback Callback providing a status code and an optional `ModioModInfo` object with the mod's extended * information * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ MODIO_API void GetModInfoAsync(FModioModID ModId, FOnGetModInfoDelegateFast Callback); @@ -452,11 +470,11 @@ class UModioSubsystem : public UEngineSubsystem * downloaded image * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod media does not exist or was deleted - * @errorcategory InsufficientSpace|Not enough space for the file - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod media does not exist or was deleted + * @errorcategory `InsufficientSpace`|Not enough space for the file + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ MODIO_API void GetModMediaAsync(FModioModID ModId, EModioLogoSize LogoSize, FOnGetMediaDelegateFast Callback); @@ -473,11 +491,11 @@ class UModioSubsystem : public UEngineSubsystem * @param Callback Callback containing a status code and an Optional containing a path to the image file on disk * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod media does not exist or was deleted - * @errorcategory InsufficientSpace|Not enough space for the file - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod media does not exist or was deleted + * @errorcategory `InsufficientSpace`|Not enough space for the file + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ MODIO_API void GetModMediaAsync(FModioModID ModId, EModioGallerySize GallerySize, int32 Index, FOnGetMediaDelegateFast Callback); @@ -496,11 +514,11 @@ class UModioSubsystem : public UEngineSubsystem * downloaded image * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod media does not exist or was deleted - * @errorcategory InsufficientSpace|Not enough space for the file - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod media does not exist or was deleted + * @errorcategory `InsufficientSpace`|Not enough space for the file + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ MODIO_API void GetModMediaAsync(FModioModID ModId, EModioAvatarSize AvatarSize, FOnGetMediaDelegateFast Callback); @@ -510,36 +528,35 @@ class UModioSubsystem : public UEngineSubsystem public: /** * @brief Fetches the available tags used on mods for the current game. These tags can them be used in conjunction - * with the FilterParams passed to ListAllMods + * with the [`ModioFilterParams`](#modiofilterparams) passed to [`ListAllModsAsync`](#listallmodsasync) * Will be cached when first received - * @param Callback Callback providing a status code and an optional ModTagOptions object containing the available - * tags + * @param Callback Callback providing a status code and an optional `ModioModTagOptions` object containing the + *available tags * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized **/ MODIO_API void GetModTagOptionsAsync(FOnGetModTagOptionsDelegateFast Callback); /** * @brief For a given Mod ID, fetches a list of any mods that the creator has marked as dependencies * @param ModID The mod to retrieve dependencies for - * @param Recursive Include child dependencies in a recursive manner. \r\n NOTE: Recursion supports a maximum depth - *of 5. - * @param Callback Callback providing a status code and an optional ModTagOptions object containing the available - * tags + * @param Recursive Include child dependencies in a recursive manner to a maximum depth of 5. + * @param Callback Callback providing a status code and an optional `ModioModTagOptions` object containing the + *available tags * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid * @experimental **/ MODIO_API void GetModDependenciesAsync(FModioModID ModID, bool Recursive, FOnGetModDependenciesDelegateFast Callback); /** - * @brief Gets a new mod handle for use with SubmitNewModAsync. + * @brief Gets a new mod handle for use with [`SubmitNewModAsync`](#submitnewmodasync). * @returns New handle * @experimental */ @@ -547,29 +564,29 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Requests the creation of a new mod on the server with the specified parameters - * @param Handle The ModCreationHandle for this submission. Once this method invokes your callback indicating - * success, the ModCreationHandle is invalid for the rest of the session and you should request a new one for the - * next submission attempt. + * @param Handle The `ModioModCreationHandle` for this submission. Once this method invokes your callback indicating + * success, the `ModioModCreationHandle` is invalid for the rest of the session and you should request a new one for + * the next submission attempt. * @param Params Information about the new mod to create - * @param Callback Callback providing a status code and an optional FModioModID for the newly created mod + * @param Callback Callback providing a status code and an optional `ModioModID` for the newly created mod * @experimental * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory InvalidArgsError|Some fields in Params did not pass validation - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidArgsError`|Some fields in `Params` did not pass validation + * @errorcategory `UserNotAuthenticatedError`|No authenticated user */ MODIO_API void SubmitNewModAsync(FModioModCreationHandle Handle, FModioCreateModParams Params, FOnSubmitNewModDelegateFast Callback); /** * @brief Queues the upload of a new modfile release for the specified mod using the submitted parameters. This - * function takes an FModioCreateModFileParams object to specify the path to the root folder of the new modfile. + * function takes an `ModioCreateModFileParams` object to specify the path to the root folder of the new modfile. * The plugin will compress the folder's contents into a .zip archive and queue the result for upload. When the - * upload completes, a Mod Management Event will be triggered. Note the plugin is also responsible for - * decompressing the archive upon its installation at a later point in time. + * upload completes, a mod management event will be triggered. Note the plugin is also responsible for decompressing + * the archive upon its installation at a later point in time. * @param Mod The ID of the mod you are submitting a file for * @param Params Information about the mod file being created, including the root path of the directory that will * be archived @@ -577,40 +594,40 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid */ MODIO_API void SubmitNewModFileForMod(FModioModID Mod, FModioCreateModFileParams Params); /** - * @brief Edits the parameters of a mod, by updating any fields set in the Params object to match the passed-in - * values. Fields left empty on the Params object will not be updated. + * @brief Edits the parameters of a mod by updating any fields set in the `Params` object to match the passed-in + * values. Fields left empty on the `Params` object will not be updated. * @param Mod The ID of the mod you wish to edit * @param Params Descriptor containing the fields that should be altered. - * @param Callback The callback invoked when the changes have been submitted, containing an optional updated - * ModInfo object if the edits were performed successfully + * @param Callback The callback invoked when the changes have been submitted containing an optional updated + * `ModioModInfo` object if the edits were performed successfully * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory InvalidArgsError|Some fields in Params did not pass validation - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidArgsError`|Some fields in `Params` did not pass validation + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid */ MODIO_API void SubmitModChangesAsync(FModioModID Mod, FModioEditModParams Params, FOnGetModInfoDelegateFast Callback); /** * @brief Begins email authentication for the current session by requesting a one-time code be sent to the - * specified email address if it is associated with a mod.io account + * specified email address. * @param EmailAddress The email address to send the code to * @param Callback Callback providing a status code indicating the outcome of the request * @requires initialized-sdk * @requires no-rate-limiting * @requires no-authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized **/ MODIO_API void RequestEmailAuthCodeAsync(const FModioEmailAddress& EmailAddress, FOnErrorOnlyDelegateFast Callback); @@ -622,10 +639,11 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires no-rate-limiting * @requires no-authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserAlreadyAuthenticatedError|Authenticated user already signed-in. De-authenticate the current - * user with ClearUserDataAsync(), and re-initialize the SDK by calling ShutdownAsync() then InitializeAsync(). + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserAlreadyAuthenticatedError`|Current user is already authenticated. De-authenticate the current + * user with [`ClearUserDataAsync`](#clearuserdataasync), and re-initialize the SDK by calling + * [`ShutdownAsync`](#shutdownasync) followed by [`InitializeAsync`](#initializeasync). **/ MODIO_API void AuthenticateUserEmailAsync(const FModioEmailAuthCode& AuthenticationCode, FOnErrorOnlyDelegateFast Callback); @@ -638,40 +656,41 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires no-rate-limiting * @requires no-authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory ConfigurationError|The SDK's configuration is not valid - * @errorcategory InvalidArgsError|The arguments passed to the function have failed validation - * @errorcategory UserTermsOfUseError|The user has not yet accepted the mod.io Terms of Use - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserAlreadyAuthenticatedError|Authenticated user already signed-in. De-authenticate the current - * user with ClearUserDataAsync(), and re-initialize the SDK by calling ShutdownAsync() then InitializeAsync(). + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `ConfigurationError`|The SDK's configuration is not valid + * @errorcategory `InvalidArgsError`|The arguments passed to the function have failed validation + * @errorcategory `UserTermsOfUseError`|The user has not yet accepted the mod.io Terms of Use + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserAlreadyAuthenticatedError`|Current user is already authenticated. De-authenticate the current + * user with [`ClearUserDataAsync`](#clearuserdataasync), and re-initialize the SDK by calling + * [`ShutdownAsync`](#shutdownasync) followed by [`InitializeAsync`](#initializeasync). **/ MODIO_API void AuthenticateUserExternalAsync(const FModioAuthenticationParams& User, EModioAuthenticationProvider Provider, FOnErrorOnlyDelegateFast Callback); /** * @docpublic - * @brief Queries the server to verify the state of the currently authenticated user if there is one present. An - * empty ErrorCode is passed to the callback to indicate successful verification, i.e. the mod.io server was - * contactable and the user's authentication remains valid. - * @param Callback Callback invoked with the results of the verification process + * @brief Queries the server to verify the state of the currently authenticated user if there is one present. + * @param Callback Callback invoked with the results of the verification process. An empty `ModioErrorCode` + * indicates successful verification i.e. the mod.io server was contactable and the user's authentication remains + * valid. * @requires initialized-sdk * @requires no-rate-limiting * @requires authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user */ MODIO_API void VerifyUserAuthenticationAsync(FOnErrorOnlyDelegateFast Callback); /** * @docpublic * @brief This function retrieves the information required for a game to display the mod.io terms of use to a - *player who wishes to create a mod.io account + * player who wishes to create a mod.io account * @param Callback Callback invoked with the terms of use data once retrieved from the server * @requires initialized-sdk - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized **/ MODIO_API void GetTermsOfUseAsync(FOnGetTermsOfUseDelegateFast Callback); @@ -691,8 +710,7 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief De-authenticates the current mod.io user for the current session, and clears all user-specific data - * stored on the current device. Any subscribed mods that are installed but do not have other local users - * subscribed will be uninstalled + * stored on the current device. Any mods that do not have other local users subscribed will be uninstalled * @param Callback Callback providing a status code indicating the outcome of clearing the user data. Error codes * returned by this function are informative only - it will always succeed. **/ @@ -710,9 +728,9 @@ class UModioSubsystem : public UEngineSubsystem * @docpublic * @brief Purchases a mod for the current player * @param ModID ID of the mod to purchase - * @param ExpectedPrice The price the user is expected to pay for the mod, generally ModInfo.Price. This ensures - *that there is consistency between the displayed price and the price in the backend. If there is a mismatch, the - *purchase will fail. + * @param ExpectedPrice The price the user is expected to pay for the mod, generally + * [`ModioModInfo.Price`](#modiomodinfo) . This ensures that there is consistency between the displayed price and + *the price in the backend. If there is a mismatch, the purchase will fail. * @param Callback Callback invoked with purchase information once the purchase is completed. * @requires initialized-sdk * @requires authenticated-user @@ -723,7 +741,7 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic * @brief Gets the users current wallet balance. This will also create a wallet for a user if one does not exist. - * You should ensure this is called prior to calling PurchaseModAsync + * You should ensure this is called prior to calling [`PurchaseModAsync`](#purchasemodasync) * purchase will fail. * @param Callback Callback invoked with the users wallet balance * @requires initialized-sdk @@ -735,7 +753,7 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic * @brief Fetches the user's purchases. This populates a runtime cache of purchase information - * that can be accessed using QueryUserPurchasedMods. + * that can be accessed using [`QueryUserPurchasedMods`](#query-user-purchased-mods). * @param Callback Callback invoked once the call has been completed. * @requires initialized-sdk * @requires authenticated-user @@ -745,7 +763,7 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic - * @brief Get a UserDelegationToken that can be used for S2S service calls. + * @brief Get a user delegation token that can be used for S2S service calls. * @param Callback Callback invoked once the call has been completed. * @requires initialized-sdk * @requires authenticated-user @@ -755,8 +773,9 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic - * @brief Returns the users purchased mods. You must call FetchUserPurchasesAsync to populate the cache/. - * @return Map of FModioModID to FModioModInfo for all purchases a user has made. + * @brief Returns the user's purchased mods. [`FetchUserPurchasesAsync`](#fetchuserpurchasesasync) must be called + *first to populate the cache. + * @return A `TMap` of all purchases a user has made. * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting @@ -778,189 +797,190 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid */ MODIO_API void SubmitModRatingAsync(FModioModID Mod, EModioRating Rating, FOnErrorOnlyDelegateFast Callback); /** * @brief Sends a content report to mod.io. When using this function, please inform your users that if they provide - * their contact name or details in the Report parameter, that those may be shared with the person responsible for + * their contact name or details in the `Report` parameter, this data may be shared with the person responsible for * the content being reported. For more information on what data in a report will be shared with whom, please see - * link:https://mod.io/report/widget[our website's report form] for more information. + * [our website's report form](https://mod.io/report). * @param Report Information about the content being reported and a description of the report. * @param Callback Callback providing a status code to indicate successful submission of the report. * @requires initialized-sdk - * @errorcategory NetworkError|Couldn't Connect to mod.io servers - * @errorcategory InvalidArgsError|Required information in the report did not pass validation - * @errorcategory InvalidArgsError|The mod ID, game ID, or user ID supplied to Report is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `InvalidArgsError`|Required information in the report did not pass validation + * @errorcategory `InvalidArgsError`|The mod ID, game ID, or user ID supplied to `Report` is invalid */ MODIO_API void ReportContentAsync(FModioReportParams Report, FOnErrorOnlyDelegateFast Callback); /** Get our image cache */ struct FModioImageCache& GetImageCache() const; - /* - @brief Archives a mod. This mod will no longer be able to be viewed or retrieved via the SDK, but it will still - exist should you choose to restore it at a later date. Archiving is restricted to team managers and administrators - only. Note that restoration and permanent deletion of a mod is possible only via web interface. - @param Mod The mod to be archived. - @param Callback Callback providing a status code indicating success or failure of archiving the mod. - @requires authenticated-user - @requires initialized-sdk - @requires no-rate-limiting - @errorcategory InsufficientPermissions|The authenticated user does not have permission to archive this mod. This - action is restricted to team managers and administrators only. - @errorcategory NetworkError|Couldn't connect to mod.io servers - @errorcategory SDKNotInitialized|SDK not initialized - @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - @errorcategory InvalidArgsError|The supplied mod ID is invalid - */ + /** + * @brief Archives a mod. This mod will no longer be able to be viewed or retrieved via the SDK, but it will still + * exist should you choose to restore it at a later date. Archiving is restricted to team managers and + * administrators only. Note that restoration and permanent deletion of a mod is possible only via web interface. + * @param Mod The mod to be archived. + * @param Callback Callback providing a status code indicating success or failure of archiving the mod. + * @requires authenticated-user + * @requires initialized-sdk + * @requires no-rate-limiting + * @errorcategory `InsufficientPermissions`|The authenticated user does not have permission to archive this mod. + * This action is restricted to team managers and administrators only. + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid + */ MODIO_API void ArchiveModAsync(FModioModID Mod, FOnErrorOnlyDelegateFast Callback); /** * @brief Forcibly uninstalls a mod from the system. This can be used when the host application requires additional * space for other mods. The current user must not be subscribed to the mod to force uninstall. To remove a mod the - * current user is subscribed to, first use xref:UnsubscribeFromModAsync[]. If the mod does not uninstall (due to a - * different user on the same system remaining subscribed), ForceUninstallModAsync can be called next. + * current user is subscribed to, first use [`UnsubscribeFromModAsync`](#unsubscribefrommodasync). If the mod does + * not uninstall (due to a different user on the same system remaining subscribed), `ForceUninstallModAsync` can be + * called next. * @param ModToRemove The mod ID to force uninstall. * @param Callback Callback indicating success or failure of the uninstallation. - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory ApiErrorRefSuccess|User is still subscribed to the specified mod - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `ApiErrorRefSuccess`|User is still subscribed to the specified mod + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid */ MODIO_API void ForceUninstallModAsync(FModioModID ModToRemove, FOnErrorOnlyDelegateFast Callback); /** * @docpublic - * @brief Mute a user. This will prevent mod.io from returning mods authored by the muted user. + * @brief Mute a user. This will prevent mod.io servers from returning mods authored by the muted user. * when performing searches. * @requires authenticated-user * @requires initialized-sdk * @param UserID ID of the User to mute * @param Callback Callback providing a status code indicating success or failure of muting the user. - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied user ID is invalid + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied user ID is invalid */ MODIO_API void MuteUserAsync(FModioUserID UserID, FOnErrorOnlyDelegateFast Callback); /* * @docpublic - * @brief Unmute a user. This will allow mod.io to mods authored by the previously muted user. - * when performing searches. + * @brief Unmute a user. This allows mod.io to display mods authored by the now unmuted user when performing + * searches. * @requires authenticated-user * @requires initialized-sdk - * @param UserID ID of the User to unmute + * @param UserID ID of the user to unmute * @param Callback Callback providing a status code indicating success or failure of unmuting the user. - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied user ID is invalid + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied user ID is invalid */ MODIO_API void UnmuteUserAsync(FModioUserID UserID, FOnErrorOnlyDelegateFast Callback); /* * @docpublic * @brief List all the users that have been muted by the current user. - * @param Callback Callback providing a status code indicating success or failure of the operation, and an Optional - * containing a list of muted users if successful. + * @param Callback Callback providing a status code indicating success or failure of the operation, and an + * `Optional` containing a list of muted users if successful. * @requires authenticated-user * @requires initialized-sdk - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user */ MODIO_API void GetMutedUsersAsync(FOnMuteUsersDelegateFast Callback); /** - * @brief Provides a list of mods that the user has submitted, or is a team member for, for the current game, + * @brief Provides a list of mods that the user has submitted or is a team member of for the current game, * applying the parameters specified in the filter. * @param Filter Filter to apply to listing the user's created mods. * @param Callback Callback invoked when the call succeeds, or when an error occurs. * @requires authenticated-user * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory RateLimited|Too many frequent calls to the API. Wait some time and try again. + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `RateLimited`|Too many frequent calls to the API. Wait some time and try again. */ MODIO_API void ListUserCreatedModsAsync(FModioFilterParams Filter, FOnListUserCreatedModsDelegateFast Callback); /** * @brief Returns the default mod installation directory for this game and platform, ignoring overrides and without * requiring the SDK to be initialized. - * @param GameID The FModioGameID of the game we're fetching the default mod installation directory for. + * @param GameID The `ModioGameID` of the game that we're fetching the default mod installation directory for. * @return The default mod installation directory for the specified game on the current platform */ static MODIO_API FString GetDefaultModInstallationDirectory(FModioGameID GameID); /** * @brief Returns the default mod installation directory for this game and platform, ignoring overrides and without - * requiring the SDK to be initialized. This overload takes an int64 for the GameID param, allowing other modules - * to execute this function via a delegate without depending on this module and the FModioGameID type. - * @param GameID An int64 representing the GameID of the game we're fetching the default mod installation directory - * for. + * requiring the SDK to be initialized. This overload takes an `int64` for the `GameID` param, allowing other + * modules to execute this function via a delegate without depending on this module and the `ModioGameID` type. + * @param GameID An `int64` representing the `GameID` of the game that we're fetching the default mod installation + * directory for. * @return The default mod installation directory for the specified game on the current platform */ static MODIO_API FString GetDefaultModInstallationDirectory(int64 GameID); /** - * @brief Install every temp mod given in the param if they are not already subbed. - * @param TArray of ModID to install as temp mod - * @return Error code indicating the status of the TempModSet. Will be empty if it was successful + * @brief Install every temp mod specified by `ModIds` if not already installed. + * @param ModIds TArray of `ModioModID`s to install as temp mods + * @return Error code indicating the status of the `TempModSet`. Will be empty if it was successful * @requires initialized-sdk - * @requires modmanagement enabled - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory ModManagementDisabled|ModManagement not enable + * @requires mod-management-enabled + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementDisabled`|Mod management not enabled */ MODIO_API FModioErrorCode InitTempModSet(TArray ModIds); /** - * @brief Add mods to a Temp Mod Set, install every temp mod given in the param if they are not already subbed. - * @param TArray of ModID to install as temp mod - * @return Error code indicating the status of the TempModSet. Will be empty if it was successful + * @brief Add mods to a temp mod set. Every temp mod specified by `ModIds` will be installed if not already + * installed. + * @param ModIds TArray of `ModioModID`s to install as temp mods + * @return Error code indicating the status of the `TempModSet`. Will be empty if it was successful * @requires initialized-sdk - * @requires modmanagement enabled - * @requires TempModSet initialized - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory ModManagementDisabled|ModManagement not enable - * @errorcategory TempModSetNotInitialize| TempModSet not initialized + * @requires mod-management-enabled + * @requires temp-mod-set-initialized + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementDisabled`|Mod management not enabled + * @errorcategory `TempModSetNotInitialized`|`TempModSet` not initialized */ MODIO_API FModioErrorCode AddToTempModSet(TArray ModIds); /** - * @brief Remove mods from a Temp Mod Set, uninstall every temp mod given in the param if they are not already - * subbed. - * @param TArray of ModID to install as temp mod - * @return Error code indicating the status of the TempModSet. Will be empty if it was successful + * @brief Remove mods from a temp mod set. Every temp mod specified by `ModIds` will be uninstalled unless the + * user is already subscribed. + * @param ModIds TArray of `ModioModID`s to remove as temp mods + * @return Error code indicating the status of the `TempModSet`. Will be empty if it was successful * @requires initialized-sdk - * @requires modmanagement enabled - * @requires TempModSet initialized - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory ModManagementDisabled|ModManagement not enable - * @errorcategory TempModSetNotInitialize| TempModSet not initialized + * @requires mod-management-enabled + * @requires temp-mod-set-initialized + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementDisabled`|Mod management not enabled + * @errorcategory `TempModSetNotInitialized`|`TempModSet` not initialized */ MODIO_API FModioErrorCode RemoveFromTempModSet(TArray ModIds); /** - * @brief Uninstall every temp mod. - * @return Error code indicating the status of the TempModSet. Will be empty if it was successful + * @brief Uninstall every temp mod unless the user is subscribed. + * @return Error code indicating the status of the `TempModSet`. Will be empty if it was successful * @requires initialized-sdk - * @requires modmanagement enabled - * @requires TempModSet initialized - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory ModManagementDisabled|ModManagement not enable - * @errorcategory TempModSetNotInitialize| TempModSet not initialized + * @requires mod-management-enabled + * @requires temp-mod-set-initialized + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementDisabled`|Mod management not enabled + * @errorcategory `TempModSetNotInitialized`|`TempModSet` not initialized */ MODIO_API FModioErrorCode CloseTempModSet(); /** - * @brief Query every System and Temp Mod in TempModSet - * @return TMap using Mod IDs as keys and ModCollectionEntry objects providing information about mods - * in TempModSet + * @brief Query every system and temp mod in `TempModSet` + * @return `TMap` providing information about mods in `TempModSet` */ MODIO_API TMap QueryTempModSet(); @@ -979,16 +999,16 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic - * @brief Start a Metrics play session - * @param Params MetricsServiceParams struct containing information of what and how to start a metrics + * @brief Start a metrics play session + * @param Params `ModioMetricsSessionParams` struct containing information of what and how to start a metrics * session * @param Callback Callback providing an error code indicating success or failure of the session start operation - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory RateLimited|Too many frequent calls to the API. Wait some time and try again. - * @errorcategory InvalidUser|No authenticated user - * @errorcategory SessionNotInitialized|Metrics session has not yet been initialized - * @errorcategory SessionIsActive|Metrics session is currently active and running - * @errorcategory BadParameter|One or more values in the Metric Session Parameters are invalid + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `RateLimited`|Too many frequent calls to the API. Wait some time and try again. + * @errorcategory `InvalidUser`|No authenticated user + * @errorcategory `SessionNotInitialized`|Metrics session has not yet been initialized + * @errorcategory `SessionIsActive`|Metrics session is currently active and running + * @errorcategory `BadParameter`|One or more values in the `ModioMetricsSessionParams` are invalid * @premiumfeature Metrics */ MODIO_API void MetricsSessionStartAsync(const FModioMetricsSessionParams& Params, @@ -999,12 +1019,11 @@ class UModioSubsystem : public UEngineSubsystem * @brief Sends a single heartbeat to the mod.io server to indicate a session is still active * @param Callback Callback providing an error code indicating success or failure of the session heartbeat * operation - * - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory InvalidUser|No authenticated user - * @errorcategory SessionNotInitialized|Metrics session has not yet been initialized - * @errorcategory SessionIsNotActive|Metrics session is not currently running. Call MetricsSessionStartAsync before - * attempting to sending a heartbeat. + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidUser`|No authenticated user + * @errorcategory `SessionNotInitialized`|Metrics session has not yet been initialized + * @errorcategory `SessionIsNotActive`|Metrics session is not currently running. Call + * [`MetricsSessionStartAsync`](#metricssessionstartasync) before attempting to sending a heartbeat. * @premiumfeature Metrics */ MODIO_API void MetricsSessionSendHeartbeatOnceAsync(FOnErrorOnlyDelegateFast Callback); @@ -1016,11 +1035,11 @@ class UModioSubsystem : public UEngineSubsystem * @param Callback Callback providing an error code indicating success or failure of the session heartbeat * operation * - * @errorcategory GenericError::SDKNotInitialized|SDK not initialized - * @errorcategory UserDataError::InvalidUser|No authenticated user - * @errorcategory MetricsError::SessionNotInitialized|Metrics session has not yet been initialized - * @errorcategory MetricsError::SessionIsNotActive|Metrics session is not currently running. - * Call MetricsSessionStartAsync before attempting to sending a heartbeat. + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidUser`|No authenticated user + * @errorcategory `SessionNotInitialized`|Metrics session has not yet been initialized + * @errorcategory `SessionIsNotActive`|Metrics session is not currently running. + * Call [`MetricsSessionStartAsync`](#metricssessionstartasync) before attempting to sending a heartbeat. * @premiumfeature Metrics */ MODIO_API void MetricsSessionSendHeartbeatAtIntervalAsync(FModioUnsigned64 IntervalSeconds, @@ -1028,15 +1047,15 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic - * @brief Ends a Metrics play session + * @brief Ends a metrics play session * @param Callback Callback providing an error code indicating success or failure of the session end operation * - * @errorcategory GenericError::SDKNotInitialized|SDK not initialized - * @errorcategory HttpError::RateLimited|Too many frequent calls to the API. Wait some time and try again. - * @errorcategory UserDataError::InvalidUser|No authenticated user - * @errorcategory MetricsError::SessionNotInitialized|Metrics session has not yet been initialized - * @errorcategory MetricsError::SessionIsNotActive|Metrics session is not currently running. - * Call MetricsSessionStartAsync before attempting to end a session. + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `RateLimited`|Too many frequent calls to the API. Wait some time and try again. + * @errorcategory `InvalidUser`|No authenticated user + * @errorcategory `SessionNotInitialized`|Metrics session has not yet been initialized + * @errorcategory `SessionIsNotActive`|Metrics session is not currently running. + * Call [`MetricsSessionStartAsync`](#metricssessionstartasync) before attempting to end a session. * @premiumfeature Metrics */ MODIO_API void MetricsSessionEndAsync(FOnErrorOnlyDelegateFast Callback); @@ -1077,11 +1096,11 @@ class UModioSubsystem : public UEngineSubsystem * @param InitializeOptions Parameters to the function packed as a struct where all members needs to be initialized * for the call to succeed * @param OnInitComplete Callback which will be invoked with the result of initialization - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory FilesystemError|Couldn't create the user data or common data folders - * @errorcategory ConfigurationError|InitializeOptions contains an invalid value - inspect ec.value() to determine - * what was incorrect - * @errorcategory SDKAlreadyInitialized|SDK already initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `FilesystemError`|Couldn't create the user data or common data folders + * @errorcategory `ConfigurationError`|InitializeOptions contains an invalid value - inspect `ec.value()` to + * determine what was incorrect + * @errorcategory `SDKAlreadyInitialized`|SDK already initialized */ UFUNCTION(BlueprintCallable, DisplayName = "InitializeAsync", Category = "mod.io") MODIO_API void K2_InitializeAsync(const FModioInitializeOptions& InitializeOptions, @@ -1096,11 +1115,11 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ UFUNCTION(BlueprintCallable, DisplayName = "SubscribeToModAsync", Category = "mod.io|Mods") MODIO_API void K2_SubscribeToModAsync(FModioModID ModToSubscribeTo, bool IncludeDependencies, @@ -1108,9 +1127,9 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Cancels any running internal operations, frees SDK resources, and invokes any pending callbacks with - * an OperationCanceled error category. This function will NOT block while the deinitialization occurs. - * @param OnShutdownComplete Callback invoked when the plugin is shut down and calling <> is no - * longer required + * an `OperationCanceled` error category. This function will NOT block while the deinitialization occurs. + * @param OnShutdownComplete Callback invoked when the plugin is shut down and calling + *[`RunPendingHandlers`](#run-pending-handlers) is no longer required **/ UFUNCTION(BlueprintCallable, DisplayName = "ShutdownAsync", Category = "mod.io") MODIO_API void K2_ShutdownAsync(FOnErrorOnlyDelegate OnShutdownComplete); @@ -1124,11 +1143,11 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ UFUNCTION(BlueprintCallable, DisplayName = "UnsubscribeFromModAsync", Category = "mod.io|Mods") MODIO_API void K2_UnsubscribeFromModAsync(FModioModID ModToUnsubscribeFrom, @@ -1145,25 +1164,32 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Retrieve a list of updates between the users local mod state, and the server-side state. This allows - * you to identify which mods will be modified by the next call to <> in order to - * perform any content management (such as unloading files) that might be required. + * you to identify which mods will be modified by the next call to + *[`FetchExternalUpdatesAsync`](#fetchexternalupdatesasync) in order to perform any content management (such as + *unloading files) that might be required. * @param OnPreviewDone Callback invoked when the external state has been retrieved. It contains a dictionary with - *ModID as keys and changes as values. Empty when there are no differences between local and the mod.io API service + * ModID as keys and change maps as values. Empty when there are no differences between local and the mod.io API + *service **/ UFUNCTION(BlueprintCallable, DisplayName = "PreviewExternalUpdatesAsync", Category = "mod.io|Mod Management") MODIO_API void K2_PreviewExternalUpdatesAsync(FOnPreviewExternalUpdatesDelegate OnPreviewDone); /** * @brief Enables the automatic management of installed mods on the system based on the user's subscriptions. - * @param Callback This callback handler will be invoked with a ModManagementEvent for each mod operation performed - * by the SDK + * @param Callback This callback handler will be invoked with a + * [`ModioModManagementEvent`](#modiomodmanagementevent) for each mod operation performed by the SDK + * @return An error code indicating success or failure of enabling mod management. Note that this is independent of + * error codes for mod management events. Inspect the `Callback` for information on each mod management event. + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementAlreadyEnabled`|Mod management was already enabled. The mod management callback has + * not been changed. */ UFUNCTION(BlueprintCallable, DisplayName = "EnableModManagement", Category = "mod.io|Mod Management") MODIO_API FModioErrorCode K2_EnableModManagement(FOnModManagementDelegate Callback); /** * @brief Provides progress information for a mod installation or update operation if one is currently in progress. - * @return Optional ModProgressInfo object containing information regarding the progress of the installation + * @return `ModioOptionalModProgressInfo` object containing information regarding the progress of the installation * operation. */ UFUNCTION(BlueprintPure, DisplayName = "QueryCurrentModUpdate", Category = "mod.io|Mod Management") @@ -1172,48 +1198,50 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Forcibly uninstalls a mod from the system. This can be used when the host application requires additional * space for other mods. The current user must not be subscribed to the mod to force uninstall. To remove a mod the - * current user is subscribed to, first use xref:UnsubscribeFromModAsync[]. If the mod does not uninstall (due to a - * different user on the same system remaining subscribed), ForceUninstallModAsync can be called next. + * current user is subscribed to, first use [`UnsubscribeFromModAsync`](#unsubscribefrommodasync). If the mod does + * not uninstall (due to a different user on the same system remaining subscribed), `ForceUninstallModAsync` can be + * called next. * @param ModToRemove The mod ID to force uninstall. * @param Callback Callback invoked indicating success or failure of the uninstallation. - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user * @errorcategory ApiErrorRefSuccess|User is still subscribed to the specified mod - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid */ UFUNCTION(BlueprintCallable, DisplayName = "ForceUninstallModAsync", Category = "mod.io|Mod Management") MODIO_API void K2_ForceUninstallModAsync(FModioModID ModToRemove, FOnErrorOnlyDelegate Callback); /** - * @brief Fetches the currently authenticated mod.io user profile if there is one associated with the current - * platform user - * @return FModioOptionalUser object containing profile information + * @brief Fetches the currently authenticated mod.io user profile if there is one + * @return `ModioOptionalUser` object containing profile information **/ UFUNCTION(BlueprintPure, DisplayName = "QueryUserProfile", Category = "mod.io|User") MODIO_API FModioOptionalUser K2_QueryUserProfile(); /** - * @brief Provides a list of mods for the current game, that match the parameters specified in the filter - * @param Filter FModioFilterParams object containing any filters that should be applied to the query - * @param Callback Callback invoked with a status code and an optional ModInfoList providing mod profiles + * @brief Provides a list of mods for the current game that match the parameters specified in the filter + * @param Filter [`ModioFilterParams`](#modiofilterparams) object containing any filters that should be applied to + * the query + * @param Callback Callback invoked with a status code and an optional `ModioModInfoList` providing mod profiles * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized */ UFUNCTION(BlueprintCallable, DisplayName = "ListAllModsAsync", Category = "mod.io|Mods") MODIO_API void K2_ListAllModsAsync(const FModioFilterParams& Filter, FOnListAllModsDelegate Callback); /** - * @brief Provides a list of games for the current user, that match the parameters specified in the filter - * @param Filter FModioFilterParams object containing any filters that should be applied to the query - * @param Callback Callback invoked with a status code and an optional GameInfoList providing game profiles + * @brief Provides a list of games for the current user that match the parameters specified in the filter + * @param Filter [`ModioFilterParams`](#modiofilterparams) object containing any filters that should be applied to + * the query + * @param Callback Callback invoked with a status code and an optional `ModioGameInfoList` providing game profiles * @requires initialized-sdk * @requires no-rate-limiting * @requires authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `SDKNotInitialized`|SDK not initialized */ UFUNCTION(BlueprintCallable, DisplayName = "ListUserGamesAsync", Category = "mod.io|Mods") MODIO_API void K2_ListUserGamesAsync(const FModioFilterParams& Filter, FOnListUserGamesDelegate Callback); @@ -1221,14 +1249,14 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Fetches detailed information about the specified game * @param GameID Game ID of the game data to fetch - * @param Callback Callback providing a status code and an optional FModioGameInfo object with the game's extended + * @param Callback Callback providing a status code and an optional `ModioGameInfo` object with the game's extended * information * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory EntityNotFoundError|Specified game does not exist - * @errorcategory InvalidArgsError|The supplied game ID is invalid + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `EntityNotFoundError`|Specified game does not exist + * @errorcategory `InvalidArgsError`|The supplied game ID is invalid **/ UFUNCTION(BlueprintCallable, DisplayName = "GetGameInfoAsync", Category = "mod.io|Game") MODIO_API void K2_GetGameInfoAsync(FModioGameID GameID, FOnGetGameInfoDelegate Callback); @@ -1237,14 +1265,14 @@ class UModioSubsystem : public UEngineSubsystem * @brief Fetches detailed information about the specified mod, including description and file metadata for the * most recent release * @param ModId Mod ID of the mod to fetch data - * @param Callback Callback providing a status code and an optional FModioModInfo object with the mod's + * @param Callback Callback providing a status code and an optional `ModioModInfo` object with the mod's * extended information * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ UFUNCTION(BlueprintCallable, DisplayName = "GetModInfoAsync", Category = "mod.io|Mods") MODIO_API void K2_GetModInfoAsync(FModioModID ModId, FOnGetModInfoDelegate Callback); @@ -1257,11 +1285,11 @@ class UModioSubsystem : public UEngineSubsystem * downloaded image * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod media does not exist or was deleted - * @errorcategory InsufficientSpace|Not enough space for the file - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod media does not exist or was deleted + * @errorcategory `InsufficientSpace`|Not enough space for the file + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ UFUNCTION(BlueprintCallable, DisplayName = "GetModMediaAsync (Logo)", Category = "mod.io|Mods") MODIO_API void K2_GetModMediaLogoAsync(FModioModID ModId, EModioLogoSize LogoSize, FOnGetMediaDelegate Callback); @@ -1275,11 +1303,11 @@ class UModioSubsystem : public UEngineSubsystem * @param Callback Callback containing a status code and an Optional containing a path to the image file on disk * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod media does not exist or was deleted - * @errorcategory InsufficientSpace|Not enough space for the file - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod media does not exist or was deleted + * @errorcategory `InsufficientSpace`|Not enough space for the file + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ UFUNCTION(BlueprintCallable, DisplayName = "GetModMediaAsync (Gallery Image)", Category = "mod.io|Mods") MODIO_API void K2_GetModMediaGalleryImageAsync(FModioModID ModId, EModioGallerySize GallerySize, int32 Index, @@ -1294,11 +1322,11 @@ class UModioSubsystem : public UEngineSubsystem * downloaded image * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod media does not exist or was deleted - * @errorcategory InsufficientSpace|Not enough space for the file - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod media does not exist or was deleted + * @errorcategory `InsufficientSpace`|Not enough space for the file + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid **/ UFUNCTION(BlueprintCallable, DisplayName = "GetModMediaAsync (Avatar)", Category = "mod.io|Mods") MODIO_API void K2_GetModMediaAvatarAsync(FModioModID ModId, EModioAvatarSize AvatarSize, @@ -1306,14 +1334,14 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Fetches the available tags used on mods for the current game. These tags can them be used in conjunction - * with the FilterParams passed to ListAllMods + * with the FilterParams passed to [`ListAllModsAsync`](#listallmodsasync) * Will be cached when first received - * @param Callback Callback providing a status code and an optional ModTagOptions object containing the available - * tags + * @param Callback Callback providing a status code and an optional `ModioModTagOptions` object containing the + *available tags * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized **/ UFUNCTION(BlueprintCallable, DisplayName = "GetModTagOptionsAsync", Category = "mod.io|Mods") MODIO_API void K2_GetModTagOptionsAsync(FOnGetModTagOptionsDelegate Callback); @@ -1322,13 +1350,13 @@ class UModioSubsystem : public UEngineSubsystem * @brief For a given Mod ID, fetches a list of any mods that the creator has marked as dependencies * @param ModID The mod to retrieve dependencies for * @param Recursive Fetches dependencies recursively up to a depth of 5 - * @param Callback Callback providing a status code and an optional ModTagOptions object containing the available - * tags + * @param Callback Callback providing a status code and an optional `ModioModTagOptions` object containing the + *available tags * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid * @experimental **/ UFUNCTION(BlueprintCallable, DisplayName = "GetModDependenciesAsync", Category = "mod.io|Mods") @@ -1337,16 +1365,17 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Begins email authentication for the current session by requesting a one-time code be sent to the - * specified email address if it is associated with a mod.io account + * specified email address. * @param EmailAddress The email address to send the code to * @param Callback Callback providing a status code indicating the outcome of the request * @requires initialized-sdk * @requires no-rate-limiting * @requires no-authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserAlreadyAuthenticatedError|Authenticated user already signed-in. De-authenticate the current - * user with ClearUserDataAsync(), and re-initialize the SDK by calling ShutdownAsync() then InitializeAsync(). + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserAlreadyAuthenticatedError`|Current user is already authenticated. De-authenticate the current + * user with [`ClearUserDataAsync`](#clearuserdataasync), and re-initialize the SDK by calling + * [`ShutdownAsync`](#shutdownasync) then [`InitializeAsync`](#initializeasync). **/ UFUNCTION(BlueprintCallable, DisplayName = "RequestEmailAuthCodeAsync", Category = "mod.io|Authentication") MODIO_API void K2_RequestEmailAuthCodeAsync(const FModioEmailAddress& EmailAddress, FOnErrorOnlyDelegate Callback); @@ -1359,10 +1388,11 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires no-rate-limiting * @requires no-authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserAlreadyAuthenticatedError|Authenticated user already signed-in. De-authenticate the current - * user with ClearUserDataAsync(), and re-initialize the SDK by calling ShutdownAsync() then InitializeAsync(). + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserAlreadyAuthenticatedError`|Current user is already authenticated. De-authenticate the current + * user with [`ClearUserDataAsync`](#clearuserdataasync), and re-initialize the SDK by calling + * [`ShutdownAsync`](#shutdownasync) then [`InitializeAsync`](#initializeasync). **/ UFUNCTION(BlueprintCallable, DisplayName = "AuthenticateUserEmailAsync", Category = "mod.io|Authentication") MODIO_API void K2_AuthenticateUserEmailAsync(const FModioEmailAuthCode& AuthenticationCode, @@ -1376,10 +1406,14 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires no-rate-limiting * @requires no-authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserAlreadyAuthenticatedError|Authenticated user already signed-in. De-authenticate the current - * user with ClearUserDataAsync(), and re-initialize the SDK by calling ShutdownAsync() then InitializeAsync(). + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `ConfigurationError`|The SDK's configuration is not valid + * @errorcategory `InvalidArgsError`|The arguments passed to the function have failed validation + * @errorcategory `UserTermsOfUseError`|The user has not yet accepted the mod.io Terms of Use + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserAlreadyAuthenticatedError`|Current user is already authenticated. De-authenticate the current + * user with [`ClearUserDataAsync`](#clearuserdataasync), and re-initialize the SDK by calling + * [`ShutdownAsync`](#shutdownasync) followed by [`InitializeAsync`](#initializeasync). **/ UFUNCTION(BlueprintCallable, DisplayName = "AuthenticateUserExternalAsync", Category = "mod.io|Authentication") MODIO_API void K2_AuthenticateUserExternalAsync(const FModioAuthenticationParams& User, @@ -1388,16 +1422,16 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic - * @brief Queries the server to verify the state of the currently authenticated user if there is one present. An - * empty ErrorCode is passed to the callback to indicate successful verification, i.e. the mod.io server was - * contactable and the user's authentication remains valid. - * @param Callback Callback invoked with the results of the verification process + * @brief Queries the server to verify the state of the currently authenticated user if there is one present + * @param Callback Callback invoked with the results of the verification process. An empty `ModioErrorCode` + * indicates successful verification i.e. the mod.io server was contactable and the user's authentication remains + * valid. * @requires initialized-sdk * @requires no-rate-limiting * @requires authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user */ UFUNCTION(BlueprintCallable, DisplayName = "VerifyUserAuthenticationAsync", Category = "mod.io|Authentication") MODIO_API void K2_VerifyUserAuthenticationAsync(FOnErrorOnlyDelegate Callback); @@ -1407,8 +1441,8 @@ class UModioSubsystem : public UEngineSubsystem *player who wishes to create a mod.io account * @param Callback Callback invoked with the terms of use data once retrieved from the server * @requires initialized-sdk - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized **/ UFUNCTION(BlueprintCallable, DisplayName = "GetTermsOfUseAsync", Category = "mod.io|Authentication") MODIO_API void K2_GetTermsOfUseAsync(FOnGetTermsOfUseDelegate Callback); @@ -1438,8 +1472,8 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires no-rate-limiting * @requires authenticated-user - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user **/ UFUNCTION(BlueprintCallable, DisplayName = "ClearUserDataAsync", Category = "mod.io|User") MODIO_API void K2_ClearUserDataAsync(FOnErrorOnlyDelegate Callback); @@ -1452,9 +1486,9 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires no-rate-limiting * @requires authenticated-user - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user **/ UFUNCTION(BlueprintCallable, DisplayName = "GetUserMediaAsync (Avatar)", Category = "mod.io|User") MODIO_API void K2_GetUserMediaAvatarAsync(EModioAvatarSize AvatarSize, FOnGetMediaDelegate Callback); @@ -1469,17 +1503,17 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid */ UFUNCTION(BlueprintCallable, DisplayName = "SubmitModRatingAsync", Category = "mod.io|Mods") MODIO_API void K2_SubmitModRatingAsync(FModioModID Mod, EModioRating Rating, FOnErrorOnlyDelegate Callback); /** - * @brief Gets a new mod handle for use with SubmitNewModAsync. + * @brief Gets a new mod handle for use with [`SubmitNewModAsync`](#submitnewmodasync). * @returns New handle * @experimental */ @@ -1488,28 +1522,30 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Requests the creation of a new mod on the server with the specified parameters - * @param Handle The ModCreationHandle for this submission. Once this method invokes your callback indicating - * success, the ModCreationHandle is invalid for the rest of the session and you should request a new one for the - * next submission attempt. + * @param Handle The `ModioModCreationHandle` for this submission. Once this method invokes your callback indicating + * success, the `ModioModCreationHandle` is invalid for the rest of the session. You should request a new one for + * the next submission attempt. * @param Params Information about the new mod to create - * @param Callback Callback providing a status code and an optional FModioModID for the newly created mod + * @param Callback Callback providing a status code and an optional `ModioModID` for the newly created mod * @experimental * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory InvalidArgsError|Some fields in Params did not pass validation - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidArgsError`|Some fields in `Params` did not pass validation + * @errorcategory `UserNotAuthenticatedError`|No authenticated user */ UFUNCTION(BlueprintCallable, DisplayName = "SubmitNewModAsync", Category = "mod.io|Mods|Submission") MODIO_API void K2_SubmitNewModAsync(FModioModCreationHandle Handle, FModioCreateModParams Params, FOnSubmitNewModDelegate Callback); /** - * @brief Queues the upload of a new mod file release for the specified mod, using the submitted parameters. The - * upload's progress can be tracked in the same way as downloads; when completed, a Mod Management Event will be - * triggered with the result code for the upload. + * @brief Queues the upload of a new modfile release for the specified mod using the submitted parameters. This + * function takes an `ModioCreateModFileParams` object to specify the path to the root folder of the new modfile. + * The plugin will compress the folder's contents into a .zip archive and queue the result for upload. When the + * upload completes, a mod management event will be triggered. Note the plugin is also responsible for decompressing + * the archive upon its installation at a later point in time. * @param Mod The ID of the mod you are submitting a file for * @param Params Information about the mod file being created, including the root path of the directory that will * be archived @@ -1517,27 +1553,27 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid */ UFUNCTION(BlueprintCallable, DisplayName = "SubmitNewModFileForMod", Category = "mod.io|Mods|Submission") MODIO_API void K2_SubmitNewModFileForMod(FModioModID Mod, FModioCreateModFileParams Params); /** - * @brief Edits the parameters of a mod, by updating any fields set in the Params object to match the passed-in - * values. Fields left empty on the Params object will not be updated. + * @brief Edits the parameters of a mod by updating any fields set in the `Params` object to match the passed-in + * values. Fields left empty on the `Params` object will not be updated. * @param Mod The ID of the mod you wish to edit * @param Params Descriptor containing the fields that should be altered. - * @param Callback The callback invoked when the changes have been submitted, containing an optional updated - * ModInfo object if the edits were performed successfully + * @param Callback The callback invoked when the changes have been submitted containing an optional updated + * `ModioModInfo` object if the edits were performed successfully * @requires initialized-sdk * @requires authenticated-user * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory InvalidArgsError|Some fields in Params did not pass validation - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidArgsError`|Some fields in `Params` did not pass validation + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid */ UFUNCTION(BlueprintCallable, DisplayName = "SubmitModChangesAsync", Category = "mod.io|Mods|Editing") MODIO_API void K2_SubmitModChangesAsync(FModioModID Mod, FModioEditModParams Params, @@ -1545,20 +1581,20 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Sends a content report to mod.io. When using this function, please inform your users that if they provide - * their contact name or details in the Report parameter, that those may be shared with the person responsible for + * their contact name or details in the `Report` parameter, this data may be shared with the person responsible for * the content being reported. For more information on what data in a report will be shared with whom, please see - * link:https://mod.io/report/widget[our website's report form] for more information. + * [our website's report form](https://mod.io/report). * @param Report Information about the content being reported and a description of the report. * @param Callback Callback providing a status code to indicate successful submission of the report. * @requires initialized-sdk - * @errorcategory NetworkError|Couldn't Connect to mod.io servers - * @errorcategory InvalidArgsError|Required information in the report did not pass validation - * @errorcategory InvalidArgsError|The mod ID, game ID, or user ID supplied to Report is invalid + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `InvalidArgsError`|Required information in the report did not pass validation + * @errorcategory `InvalidArgsError`|The mod ID, game ID, or user ID supplied to `Report` is invalid */ UFUNCTION(BlueprintCallable, DisplayName = "ReportContentAsync", Category = "mod.io") MODIO_API void K2_ReportContentAsync(FModioReportParams Report, FOnErrorOnlyDelegate Callback); - /* + /** * @brief Archives a mod. This mod will no longer be able to be viewed or retrieved via the SDK, but it will still * exist should you choose to restore it at a later date. Archiving is restricted to team managers and * administrators only. Note that restoration and permanent deletion of a mod is possible only via web interface. @@ -1567,12 +1603,12 @@ class UModioSubsystem : public UEngineSubsystem * @requires authenticated-user * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory InsufficientPermissions|The authenticated user does not have permission to archive this mod. This - * action is restricted to team managers and administrators only. - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory EntityNotFoundError|Specified mod does not exist or was deleted - * @errorcategory InvalidArgsError|The supplied mod ID is invalid + * @errorcategory `InsufficientPermissions`|The authenticated user does not have permission to archive this mod. + * This action is restricted to team managers and administrators only. + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `EntityNotFoundError`|Specified mod does not exist or was deleted + * @errorcategory `InvalidArgsError`|The supplied mod ID is invalid */ UFUNCTION(BlueprintCallable, DisplayName = "ArchiveModAsync", Category = "mod.io|Mods") MODIO_API void K2_ArchiveModAsync(FModioModID Mod, FOnErrorOnlyDelegate Callback); @@ -1585,24 +1621,24 @@ class UModioSubsystem : public UEngineSubsystem * @requires initialized-sdk * @param UserID ID of the User to mute * @param Callback Callback providing a status code indicating success or failure of muting the user. - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied user ID is invalid + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied user ID is invalid */ UFUNCTION(BlueprintCallable, DisplayName = "MuteUserAsync", Category = "mod.io|User") MODIO_API void K2_MuteUserAsync(FModioUserID UserID, FOnErrorOnlyDelegate Callback); /** * @docpublic - * @brief Unmute a user. This will allow mod.io to mods authored by the previously muted user. - * when performing searches. + * @brief Unmute a user. This allows mod.io to display mods authored by the now unmuted user when performing + * searches. * @requires authenticated-user * @requires initialized-sdk - * @param UserID ID of the User to unmute + * @param UserID ID of the user to unmute * @param Callback Callback providing a status code indicating success or failure of unmuting the user. - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user - * @errorcategory InvalidArgsError|The supplied user ID is invalid + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user + * @errorcategory `InvalidArgsError`|The supplied user ID is invalid */ UFUNCTION(BlueprintCallable, DisplayName = "UnmuteUserAsync", Category = "mod.io|User") MODIO_API void K2_UnmuteUserAsync(FModioUserID UserID, FOnErrorOnlyDelegate Callback); @@ -1610,27 +1646,27 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic * @brief List all the users that have been muted by the current user. - * @param Callback Callback providing a status code indicating success or failure of the operation, and an Optional + * @param Callback Callback providing a status code indicating success or failure of the operation, and an optional * containing a list of muted users if successful. * @requires authenticated-user * @requires initialized-sdk - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory UserNotAuthenticatedError|No authenticated user + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `UserNotAuthenticatedError`|No authenticated user */ UFUNCTION(BlueprintCallable, DisplayName = "GetMutedUsersAsync", Category = "mod.io|User") MODIO_API void K2_GetMutedUsersAsync(FOnMuteUsersDelegate Callback); /** - * @brief Provides a list of mods that the user has submitted, or is a team member for, for the current game, + * @brief Provides a list of mods that the user has submitted or is a team member of for the current game, * applying the parameters specified in the filter. - * @param Filter Filter to apply to listing the user's created mods. + * @param Filter Filter to apply when listing the user's created mods. * @param Callback Callback invoked when the call succeeds, or when an error occurs. * @requires authenticated-user * @requires initialized-sdk * @requires no-rate-limiting - * @errorcategory NetworkError|Couldn't connect to mod.io servers - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory RateLimited|Too many frequent calls to the API. Wait some time and try again. + * @errorcategory `NetworkError`|Couldn't connect to mod.io servers + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `RateLimited`|Too many frequent calls to the API. Wait some time and try again. */ UFUNCTION(BlueprintCallable, DisplayName = "ListUserCreatedModsAsync", Category = "mod.io|Mods") MODIO_API void K2_ListUserCreatedModsAsync(const FModioFilterParams& Filter, @@ -1639,70 +1675,71 @@ class UModioSubsystem : public UEngineSubsystem /** * @brief Returns the default mod installation directory for this game and platform, ignoring overrides and without * requiring the SDK to be initialized. - * @param GameID The FModioGameID of the game we're fetching the default mod installation directory for. + * @param GameID The `ModioGameID` of the game we're fetching the default mod installation directory for. * @return The default mod installation directory for the specified game on the current platform */ UFUNCTION(BlueprintCallable, DisplayName = "GetDefaultModInstallationDirectory", Category = "mod.io|Mods") static MODIO_API FString K2_GetDefaultModInstallationDirectory(FModioGameID GameID); /** - * @brief Install every temp mod given in the param if they are not already subbed. - * @param TArray of ModID to install as temp mod - * @return Error code indicating the status of the TempModSet. Will be empty if it was successful + * @brief Install every temp mod specified by `ModIds` if not already installed. + * @param ModIds TArray of `ModioModID`s to install as temp mods + * @return Error code indicating the status of the `TempModSet`. Will be empty if it was successful * @requires initialized-sdk - * @requires modmanagement enabled - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory ModManagementDisabled|ModManagement not enable + * @requires mod-management-enabled + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementDisabled`|Mod management not enabled */ UFUNCTION(BlueprintCallable, DisplayName = "InitTempModSet", Category = "mod.io|Mods") MODIO_API FModioErrorCode K2_InitTempModSet(TArray ModIds); /** - * @brief Add mods to a Temp Mod Set, install every temp mod given in the param if they are not already subbed. - * @param TArray of ModID to install as temp mod - * @return Error code indicating the status of the TempModSet. Will be empty if it was successful + * @brief Add mods to a Temp Mod Set. Every temp mod specified by `ModIds` will be installed if not already + * installed. + * @param ModIds TArray of `ModioModID`s to install as temp mods + * @return Error code indicating the status of the `TempModSet`. Will be empty if it was successful * @requires initialized-sdk - * @requires modmanagement enabled - * @requires TempModSet initialized - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory ModManagementDisabled|ModManagement not enable - * @errorcategory TempModSetNotInitialize| TempModSet not initialized + * @requires mod-management-enabled + * @requires temp-mod-set-initialized + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementDisabled`|Mod management not enabled + * @errorcategory `TempModSetNotInitialized`|`TempModSet` not initialized */ UFUNCTION(BlueprintCallable, DisplayName = "AddToTempModSet", Category = "mod.io|Mods") MODIO_API FModioErrorCode K2_AddToTempModSet(TArray ModIds); /** - * @brief Remove mods from a Temp Mod Set, uninstall every temp mod given in the param if they are not already - * subbed. - * @param TArray of ModID to install as temp mod - * @return Error code indicating the status of the TempModSet. Will be empty if it was successful + * @brief Remove mods from a temp mod set. Every temp mod specified by `ModIds` will be uninstalled unless the + * user is already subscribed. + * @param ModIds TArray of `ModioModID`s to remove as temp mods + * @return Error code indicating the status of the `TempModSet`. Will be empty if it was successful * @requires initialized-sdk - * @requires modmanagement enabled - * @requires TempModSet initialized - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory ModManagementDisabled|ModManagement not enable - * @errorcategory TempModSetNotInitialize| TempModSet not initialized + * @requires mod-management-enabled + * @requires temp-mod-set-initialized + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementDisabled`|Mod management not enabled + * @errorcategory `TempModSetNotInitialized`|`TempModSet` not initialized */ UFUNCTION(BlueprintCallable, DisplayName = "RemoveFromTempModSet", Category = "mod.io|Mods") MODIO_API FModioErrorCode K2_RemoveFromTempModSet(TArray ModIds); /** - * @brief Uninstall every temp mod. - * @return Error code indicating the status of the TempModSet. Will be empty if it was successful + * @brief Uninstall every temp mod unless the user is subscribed. + * @return Error code indicating the status of the `TempModSet`. Will be empty if it was successful * @requires initialized-sdk - * @requires modmanagement enabled - * @requires TempModSet initialized - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory ModManagementDisabled|ModManagement not enable - * @errorcategory TempModSetNotInitialize| TempModSet not initialized + * @requires mod-management-enabled + * @requires temp-mod-set-initialized + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `ModManagementDisabled`|Mod management not enabled + * @errorcategory `TempModSetNotInitialized`|`TempModSet` not initialized */ UFUNCTION(BlueprintCallable, DisplayName = "CloseTempModSet", Category = "mod.io|Mods") MODIO_API FModioErrorCode K2_CloseTempModSet(); /** - * @brief Query every System and Temp Mod in TempModSet - * @return TMap using Mod IDs as keys and ModCollectionEntry objects providing information about mods - * in TempModSet + * @brief Query every system and temp mod in `TempModSet` + * @return TMap using `ModioModID` as keys and `ModioModCollectionEntry` objects providing information about mods + * in `TempModSet` */ UFUNCTION(BlueprintCallable, DisplayName = "QueryTempModSet", Category = "mod.io|Mods") MODIO_API TMap K2_QueryTempModSet(); @@ -1711,9 +1748,9 @@ class UModioSubsystem : public UEngineSubsystem * @docpublic * @brief Purchases a mod for the current player * @param ModID ID of the mod to purchase - * @param ExpectedPrice The price the user is expected to pay for the mod, generally ModInfo.Price. This ensures - *that there is consistency between the displayed price and the price in the backend. If there is a mismatch, the - *purchase will fail. + * @param ExpectedPrice The price the user is expected to pay for the mod, generally + *[`ModioModInfo.Price`](#modiomodinfo). This ensures that there is consistency between the displayed price and the + *price in the backend. If there is a mismatch, the purchase will fail. * @param Callback Callback invoked with purchase information once the purchase is completed. * @requires initialized-sdk * @requires authenticated-user @@ -1725,7 +1762,7 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic - * @brief Get a User Delegation Token that can be used for S2S service calls + * @brief Get a user delegation token that can be used for S2S service calls * @param Callback Callback invoked with purchase information once the purchase is completed. * @requires initialized-sdk * @requires authenticated-user @@ -1737,7 +1774,7 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic * @brief Gets the users current wallet balance. This will also create a wallet for a user if one does not exist. - * You should ensure this is called prior to calling PurchaseModAsync + * You should ensure this is called prior to calling [`PurchaseModAsync`](#purchasemodasync) * purchase will fail. * @param Callback Callback invoked with the users wallet balance * @requires initialized-sdk @@ -1750,7 +1787,7 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic * @brief Fetches the user's purchases. This populates a runtime cache of purchase information - * that can be accessed using QueryUserPurchasedMods. + * that can be accessed using [`QueryUserPurchasedMods`](#query-user-purchased-mods). * @param Callback Callback invoked once the call has been completed. * @requires initialized-sdk * @requires authenticated-user @@ -1775,16 +1812,16 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic - * @brief Start a Metrics play session - * @param Params MetricsServiceParams struct containing information of what and how to start a metrics + * @brief Start a metrics play session + * @param Params `ModioMetricsSessionParams` struct containing information of what and how to start a metrics * session * @param Callback Callback providing an error code indicating success or failure of the session start operation - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory RateLimited|Too many frequent calls to the API. Wait some time and try again. - * @errorcategory InvalidUser|No authenticated user - * @errorcategory SessionNotInitialized|Metrics session has not yet been initialized - * @errorcategory SessionIsActive|Metrics session is currently active and running - * @errorcategory BadParameter|One or more values in the Metric Session Parameters are invalid + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `RateLimited`|Too many frequent calls to the API. Wait some time and try again. + * @errorcategory `InvalidUser`|No authenticated user + * @errorcategory `SessionNotInitialized`|Metrics session has not yet been initialized + * @errorcategory `SessionIsActive`|Metrics session is currently active and running + * @errorcategory `BadParameter`|One or more values in the Metric Session Parameters are invalid * @premiumfeature Metrics */ UFUNCTION(BlueprintCallable, DisplayName = "MetricsSessionStartAsync", Category = "mod.io|Metrics") @@ -1796,11 +1833,11 @@ class UModioSubsystem : public UEngineSubsystem * @param Callback Callback providing an error code indicating success or failure of the session heartbeat * operation * - * @errorcategory SDKNotInitialized|SDK not initialized - * @errorcategory InvalidUser|No authenticated user - * @errorcategory SessionNotInitialized|Metrics session has not yet been initialized - * @errorcategory SessionIsNotActive|Metrics session is not currently running. Call MetricsSessionStartAsync before - * attempting to sending a heartbeat. + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidUser`|No authenticated user + * @errorcategory `SessionNotInitialized`|Metrics session has not yet been initialized + * @errorcategory `SessionIsNotActive`|Metrics session is not currently running. Call + * [`MetricsSessionStartAsync`](#metricssessionstartasync) before attempting to sending a heartbeat. * @premiumfeature Metrics */ UFUNCTION(BlueprintCallable, DisplayName = "MetricsSessionSendHeartbeatOnceAsync", Category = "mod.io|Metrics") @@ -1813,12 +1850,12 @@ class UModioSubsystem : public UEngineSubsystem * @param Callback Callback providing an error code indicating success or failure of the session heartbeat * operation * - * @errorcategory GenericError::SDKNotInitialized|SDK not initialized - * @errorcategory UserDataError::InvalidUser|No authenticated user - * @errorcategory MetricsError::SessionNotInitialized|Metrics session has not yet been initialized - * @errorcategory MetricsError::SessionIsNotActive|Metrics session is not currently running. - * Call MetricsSessionStartAsync before attempting to sending a heartbeat. - * @premiumfeature Metrics + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `InvalidUser`|No authenticated user + * @errorcategory `SessionNotInitialized`|Metrics session has not yet been initialized + * @errorcategory `SessionIsNotActive`|Metrics session is not currently running. + * Call [`MetricsSessionStartAsync`](#metricssessionstartasync) before attempting to sending a heartbeat. + * @olden Metrics */ UFUNCTION(BlueprintCallable, DisplayName = "MetricsSessionSendHeartbeatAtIntervalAsync", Category = "mod.io|Metrics") @@ -1827,15 +1864,15 @@ class UModioSubsystem : public UEngineSubsystem /** * @docpublic - * @brief Ends a Metrics play session + * @brief Ends a metrics play session * @param Callback Callback providing an error code indicating success or failure of the session end operation * - * @errorcategory GenericError::SDKNotInitialized|SDK not initialized - * @errorcategory HttpError::RateLimited|Too many frequent calls to the API. Wait some time and try again. - * @errorcategory UserDataError::InvalidUser|No authenticated user - * @errorcategory MetricsError::SessionNotInitialized|Metrics session has not yet been initialized - * @errorcategory MetricsError::SessionIsNotActive|Metrics session is not currently running. - * Call MetricsSessionStartAsync before attempting to end a session. + * @errorcategory `SDKNotInitialized`|SDK not initialized + * @errorcategory `RateLimited`|Too many frequent calls to the API. Wait some time and try again. + * @errorcategory `InvalidUser`|No authenticated user + * @errorcategory `SessionNotInitialized`|Metrics session has not yet been initialized + * @errorcategory `SessionIsNotActive`|Metrics session is not currently running. + * Call [`MetricsSessionStartAsync`](#metricssessionstartasync) before attempting to end a session. * @premiumfeature Metrics */ UFUNCTION(BlueprintCallable, DisplayName = "MetricsSessionEndAsync", Category = "mod.io|Metrics") diff --git a/Source/Modio/Public/Types/ModioCommonTypes.h b/Source/Modio/Public/Types/ModioCommonTypes.h index 2ce4705c..1c695011 100644 --- a/Source/Modio/Public/Types/ModioCommonTypes.h +++ b/Source/Modio/Public/Types/ModioCommonTypes.h @@ -27,21 +27,20 @@ namespace Modio } // namespace Modio /** - * Enum representing what environment the game is deployed in, between - * test or a live environment + * @brief Enum representing which environment the game is deployed to: `Test` or `Live`. **/ UENUM(BlueprintType) enum class EModioEnvironment : uint8 { - /** Test/Private environment **/ + /** Test (private) environment */ Test, - /** Live/Public environment **/ + /** Live (public) environment */ Live }; /** - * Enum representing the store or service your game is being distributed through + * @brief Enum representing the store or service your game is being distributed through **/ UENUM(BlueprintType) enum class EModioPortal : uint8 @@ -59,7 +58,7 @@ enum class EModioPortal : uint8 }; /** - * Enum representing a named platform that the plugin is running on. + * @brief Enum representing a named platform that the plugin is running on. **/ UENUM(BlueprintType) enum class EModioPlatformName : uint8 @@ -78,7 +77,7 @@ enum class EModioPlatformName : uint8 }; /** - * Enum representing the platform(s) that a modfile is enabled for + * @brief Enum representing the platform(s) that a modfile is enabled for **/ UENUM(BlueprintType) enum class EModioModfilePlatform : uint8 @@ -103,16 +102,13 @@ enum class EModioModfilePlatform : uint8 UENUM(BlueprintType) enum class EModioLogoSize : uint8 { - /** 320x180px **/ + /** 320x180px */ Thumb320 = 0, - - /** 640x360px **/ + /** 640x360px */ Thumb640 = 1, - - /** 1280x720px **/ + /** 1280x720px */ Thumb1280 = 2, - - /** Original Size **/ + /** Original Size */ Original = 3 }; @@ -122,13 +118,11 @@ enum class EModioLogoSize : uint8 UENUM(BlueprintType) enum class EModioAvatarSize : uint8 { - /** Original Size **/ + /** Original Size */ Original, - - /** 50x50px Thumbnail **/ + /** 50x50px Thumbnail */ Thumb50, - - /** 100x100px Thumbnail **/ + /** 100x100px Thumbnail */ Thumb100 }; @@ -138,13 +132,13 @@ enum class EModioAvatarSize : uint8 UENUM(BlueprintType) enum class EModioGallerySize : uint8 { - /** Original Size **/ + /** Original Size */ Original, - /** 320x180px Thumbnail **/ + /** 320x180px Thumbnail */ Thumb320, - /** 1280x720 Thumbnail **/ + /** 1280x720 Thumbnail */ Thumb1280 }; @@ -154,19 +148,19 @@ enum class EModioGallerySize : uint8 UENUM(BlueprintType) enum class EModioLogLevel : uint8 { - /** Detailed low-level debugging output. Not intended for general use **/ + /** Detailed low-level debugging output. Not intended for general use */ Trace = 0, /* Detailed but not low-level. Generally useful for some mid-level information for debugging. */ Detailed = 1, - /** Informational output containing status messages **/ + /** Informational output containing status messages */ Info = 2, - /** Warnings about incorrect plugin usage, timeouts **/ + /** Warnings about incorrect plugin usage, timeouts */ Warning = 3, - /** Only errors **/ + /** Only errors */ Error = 4 }; @@ -201,9 +195,12 @@ ENUM_RANGE_BY_COUNT(EModioLanguage, EModioLanguage::Count); UENUM(BlueprintType) enum class EModioModChangeType : uint8 { - Added, // The user's list has a new mod to synchronize - Removed, // The user's list must remove a mod to synchronize - Updated // The user's list must update a mod to synchronize + // The user's list has a new mod to synchronize + Added, + // The user's list must remove a mod to synchronize + Removed, + // The user's list must update a mod to synchronize + Updated }; /** @@ -270,13 +267,16 @@ struct MODIO_API FModioModID **/ FString ToString() const { - if (ModID < 0) - { - return TEXT("InvalidModID"); - } return FString::Printf(TEXT("%lld"), ModID); } + /// @docinternal + /// @brief Compare the ModID to the invalid state + bool IsValid() const + { + return ModID != INDEX_NONE; + } + /** * Stream forward operator to pass the ModID along * @param Ar The archive class that receives information @@ -387,7 +387,7 @@ struct MODIO_API FModioGameID /** * Transform a GameID into its 32 bit integer representation - * @param ModGameId String type for the GameID to read its hash from + * @param ModioGameId String type for the GameID to read its hash from * @return unsigned 32 bit integer that matches the hash of the GameID provided **/ MODIO_API friend uint32 GetTypeHash(FModioGameID ModioGameId); @@ -406,13 +406,16 @@ struct MODIO_API FModioGameID **/ FString ToString() const { - if (GameId < 0) - { - return TEXT("InvalidGameID"); - } return FString::Printf(TEXT("%lld"), GameId); } + /// @docinternal + /// @brief Compare the GameID to the invalid state defined by the SDK + bool IsValid() const + { + return GameId != InvalidGameID().GameId; + } + /** * An always invalid GameID, helpful to compare against other GameIDs * @return String value of the stored GameID @@ -513,13 +516,16 @@ struct MODIO_API FModioFileMetadataID **/ FString ToString() const { - if (FileMetadataID < 0) - { - return TEXT("InvalidFileMetadataID"); - } return FString::Printf(TEXT("%lld"), FileMetadataID); } + /// @docinternal + /// @brief Compare the FileMetadataID to the invalid state + bool IsValid() const + { + return FileMetadataID != INDEX_NONE; + } + /** * Stream forward operator to pass the FileMetadataID along * @param Ar The archive class that receives information @@ -612,13 +618,16 @@ struct MODIO_API FModioUserID **/ FString ToString() const { - if (UserID < 0) - { - return TEXT("InvalidUserID"); - } return FString::Printf(TEXT("%lld"), UserID); } + /// @docinternal + /// @brief Compare the UserID to the invalid state + bool IsValid() const + { + return UserID != INDEX_NONE; + } + /** * Stream forward operator to pass the UserID along * @param Ar The archive class that receives information @@ -696,16 +705,16 @@ struct MODIO_API FModioApiKey **/ const FString& ToString() const { - // Put in the function instead of default constructor to avoid having to allocate memory for - // each empty instance - if (ApiKey.Len() == 0) - { - static FString Invalid(TEXT("InvalidApiKey")); - return Invalid; - } return ApiKey; } + /// @docinternal + /// @brief Compare the ApiKey to the invalid state defined by the SDK + bool IsValid() const + { + return ApiKey != InvalidAPIKey().ToString(); + } + /** * An always invalid APIKey, helpful to compare against other APIKeys * @return String value of the stored APIKey @@ -750,18 +759,11 @@ struct MODIO_API FModioGuid **/ const FString& ToString() const { - // Put in the function instead of default constructor to avoid having to allocate memory for - // each empty instance - if (InternalGuid.Len() == 0) - { - static FString Invalid(TEXT("InvalidGuid")); - return Invalid; - } return InternalGuid; } /// @docinternal - /// @brief Compare the InternalGuid to the invalid state define by the SDK + /// @brief Compare the InternalGuid to the invalid state defined by the SDK bool IsValid() const { return InternalGuid != InvalidGuid().InternalGuid; @@ -818,16 +820,16 @@ struct MODIO_API FModioEmailAddress **/ const FString& ToString() const { - // Put in the function instead of default constructor to avoid having to allocate memory for - // each empty instance - if (EmailAddress.Len() == 0) - { - static FString Invalid(TEXT("InvalidEmailAddress")); - return Invalid; - } return EmailAddress; } + /// @docinternal + /// @brief Compare the EmailAddress to the invalid state + bool IsValid() const + { + return EmailAddress.Len() > 0; + } + private: FString EmailAddress; }; @@ -860,6 +862,13 @@ struct MODIO_API FModioEmailAuthCode return EmailAuthCode; } + /// @docinternal + /// @brief Compare the EmailAuthCode to the invalid state + bool IsValid() const + { + return EmailAuthCode.Len() > 0; + } + private: FString EmailAuthCode; }; @@ -937,18 +946,114 @@ struct MODIO_API FModioMetricsSessionParams UENUM(BlueprintType) enum EFileSizeUnit { - /** Will take the largest one that becomes a number larger than 1 (i.e, 1300mb becomes 1.3gb) **/ + /** Will take the largest one that becomes a number larger than 1 (i.e, 1300mb becomes 1.3gb) */ Largest = 0, - - /** A single byte **/ + /** A single byte */ B = 1, - - /** Kilo bytes **/ + /** Kilobytes */ KB = 1024, - - /** Mega bytes **/ + /** Megabytes */ MB = 1024 * 1024, - - /** Giga bytes **/ + /** Gigabytes */ GB = 1024 * 1024 * 1024 -}; \ No newline at end of file +}; + +USTRUCT(BlueprintType) +struct MODIO_API FModioTokenPackID +{ + GENERATED_BODY() + + /** + * Default constructor without parameters + **/ + FModioTokenPackID(); + + /** + * Preferred constructor with Token Pack initialization parameter + * @param InTokenPAckId Base Token Pack ID to create this strong type + **/ + FModioTokenPackID(FString InTokenPackId) + { + TokenPackID = InTokenPackId; + } + + /** + * Transform a TokenPackID into its 32 bit integer representation + * @param ModioTokenPackId String type for the TokenPackID to read its hash from + * @return unsigned 32 bit integer that matches the hash of the TokenPackID provided + **/ + MODIO_API friend uint32 GetTypeHash(FModioTokenPackID ModioTokenPackId); + + /** + * Comparison operator between TokenPackID elements + **/ + MODIO_API friend bool operator==(FModioTokenPackID A, FModioTokenPackID B) + { + return A.TokenPackID == B.TokenPackID; + } + + /** + * Comparison operator between TokenPackID elements + **/ + MODIO_API friend bool operator!=(FModioTokenPackID A, FModioTokenPackID B) + { + return A.TokenPackID != B.TokenPackID; + } + + /** + * Transform a TokenPackID into its string representation + * @return String value of the stored TokenPackID + **/ + FString ToString() const + { + return TokenPackID; + } + + /** + * Stream forward operator to pass the TokenPackID along + * @param Ar The archive class that receives information + * @param ID The TokenPackID to pass along this operator + * @return FArchive The updated archive with the TokenPackID passed along + **/ + MODIO_API friend FArchive& operator<<(FArchive& Ar, FModioTokenPackID& ID) + { + return Ar << ID.TokenPackID; + } + + /** + * Store this instance TokenPackID into an archive + * @param Ar The archive class that receives information + * @return Always true when the TokenPackID is forwarded to Ar + **/ + bool Serialize(FArchive& Ar) + { + Ar << *this; + return true; + } + + /** + * Store this instance TokenPackID into an archive with reference flag as parameter + * @param Ar The archive class that receives information + * @param Map A dictionary from indices for network communication + * @param bOutSuccess Flag to signal the result of this operation, True when stored + * @return Always true when the ModID is forwarded to Ar + **/ + bool NetSerialize(FArchive& Ar, class UPackageMap* Map, bool& bOutSuccess) + { + Ar << *this; + bOutSuccess = true; + return true; + } + + private: + /// @brief Function for retrieving the underlying value of an FModioTokenPackID for use in custom serialization. Not + /// recommended for any other use. FModioTokenPackIDs should be treated as opaque. + /// @param In the FModioTokenPackID to retrieve the value for + /// @return the underlying string ID. + MODIO_API friend FString GetUnderlyingValue(const FModioTokenPackID& In) + { + return In.TokenPackID; + } + + FString TokenPackID; +}; diff --git a/Source/Modio/Public/Types/ModioErrorCode.h b/Source/Modio/Public/Types/ModioErrorCode.h index ca16a698..cfa5c00a 100644 --- a/Source/Modio/Public/Types/ModioErrorCode.h +++ b/Source/Modio/Public/Types/ModioErrorCode.h @@ -20,7 +20,7 @@ namespace Modio class ErrorCode; } -/** @brief wrapper around Modio::ErrorCode */ +/** @brief Wrapper around `Modio::ErrorCode` */ USTRUCT(BlueprintType) struct MODIO_API FModioErrorCode { diff --git a/Source/Modio/Public/Types/ModioFilterParams.h b/Source/Modio/Public/Types/ModioFilterParams.h index dc114b4e..c8c3ee8f 100644 --- a/Source/Modio/Public/Types/ModioFilterParams.h +++ b/Source/Modio/Public/Types/ModioFilterParams.h @@ -27,14 +27,22 @@ namespace Modio UENUM(BlueprintType) enum class EModioSortFieldType : uint8 { - ID, /** use mod ID (default) */ - DownloadsToday, /** use number of downloads in last 24 (exposed in REST API as 'popular' */ - SubscriberCount, /** use number of subscribers */ - Rating, /** use mod rating */ - DateMarkedLive, /** use date mod was marked live */ - DateUpdated, /** use date mod was last updated */ - DownloadsTotal, /** use downloads total */ - Alphabetical /** use mod name */ + /** Use mod ID (default) */ + ID, + /** Use number of downloads in last 24 (exposed in REST API as "popular") */ + DownloadsToday, + /** Use number of subscribers */ + SubscriberCount, + /** Use mod rating */ + Rating, + /** Use date mod was marked live */ + DateMarkedLive, + /** Use date mod was last updated */ + DateUpdated, + /** Use downloads total */ + DownloadsTotal, + /** Use mod name */ + Alphabetical }; /// @brief Enum indicating which direction sorting should be applied @@ -49,18 +57,22 @@ enum class EModioSortDirection : uint8 UENUM(BlueprintType) enum class EModioRevenueFilterType : uint8 { - Free = 0, /** Return only free mods */ - Paid = 1, /** Return only paid mods */ - FreeAndPaid = 2 /** Return both free and paid mods */ + /** Return only free mods */ + Free = 0, + /** Return only paid mods */ + Paid = 1, + /** Return both free and paid mods */ + FreeAndPaid = 2 }; -/** @brief Class storing a set of filter parameters for use in xref:ListAllModsAsync[] */ +/** @brief Class storing a set of filter parameters for use in [`ListAllModsAsync`](#listallmodsasync) */ USTRUCT(BlueprintType) struct MODIO_API FModioFilterParams { GENERATED_BODY() /** + * @docpublic * @brief Only include mods with the specified author User ID * @param UserID The User ID to filter on * @return *this @@ -68,6 +80,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& MatchingAuthor(const FModioUserID& UserID); /** + * @docpublic * @brief Only include mods with the specified author User IDs * @param UserIDs The list of User IDs to filter on * @return *this @@ -75,6 +88,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& MatchingAuthors(const TArray& UserIDs); /** + * @docpublic * @brief Indicates the filter should only include the specified mods * @param IDs the set of mods to match * @return *this @@ -82,6 +96,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& MatchingIDs(const TArray& IDs); /** + * @docpublic * @brief Indicates the filter should exclude the specified mods. * @param IDs the set of mods to exclude * @return *this @@ -89,6 +104,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& ExcludingIDs(const TArray& IDs); /** + * @docpublic * @brief Indicates results should be sorted using the specified field and direction * @param ByField Field to sort with * @param ByDirection Direction to sort @@ -97,6 +113,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& SortBy(EModioSortFieldType ByField, EModioSortDirection ByDirection); /** + * @docpublic * @brief Only include mods where the name contains the provided string * @param SearchString Search string * @return *this @@ -104,16 +121,16 @@ struct MODIO_API FModioFilterParams FModioFilterParams& NameContains(const FString& SearchString); /** + * @docpublic * @brief Only include mods where the name contains at least one of the provided strings (string1 OR string2 OR - *stringN...) - * @tparam ...Args std::string + * stringN...) * @param SearchString First search string - * @param ...args Additional search strings * @return *this **/ FModioFilterParams& NameContains(const TArray& SearchString); /** + * @docpublic * @brief Only include mods that were marked live (i.e released) after the specified date * @param LiveAfter Minimum date * @return *this @@ -121,6 +138,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& MarkedLiveAfter(FDateTime LiveAfter); /** + * @docpublic * @brief Only include mods that were marked live (i.e released) before the specified date * @param LiveBefore Maximum date * @return *this @@ -128,6 +146,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& MarkedLiveBefore(FDateTime LiveBefore); /** + * @docpublic * @brief Only include mods with a metadata blob containing the specified substring * @param SearchString The substring to search for * @return *this @@ -135,6 +154,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& MetadataLike(FString SearchString); /** + * @docpublic * @brief Only include mods that have the specified tag * @param Tag Tag to include * @return *this @@ -142,6 +162,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& WithTags(const FString& Tag); /** + * @docpublic * @brief Only include mods that have all the specified tags (tag1 AND tag2 AND tagN...) * @param NewTags The set of tags to filter on * @return *this @@ -149,6 +170,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& WithTags(const TArray& NewTags); /** + * @docpublic * @brief Only include mods that do not have the specified tag * @param Tag Tag to exclude * @return *this @@ -156,6 +178,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& WithoutTags(const FString& Tag); /** + * @docpublic * @brief Only include mods that do not have any of the specified tags ( NOT (tag1 OR tag2 OR tagN...)) * @param NewTags Tags to exclude * @return *this @@ -163,6 +186,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& WithoutTags(const TArray& NewTags); /** + * @docpublic * @brief Returns a sub-range of query results from StartIndex to StartIndex + ResultCount * @param StartIndex Zero-based index of first result to return * @param ResultCount Number of results to return @@ -171,6 +195,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& IndexedResults(uint64 StartIndex, uint64 ResultCount); /** + * @docpublic * @brief Returns a sub-range of query results based on a specified page size and index * @param PageNumber Zero-based index of page to return * @param PageSize Number of results in a page @@ -179,6 +204,7 @@ struct MODIO_API FModioFilterParams FModioFilterParams& PagedResults(uint64 PageNumber, uint64 PageSize); /** + * @docpublic * @brief Returned mods according to the specified revenue type (free, paid, both) * @param RevenueFilter Filter to use * @return *this @@ -186,12 +212,14 @@ struct MODIO_API FModioFilterParams FModioFilterParams& RevenueType(EModioRevenueFilterType RevenueFilter); /** + * @docpublic * @brief Indicates results should exclude all mods which contain mature content * @return *this */ FModioFilterParams& DisallowMatureContent(); /** + * @docpublic * @brief Indicates results should be filtered by maturity options * @param ByMaturity Maturity flags to filter by * @return *this @@ -199,9 +227,10 @@ struct MODIO_API FModioFilterParams FModioFilterParams& WithMatureContentFlags(EModioMaturityFlags ByMaturity); /** - * @brief Converts the filter params to a string suitable for use in the REST API. - * @note Performs a allocation to acquire the string - * @return FString containing the filter parameters + * @docpublic + * @brief Converts the filter params to a string suitable for use in the REST API. + * @note Performs allocation to acquire the string + * @return FString containing the filter parameters */ FString ToString() const; @@ -227,7 +256,7 @@ struct MODIO_API FModioFilterParams }; /** - * Helper struct for named preset filter parameters + * @brief Helper struct for named preset filter parameters */ USTRUCT(BlueprintType) struct MODIO_API FModioPresetFilterParams @@ -262,13 +291,20 @@ struct MODIO_API FModioPresetFilterParams } }; +/** + * @brief Blueprint library for working with preset filter parameters + */ UCLASS() class MODIO_API UModioPresetFilterParamsLibrary : public UBlueprintFunctionLibrary { GENERATED_BODY() protected: - // Compiles a preset filter into a concrete set of filter parameters that can be passed to the mod.io plugin + /** + * @brief Converts a preset filter into a concrete set of filter parameters that can be passed to the mod.io plugin + * @param Preset The preset to convert + * @return The converted filter params + */ UFUNCTION(BlueprintCallable, Category = "mod.io|Filter Params") static FModioFilterParams ToFilterParams(const FModioPresetFilterParams& Preset) { diff --git a/Source/Modio/Public/Types/ModioGameInfo.h b/Source/Modio/Public/Types/ModioGameInfo.h index bd4f7d39..835290fa 100644 --- a/Source/Modio/Public/Types/ModioGameInfo.h +++ b/Source/Modio/Public/Types/ModioGameInfo.h @@ -30,31 +30,31 @@ namespace Modio /** * @brief Monetization properties of a game -* 0 = None set (default) -* 1 = Monetization is enabled -* 2 = Marketplace is enabled -* 4 = Partner Program is enabled **/ UENUM(BlueprintType, meta = (Bitflags, UseEnumValuesAsMaskValuesInEditor = "true")) enum class EGameMonetizationFlags : uint8 { + // None set (default) None = 0, + // Monetization is enabled Monetization = 1, + // Marketplace is enabled Marketplace = 2, + // Partner Program is enabled PartnerProgram = 4, }; /** * @brief Maturity options for a game -* 0 = Don't allow mature content in mods (default) -* 1 = This game allows mods containing mature content -* 2 = This game is for mature audiences only **/ UENUM(BlueprintType, meta = (Bitflags, UseEnumValuesAsMaskValuesInEditor = "true")) enum class EGameMaturityFlags : uint8 { + // Don't allow mature content in mods (default) None = 0, + // This game allows mods containing mature content MatureModsAllowed = 1, + // This game is for mature audiences only MatureAudiencesOnly = 2, }; @@ -110,8 +110,7 @@ struct MODIO_API FModioGameInfo UPROPERTY(BlueprintReadOnly, Category = "mod.io|GameInfo") FString Instructions; - /** @brief Link to a mod.io guide, modding wiki, or a page where modders can learn how to make and submit mods */ - /// to this game's profile + /** @brief Link to a mod.io guide, modding wiki, or a page where modders can learn how to make and submit mods to this game's profile */ UPROPERTY(BlueprintReadOnly, Category = "mod.io|GameInfo") FString InstructionsUrl; diff --git a/Source/Modio/Public/Types/ModioImageState.h b/Source/Modio/Public/Types/ModioImageState.h index 4ef524f2..721bf780 100644 --- a/Source/Modio/Public/Types/ModioImageState.h +++ b/Source/Modio/Public/Types/ModioImageState.h @@ -19,15 +19,15 @@ UENUM(BlueprintType) enum class EModioImageState : uint8 { - /** Image data is located on hard drive **/ + /** Image data is located on hard drive */ OnDisc, - /** Image data is transferring to a memory location **/ + /** Image data is transferring to a memory location */ LoadingIntoMemory, - /** Image data is located in memory **/ + /** Image data is located in memory */ InMemory, - /** Image data is not readable **/ + /** Image data is not readable */ Corrupted }; \ No newline at end of file diff --git a/Source/Modio/Public/Types/ModioInitializeOptions.h b/Source/Modio/Public/Types/ModioInitializeOptions.h index 8a9dbd83..0f796364 100644 --- a/Source/Modio/Public/Types/ModioInitializeOptions.h +++ b/Source/Modio/Public/Types/ModioInitializeOptions.h @@ -45,7 +45,7 @@ struct MODIO_API FModioInitializeOptions UPROPERTY(BlueprintReadOnly, EditAnywhere, Category = "mod.io") TMap ExtendedInitializationParameters; - /** @brief Set Mod.io to run with background thread*/ + /** @brief Set mod.io to run with background thread*/ UPROPERTY(BlueprintReadOnly, EditAnywhere, Category = "mod.io") bool bUseBackgroundThread = false; diff --git a/Source/Modio/Public/Types/ModioModCollectionEntry.h b/Source/Modio/Public/Types/ModioModCollectionEntry.h index 3a106406..b32d687f 100644 --- a/Source/Modio/Public/Types/ModioModCollectionEntry.h +++ b/Source/Modio/Public/Types/ModioModCollectionEntry.h @@ -26,12 +26,18 @@ namespace Modio UENUM(BlueprintType) enum class EModioModState : uint8 { - InstallationPending, // dont save + // The mod is pending installation. This state is not saved. + InstallationPending, + // The mod is installed. Installed, - UpdatePending, // saved as installed - Downloading, // installing - dont save - Extracting, // installing- don't save - UninstallPending, // saved as installed + // The mod is pending an update. This state is saved as installed. + UpdatePending, + // The mod is downloading as part of the installation process. This state is not saved. + Downloading, + // The mod is extracting as part of the installation process. This state is not saved. + Extracting, + // The mod is pending uninstallation. This state is saved as installed. + UninstallPending, }; /** diff --git a/Source/Modio/Public/Types/ModioModInfo.h b/Source/Modio/Public/Types/ModioModInfo.h index e3d5fff7..e3aff65c 100644 --- a/Source/Modio/Public/Types/ModioModInfo.h +++ b/Source/Modio/Public/Types/ModioModInfo.h @@ -26,43 +26,38 @@ namespace Modio } /** - * Enumeration that represent mature content for the mod to create + * @brief Enum representing whether or not a mod is visible to users **/ UENUM(BlueprintType, meta = (Bitflags, UseEnumValuesAsMaskValuesInEditor = "true")) enum class EModioObjectVisibilityFlags : uint8 { - /** Mod is concealed from users **/ + /** Mod is concealed from users */ Hidden = 0, - - /** Mod is openly available **/ + /** Mod is openly available */ Public = 1 }; /** - * Enumeration that represent mature content for the mod to create + * @brief Enum representing mature content that a mod may contain **/ UENUM(BlueprintType, meta = (Bitflags, UseEnumValuesAsMaskValuesInEditor = "true")) enum class EModioMaturityFlags : uint8 { - /** No maturity **/ + // No maturity None, - - /** Content contains alcohol references **/ + // Content contains alcohol references Alcohol = 1, - - /** Content contains drug references **/ + // Content contains drug references Drugs = 2, - - /** Content contains violence references **/ + // Content contains violence references Violence = 4, - - /** Content contains sexual references **/ + // Content contains sexual references Explicit = 8 }; ENUM_CLASS_FLAGS(EModioMaturityFlags); /** - * Enumeration that represent mod server side status + * @brief Enum representing a mod's server side status **/ UENUM(BlueprintType, meta = (Bitflags, UseEnumValuesAsMaskValuesInEditor = "true")) enum class EModioModServerSideStatus : uint8 @@ -121,9 +116,8 @@ struct MODIO_API FModioModInfo FDateTime ProfileDateLive; /** - * @brief Flags for maturity options - * Maturity options flagged by the mod developer, this is only relevant if the parent game allows mods to - * be labeled as mature. + * @brief Flags for maturity options. Maturity options are flagged by the mod developer. This is only relevant if + *the parent game allows mods to be labeled as mature. **/ UPROPERTY(BlueprintReadOnly, EditInstanceOnly, Category = "Profile") EModioMaturityFlags ProfileMaturityOption {}; @@ -131,8 +125,10 @@ struct MODIO_API FModioModInfo /** @brief Is the mod marked as visible? * @deprecated Use Visibility property instead */ - UPROPERTY(meta = (DeprecatedProperty, - DeprecationMessage = "Deprecated as of 2023.6 release. Please use the <> instead.")) + UPROPERTY( + meta = (DeprecatedProperty, + DeprecationMessage = + "Deprecated as of 2023.6 release. Please use `EModioObjectVisibilityFlags Visibility` instead.")) bool bVisible_DEPRECATED {}; /** diff --git a/Source/Modio/Public/Types/ModioModManagementEvent.h b/Source/Modio/Public/Types/ModioModManagementEvent.h index fdad2d4b..4894579c 100644 --- a/Source/Modio/Public/Types/ModioModManagementEvent.h +++ b/Source/Modio/Public/Types/ModioModManagementEvent.h @@ -19,14 +19,22 @@ UENUM(BlueprintType) enum class EModioModManagementEventType : uint8 { - Installed, /** Mod installation to local storage completed*/ - Uninstalled, /** Mod uninstallation from local storage completed*/ - Updated, /** Mod local installation updated to latest version*/ - Uploaded, /** Mod file was uploaded*/ - BeginInstall, /** Mod download and installation has started */ - BeginUninstall, /** Mod uninstallation has started */ - BeginUpdate, /** Mod download and update has started*/ - BeginUpload /** Mod upload has started*/ + /** Mod installation to local storage completed*/ + Installed, + /** Mod uninstallation from local storage completed*/ + Uninstalled, + /** Mod local installation updated to latest version*/ + Updated, + /** Mod file was uploaded*/ + Uploaded, + /** Mod download and installation has started */ + BeginInstall, + /** Mod uninstallation has started */ + BeginUninstall, + /** Mod download and update has started*/ + BeginUpdate, + /** Mod upload has started*/ + BeginUpload }; /** @brief Simple struct representing the outcome of a mod management operation */ diff --git a/Source/Modio/Public/Types/ModioModProgressInfo.h b/Source/Modio/Public/Types/ModioModProgressInfo.h index 50d8a97d..5f4ae8d1 100644 --- a/Source/Modio/Public/Types/ModioModProgressInfo.h +++ b/Source/Modio/Public/Types/ModioModProgressInfo.h @@ -23,11 +23,16 @@ namespace Modio UENUM(BlueprintType) enum class EModioModProgressState : uint8 { - Initializing, // Download information is being retrieved from mod.io servers - Downloading, // Mod archive is downloading from mod.io servers - Extracting, // Mod archive is downloaded and now extracting - Compressing, // Mod archive is being compressed from files on disk - Uploading // Mod archive is uploading to mod.io servers + // Download information is being retrieved from mod.io servers + Initializing, + // Mod archive is downloading from mod.io servers + Downloading, + // Mod archive is downloaded and now extracting + Extracting, + // Mod archive is being compressed from files on disk + Compressing, + // Mod archive is uploading to mod.io servers + Uploading }; /** diff --git a/Source/Modio/Public/Types/ModioOpenStoreResult.h b/Source/Modio/Public/Types/ModioOpenStoreResult.h new file mode 100644 index 00000000..ce3cb60b --- /dev/null +++ b/Source/Modio/Public/Types/ModioOpenStoreResult.h @@ -0,0 +1,32 @@ +/* + * Copyright (C) 2024 mod.io Pty Ltd. + * + * This file is part of the mod.io UE5 Plugin. + * + * Distributed under the MIT License. (See accompanying file LICENSE or + * view online at ) + * + */ + +#pragma once + +#include "ModioOpenStoreResult.generated.h" + +/** + * @brief Enumerator of potential results on attempting to open a native platform store + **/ +UENUM(BlueprintType) +enum class EModioOpenStoreResult : uint8 +{ + /** Store opened successfully **/ + Success, + + /** Monetization not active **/ + FailedInactive, + + /** The current platform does not support a native store **/ + FailedUnsupportedPlatform, + + /** Failed to open for an unknown reason **/ + FailedUnknown +}; \ No newline at end of file diff --git a/Source/Modio/Public/Types/ModioRating.h b/Source/Modio/Public/Types/ModioRating.h index a07a8fc3..69f0a41d 100644 --- a/Source/Modio/Public/Types/ModioRating.h +++ b/Source/Modio/Public/Types/ModioRating.h @@ -18,12 +18,10 @@ UENUM(BlueprintType) enum class EModioRating : uint8 { - /** A neutral rating **/ + /** A neutral rating */ Neutral, - - /** A positive rating **/ + /** A positive rating */ Positive, - - /** A negative rating **/ + /** A negative rating */ Negative }; \ No newline at end of file diff --git a/Source/Modio/Public/Types/ModioReportParams.h b/Source/Modio/Public/Types/ModioReportParams.h index 7e3afc51..92199a74 100644 --- a/Source/Modio/Public/Types/ModioReportParams.h +++ b/Source/Modio/Public/Types/ModioReportParams.h @@ -24,28 +24,21 @@ namespace Modio UENUM(BlueprintType) enum class EModioReportType : uint8 { - /** A generic mod report **/ + /** A generic mod report */ Generic = 0, - - /** Digital Millennium Copyright Act mod report **/ + /** Digital Millennium Copyright Act mod report */ DMCA = 1, - - /** Not working mod report **/ + /** Not working mod report */ NotWorking = 2, - - /** Rude content mod report **/ + /** Rude content mod report */ RudeContent = 3, - - /** Illegal content mod report **/ + /** Illegal content mod report */ IllegalContent = 4, - - /** Stolen content mod report **/ + /** Stolen content mod report */ StolenContent = 5, - - /** False information mod report **/ + /** False information mod report */ FalseInformation = 6, - - /** Other type of mod report **/ + /** Other type of mod report */ Other = 7 }; ENUM_RANGE_BY_FIRST_AND_LAST(EModioReportType, EModioReportType::Generic, EModioReportType::Other); diff --git a/Source/Modio/Public/Types/ModioTokenPack.h b/Source/Modio/Public/Types/ModioTokenPack.h new file mode 100644 index 00000000..59ba3ca2 --- /dev/null +++ b/Source/Modio/Public/Types/ModioTokenPack.h @@ -0,0 +1,183 @@ +/* + * Copyright (C) 2024 mod.io Pty Ltd. + * + * This file is part of the mod.io UE4 Plugin. + * + * Distributed under the MIT License. (See accompanying file LICENSE or + * view online at ) + * + */ + +#pragma once + +#include "Types/ModioUnsigned64.h" +#include "Types/ModioCommonTypes.h" + +#include "Interfaces/OnlineStoreInterfaceV2.h" + +#include "ModioTokenPack.generated.h" + +/** + * @brief A wrapper type around the Unreal Engine type FOnlineStoreOffer, representing a platform-agnostic store offer/product, e.g a Premium Currency pack. + */ +USTRUCT(BlueprintType) +struct MODIO_API FModioTokenPack +{ + GENERATED_BODY() + +private: + + FOnlineStoreOffer Underlying; + +public: + + FModioTokenPack() = default; + + FModioTokenPack(const FOnlineStoreOffer& InOffer) + { + Underlying = InOffer; + } + + FText GetTitle() const + { + return Underlying.Title; + } + + FText GetDescription() const + { + return Underlying.Description; + } + + FText GetLongDescription() const + { + return Underlying.LongDescription; + } + + FText GetRegularPrice() const + { + return Underlying.GetDisplayRegularPrice(); + } + + FText GetDisplayPrice() const + { + return Underlying.GetDisplayPrice(); + } + + int64 GetNumericPrice() const + { + return Underlying.NumericPrice; + } + + FString GetCurrencyCode() const + { + return Underlying.CurrencyCode; + } + + FDateTime GetReleaseDate() const + { + return Underlying.ReleaseDate; + } + + FDateTime GetExpirationDate() const + { + return Underlying.ExpirationDate; + } + + bool IsPurchasable() const + { + return Underlying.IsPurchaseable(); + } + + TMap GetFields() const + { + return Underlying.DynamicFields; + } + + FString GetId() const + { + return Underlying.OfferId; + } + + FModioTokenPackID GetModioId() const + { + return FModioTokenPackID(Underlying.OfferId); + } + +}; + +USTRUCT(BlueprintType) +struct MODIO_API FModioOptionalTokenPack +{ + GENERATED_BODY() + + TOptional Internal; +}; + +UCLASS() +class UModioTokenPackLibrary : public UBlueprintFunctionLibrary +{ + GENERATED_BODY() + +public: + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static FText GetTitle(const FModioTokenPack& In) + { + return In.GetTitle(); + } + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static FText GetDescription(const FModioTokenPack& In) + { + return In.GetDescription(); + } + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static FText GetLongDescription(const FModioTokenPack& In) + { + return In.GetLongDescription(); + } + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static FText GetRegularPrice(const FModioTokenPack& In) + { + return In.GetRegularPrice(); + } + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static FText GetDisplayPrice(const FModioTokenPack& In) + { + return In.GetDisplayPrice(); + } + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static int64 GetNumericPrice(const FModioTokenPack& In) + { + return In.GetNumericPrice(); + } + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static bool IsPurchasable(const FModioTokenPack& In) + { + return In.IsPurchasable(); + } + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static TMap GetFields(const FModioTokenPack& In) + { + return In.GetFields(); + } + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static FString GetId(const FModioTokenPack& In) + { + return In.GetId(); + } + + UFUNCTION(BlueprintCallable, BlueprintPure, Meta = (Category = "Token Pack")) + static FModioTokenPackID GetModioId(const FModioTokenPack& In) + { + return In.GetModioId(); + } + +}; \ No newline at end of file diff --git a/Source/Modio/Public/Types/ModioTokenPackList.h b/Source/Modio/Public/Types/ModioTokenPackList.h new file mode 100644 index 00000000..04538d0d --- /dev/null +++ b/Source/Modio/Public/Types/ModioTokenPackList.h @@ -0,0 +1,84 @@ +/* + * Copyright (C) 2021 mod.io Pty Ltd. + * + * This file is part of the mod.io UE4 Plugin. + * + * Distributed under the MIT License. (See accompanying file LICENSE or + * view online at ) + * + */ + +#pragma once + +#include "Types/ModioList.h" +#include "Types/ModioTokenPack.h" +#include "Types/ModioPagedResult.h" + +#include "ModioTokenPackList.generated.h" + +#if CPP +struct MODIO_API FModioTokenPackList : public FModioPagedResult, public FModioList +{ + /** + * Default constructor without parameters + **/ + FModioTokenPackList() = default; + + /** + * Constructor that takes an array of token packs and an index of paged results + **/ + FModioTokenPackList(const FModioPagedResult& PagedResult, TArray&& TokenPackList); + + /** + * Constructor that takes only an array of token packs + */ + FModioTokenPackList(TArray TokenPackList); +}; + +#else + +/** +* Strong type struct to wrap multiple TokenPack indexed by a paged result +**/ +USTRUCT(NoExport, BlueprintType) +struct MODIO_API FModioTokenPackList +{ + /** + * A paged result property + **/ + UPROPERTY() + FModioPagedResult PagedResult; + + /** + * Arrray of Token Packs + **/ + UPROPERTY() + TArray InternalList; +}; + +#endif + +/** + * Struct to wrap ModInfoList into an optional parameter + **/ +USTRUCT(BlueprintType) +struct MODIO_API FModioOptionalTokenPackList +{ + GENERATED_BODY() + + /** + * Default constructor without parameters + **/ + FModioOptionalTokenPackList() = default; + + /** + * Constructor with a Token Pack list parameter to initialize an instance + * @param TokenPackList Optional value of a FModioTokenPackList + **/ + FModioOptionalTokenPackList(TOptional&& TokenPackList); + + /** + * Stored optional ModioTokenPackList + **/ + TOptional Internal; +}; \ No newline at end of file diff --git a/Source/Modio/Public/UI/ModioExampleLibrary.h b/Source/Modio/Public/UI/ModioExampleLibrary.h index d7b2c79a..359fcfba 100644 --- a/Source/Modio/Public/UI/ModioExampleLibrary.h +++ b/Source/Modio/Public/UI/ModioExampleLibrary.h @@ -39,7 +39,7 @@ class UModioExampleLibrary : public UBlueprintFunctionLibrary /** * @brief Runs a filter over the user's subscription list - * @param FilterParams Supports the following operations: + * @param FilterParams The filters to use on the user's subscription list * @param Callback Called when mod list has been processed **/ UFUNCTION(BlueprintCallable, Category = "mod.io|Example|UserSubscription") diff --git a/Source/ModioEditor/Private/DetailCustomizations/ModioBrowseModFileDetails.cpp b/Source/ModioEditor/Private/DetailCustomizations/ModioBrowseModFileDetails.cpp index 63423b71..31178bec 100644 --- a/Source/ModioEditor/Private/DetailCustomizations/ModioBrowseModFileDetails.cpp +++ b/Source/ModioEditor/Private/DetailCustomizations/ModioBrowseModFileDetails.cpp @@ -8,15 +8,16 @@ * */ -#include "../../Public/DetailCustomizations/ModioBrowseModFileDetails.h" -#include "../../Public/Widgets/SModioEditorWindowCompoundWidget.h" -#include -#include -#include -#include -#include -#include -#include +#include "DetailCustomizations/ModioBrowseModFileDetails.h" +#include "Widgets/SModioEditorWindowCompoundWidget.h" +#include "DetailCategoryBuilder.h" +#include "DetailCustomizations/SModFileRow.h" +#include "DetailLayoutBuilder.h" +#include "DetailWidgetRow.h" +#include "Objects/ModioBrowseModFileCollectionObject.h" +#include "Objects/ModioBrowseModFileObject.h" +#include "Misc/EngineVersionComparison.h" +#include "WindowManager.h" TSharedRef ModioBrowseModFileDetails::MakeInstance() { @@ -56,7 +57,6 @@ void ModioBrowseModFileDetails::DrawBrowseModFile(IDetailLayoutBuilder& DetailBu .WholeRowContent() [ SAssignNew(ListView, SListView>) - .ItemHeight(50) .SelectionMode(ESelectionMode::Multi) .ListItemsSource(&Source) .OnGenerateRow_Lambda([this](TSharedPtr Item, const TSharedRef& OwnerTable)->TSharedRef diff --git a/Source/ModioEditor/Private/DetailCustomizations/ModioBrowseModsParamsDetails.cpp b/Source/ModioEditor/Private/DetailCustomizations/ModioBrowseModsParamsDetails.cpp index 0806c4f9..3d4f843e 100644 --- a/Source/ModioEditor/Private/DetailCustomizations/ModioBrowseModsParamsDetails.cpp +++ b/Source/ModioEditor/Private/DetailCustomizations/ModioBrowseModsParamsDetails.cpp @@ -8,14 +8,15 @@ * */ -#include "../../Public/DetailCustomizations/ModioBrowseModsParamsDetails.h" +#include "DetailCustomizations/ModioBrowseModsParamsDetails.h" #include "Engine/StaticMesh.h" -#include "../../Public/Objects/ModioBrowseModsObject.h" +#include "Objects/ModioBrowseModsObject.h" #include "PropertyEditing.h" -#include -#include -#include -#include +#include "Framework/Application/SlateApplication.h" +#include "WindowManager.h" +#include "PropertyCustomizationHelpers.h" +#include "Widgets/Input/SSearchBox.h" +#include "Misc/EngineVersionComparison.h" #define LOCTEXT_NAMESPACE "ModioBrowseModsParamsDetails" @@ -86,7 +87,9 @@ void ModioBrowseModsParamsDetails::DrawEditMod(IDetailLayoutBuilder& DetailBuild .WholeRowContent() [ SAssignNew(ListView, SListView>) +#if UE_VERSION_OLDER_THAN(5, 5, 0) .ItemHeight(50) +#endif .ListItemsSource(&Source) .HeaderRow ( diff --git a/Source/ModioEditor/Private/DetailCustomizations/ModioCreateModFileParamsDetails.cpp b/Source/ModioEditor/Private/DetailCustomizations/ModioCreateModFileParamsDetails.cpp index 52b9b7e0..fb29867b 100644 --- a/Source/ModioEditor/Private/DetailCustomizations/ModioCreateModFileParamsDetails.cpp +++ b/Source/ModioEditor/Private/DetailCustomizations/ModioCreateModFileParamsDetails.cpp @@ -8,13 +8,13 @@ * */ -#include "../../Public/DetailCustomizations/ModioCreateModFileParamsDetails.h" -#include -#include -#include -#include -#include -#include +#include "DetailCustomizations/ModioCreateModFileParamsDetails.h" +#include "DetailLayoutBuilder.h" +#include "Objects/ModioCreateNewModFileParamsObject.h" +#include "Widgets/Input/SEditableTextBox.h" +#include "WindowManager.h" +#include "DetailCategoryBuilder.h" +#include "DetailWidgetRow.h" TSharedRef ModioCreateModFileParamsDetails::MakeInstance() { diff --git a/Source/ModioEditor/Private/DetailCustomizations/ModioCreateModParamsDetails.cpp b/Source/ModioEditor/Private/DetailCustomizations/ModioCreateModParamsDetails.cpp index d1619240..5ebc1d8a 100644 --- a/Source/ModioEditor/Private/DetailCustomizations/ModioCreateModParamsDetails.cpp +++ b/Source/ModioEditor/Private/DetailCustomizations/ModioCreateModParamsDetails.cpp @@ -8,23 +8,23 @@ * */ -#include "../../Public/DetailCustomizations/ModioCreateModParamsDetails.h" +#include "DetailCustomizations/ModioCreateModParamsDetails.h" #include "Engine/StaticMesh.h" -#include "../../Public/Objects/ModioCreateModParamsObject.h" +#include "Objects/ModioCreateModParamsObject.h" #include "PropertyEditing.h" #include "Widgets/Input/SCheckBox.h" #include "Misc/EngineVersionComparison.h" #if UE_VERSION_OLDER_THAN(5, 3, 0) -#include - #include +#include "DesktopPlatform/Public/IDesktopPlatform.h" + #include "DesktopPlatform/Public/DesktopPlatformModule.h" #else #include "DesktopPlatformModule.h" #include "IDesktopPlatform.h" #endif -#include -#include -#include -#include +#include "Framework/Application/SlateApplication.h" +#include "WindowManager.h" +#include "Widgets/Images/SThrobber.h" +#include "PropertyCustomizationHelpers.h" #define LOCTEXT_NAMESPACE "ModioCreateModParamsDetails" diff --git a/Source/ModioEditor/Private/DetailCustomizations/ModioEditModParamsDetails.cpp b/Source/ModioEditor/Private/DetailCustomizations/ModioEditModParamsDetails.cpp index b3717074..b4da44f4 100644 --- a/Source/ModioEditor/Private/DetailCustomizations/ModioEditModParamsDetails.cpp +++ b/Source/ModioEditor/Private/DetailCustomizations/ModioEditModParamsDetails.cpp @@ -8,23 +8,23 @@ * */ -#include "../../Public/DetailCustomizations/ModioEditModParamsDetails.h" +#include "DetailCustomizations/ModioEditModParamsDetails.h" #include "Engine/StaticMesh.h" -#include "../../Public/Objects/ModioEditModParamsObject.h" +#include "Objects/ModioEditModParamsObject.h" #include "PropertyEditing.h" #include "Widgets/Input/SCheckBox.h" #include "Misc/EngineVersionComparison.h" #if UE_VERSION_OLDER_THAN(5, 3, 0) -#include - #include +#include "DesktopPlatform/Public/IDesktopPlatform.h" + #include "DesktopPlatform/Public/DesktopPlatformModule.h" #else #include "DesktopPlatformModule.h" #include "IDesktopPlatform.h" #endif -#include -#include -#include -#include +#include "Framework/Application/SlateApplication.h" +#include "WindowManager.h" +#include "Widgets/Images/SThrobber.h" +#include "PropertyCustomizationHelpers.h" #define LOCTEXT_NAMESPACE "ModioCreateModParamsDetails" diff --git a/Source/ModioEditor/Private/DetailCustomizations/SModFileRow.cpp b/Source/ModioEditor/Private/DetailCustomizations/SModFileRow.cpp index 257a3c14..c051dbcc 100644 --- a/Source/ModioEditor/Private/DetailCustomizations/SModFileRow.cpp +++ b/Source/ModioEditor/Private/DetailCustomizations/SModFileRow.cpp @@ -8,12 +8,12 @@ * */ -#include "../../Public/DetailCustomizations/SModFileRow.h" -#include -#include +#include "DetailCustomizations/SModFileRow.h" +#include "Widgets/SModioEditorWindowCompoundWidget.h" +#include "WindowManager.h" #include "Misc/EngineVersionComparison.h" #if UE_VERSION_OLDER_THAN(5, 3, 0) - #include + #include "Launch/Resources/Version.h" #endif void SModFileRow::Construct(const FArguments& InArgs, const TSharedRef& InOwnerTableView) @@ -37,7 +37,7 @@ TSharedRef SModFileRow::GenerateWidgetForColumn(const FName& ColumnName return SNew(SHorizontalBox) +SHorizontalBox::Slot() - .Padding(FMargin(5.f, 10.f, 0.f, 0.f)) + .Padding(FMargin(5.f, 10.f, 0.f, 10.f)) [ SNew(STextBlock) .Text(FText::FromString(Item->Name)) @@ -48,7 +48,7 @@ TSharedRef SModFileRow::GenerateWidgetForColumn(const FName& ColumnName return SNew(SHorizontalBox) + SHorizontalBox::Slot() - .Padding(FMargin(5.f, 10.f, 0.f, 0.f)) + .Padding(FMargin(5.f, 10.f, 0.f, 10.f)) [ SNew(STextBlock) .Text(FText::FromString(PlatformString)) @@ -59,7 +59,7 @@ TSharedRef SModFileRow::GenerateWidgetForColumn(const FName& ColumnName return SNew(SHorizontalBox) + SHorizontalBox::Slot() - .Padding(FMargin(5.f, 10.f, 0.f, 0.f)) + .Padding(FMargin(5.f, 10.f, 0.f, 10.f)) [ SNew(STextBlock) .Text(FText::FromString(Item->Version)) @@ -70,7 +70,7 @@ TSharedRef SModFileRow::GenerateWidgetForColumn(const FName& ColumnName return SNew(SHorizontalBox) + SHorizontalBox::Slot() - .Padding(FMargin(5.f, 10.f, 0.f, 0.f)) + .Padding(FMargin(5.f, 10.f, 0.f, 10.f)) [ SNew(STextBlock) .Text(FText::FromString(Item->Status)) diff --git a/Source/ModioEditor/Private/ModioEditor.cpp b/Source/ModioEditor/Private/ModioEditor.cpp index aef22e33..be5d51f4 100644 --- a/Source/ModioEditor/Private/ModioEditor.cpp +++ b/Source/ModioEditor/Private/ModioEditor.cpp @@ -124,7 +124,7 @@ void FModioEditor::DisplayGettingStarted() void FModioEditor::PluginButtonClicked() { FModioInitializeOptions InitializeOptions = UModioSDKLibrary::GetProjectInitializeOptionsForSessionId(FString("ModioUnrealEditor")); - if (InitializeOptions.GameId.ToString().Equals("InvalidGameID") || InitializeOptions.ApiKey.ToString().Equals("InvalidApiKey")) + if (!InitializeOptions.GameId.IsValid() || !InitializeOptions.ApiKey.IsValid()) { const FText Title = FText::FromString("mod.io"); const FText Message = FText::FromString("Please specify a valid Game Id and Api Key in 'Project Settings->Plugins->mod.io'"); diff --git a/Source/ModioEditor/Private/Widgets/SModioEditorUserGamesList.cpp b/Source/ModioEditor/Private/Widgets/SModioEditorUserGamesList.cpp index bf2f08f9..feda125c 100644 --- a/Source/ModioEditor/Private/Widgets/SModioEditorUserGamesList.cpp +++ b/Source/ModioEditor/Private/Widgets/SModioEditorUserGamesList.cpp @@ -11,6 +11,7 @@ #include "Widgets/SModioEditorUserGamesList.h" #include "EngineMinimal.h" +#include "Misc/EngineVersionComparison.h" #include "Widgets/Images/SThrobber.h" #include "Widgets/SOverlay.h" @@ -145,7 +146,9 @@ void SModioEditorUserGamesList::DrawGameList() .DefaultLabel(LOCTEXT("GameListHeaderId", "ID")) ) .SelectionMode(ESelectionMode::Single) +#if UE_VERSION_OLDER_THAN(5, 5, 0) .ItemHeight(32) +#endif .ListItemsSource(&Games) .OnGenerateRow(this, &SModioEditorUserGamesList::GenerateGameInfoRow) .OnSelectionChanged(this, &SModioEditorUserGamesList::OnGameSelectedFromList) diff --git a/Source/ModioEditor/Private/WindowManager.cpp b/Source/ModioEditor/Private/WindowManager.cpp index fa6d9bff..2a61acbf 100644 --- a/Source/ModioEditor/Private/WindowManager.cpp +++ b/Source/ModioEditor/Private/WindowManager.cpp @@ -11,20 +11,20 @@ #include "WindowManager.h" #include "Misc/EngineVersionComparison.h" #if UE_VERSION_OLDER_THAN(5, 3, 0) - #include - #include + #include "DesktopPlatform/Public/IDesktopPlatform.h" + #include "DesktopPlatform/Public/DesktopPlatformModule.h" #else #include "DesktopPlatformModule.h" #include "IDesktopPlatform.h" #endif -#include -#include -#include -#include -#include -#include -#include -#include +#include "DetailCustomizations/ModioCreateModParamsDetails.h" +#include "DetailCustomizations/ModioBrowseModFileDetails.h" +#include "DetailCustomizations/ModioBrowseModsParamsDetails.h" +#include "DetailCustomizations/ModioCreateModFileParamsDetails.h" +#include "DetailCustomizations/ModioEditModParamsDetails.h" +#include "Objects/ModioCreateModParamsObject.h" +#include "Widgets/SWindow.h" +#include "Widgets/SModioEditorWindowCompoundWidget.h" TSharedPtr WindowManager::GetWindow() { diff --git a/Source/ModioEditor/Public/DetailCustomizations/SModFileRow.h b/Source/ModioEditor/Public/DetailCustomizations/SModFileRow.h index 16d47785..fa960719 100644 --- a/Source/ModioEditor/Public/DetailCustomizations/SModFileRow.h +++ b/Source/ModioEditor/Public/DetailCustomizations/SModFileRow.h @@ -14,8 +14,8 @@ #include "SlateFwd.h" #include "SlateBasics.h" #include "Widgets/DeclarativeSyntaxSupport.h" -#include -#include +#include "Objects/ModioBrowseModFileObject.h" +#include "Widgets/Views/STableRow.h" class MODIOEDITOR_API SModFileRow : public SMultiColumnTableRow> { diff --git a/Source/ModioEditor/Public/Objects/ModioBrowseModFileObject.h b/Source/ModioEditor/Public/Objects/ModioBrowseModFileObject.h index facc6ee6..aa9db436 100644 --- a/Source/ModioEditor/Public/Objects/ModioBrowseModFileObject.h +++ b/Source/ModioEditor/Public/Objects/ModioBrowseModFileObject.h @@ -11,7 +11,7 @@ #pragma once #include "CoreMinimal.h" -#include +#include "Types/ModioModInfo.h" #include "ModioBrowseModFileObject.generated.h" USTRUCT() diff --git a/Source/ModioEditor/Public/Objects/ModioBrowseModsObject.h b/Source/ModioEditor/Public/Objects/ModioBrowseModsObject.h index a2bec8ad..79fbeb05 100644 --- a/Source/ModioEditor/Public/Objects/ModioBrowseModsObject.h +++ b/Source/ModioEditor/Public/Objects/ModioBrowseModsObject.h @@ -11,7 +11,7 @@ #pragma once #include "CoreMinimal.h" -#include +#include "Types/ModioModInfo.h" #include "ModioBrowseModsObject.generated.h" diff --git a/Source/ModioEditor/Public/Objects/ModioCreateNewModFileParamsObject.h b/Source/ModioEditor/Public/Objects/ModioCreateNewModFileParamsObject.h index 3a8cec49..64e89960 100644 --- a/Source/ModioEditor/Public/Objects/ModioCreateNewModFileParamsObject.h +++ b/Source/ModioEditor/Public/Objects/ModioCreateNewModFileParamsObject.h @@ -11,7 +11,7 @@ #pragma once #include "CoreMinimal.h" -#include +#include "Types/ModioCommonTypes.h" #include "ModioCreateNewModFileParamsObject.generated.h" diff --git a/Source/ModioEditor/Public/Objects/ModioEditModParamsObject.h b/Source/ModioEditor/Public/Objects/ModioEditModParamsObject.h index 08f7506d..2d0d457e 100644 --- a/Source/ModioEditor/Public/Objects/ModioEditModParamsObject.h +++ b/Source/ModioEditor/Public/Objects/ModioEditModParamsObject.h @@ -11,7 +11,7 @@ #pragma once #include "CoreMinimal.h" -#include +#include "Types/ModioModInfo.h" #include "ModioEditModParamsObject.generated.h" @@ -33,7 +33,7 @@ class MODIOEDITOR_API UModioEditModParamsObject : public UObject UPROPERTY(EditAnywhere, Category = "Modio Edit Mod Params") FString NamePath; - UPROPERTY(meta = (DeprecatedProperty, DeprecationMessage = "Deprecated as of 2023.6 release. Please use the <> instead.")) + UPROPERTY(meta = (DeprecatedProperty, DeprecationMessage = "Deprecated as of 2023.6 release. Please use `EModioObjectVisibilityFlags Visibility` instead.")) bool bVisible_DEPRECATED; UPROPERTY(EditAnywhere, Category = "Modio Edit Mod Params") diff --git a/Source/ModioEditor/Public/Widgets/SModioEditorWindowCompoundWidget.h b/Source/ModioEditor/Public/Widgets/SModioEditorWindowCompoundWidget.h index 88284a35..229a113c 100644 --- a/Source/ModioEditor/Public/Widgets/SModioEditorWindowCompoundWidget.h +++ b/Source/ModioEditor/Public/Widgets/SModioEditorWindowCompoundWidget.h @@ -12,16 +12,16 @@ #include "CoreMinimal.h" #include "Widgets/SCompoundWidget.h" -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include +#include "Types/ModioGameInfo.h" +#include "Widgets/DeclarativeSyntaxSupport.h" +#include "Brushes/SlateDynamicImageBrush.h" +#include "Widgets/SBoxPanel.h" +#include "Widgets/Text/STextBlock.h" +#include "Widgets/Input/SEditableTextBox.h" +#include "Widgets/Images/SThrobber.h" +#include "Widgets/Input/SButton.h" +#include "Widgets/Notifications/SProgressBar.h" +#include "Widgets/Images/SImage.h" class UModioSubsystem; class UModioBrowseModsObject; diff --git a/Source/ModioEx/Public/Library/ModioSubmissionExtensionLibrary.h b/Source/ModioEx/Public/Library/ModioSubmissionExtensionLibrary.h index 09ca2c25..3a481d4a 100644 --- a/Source/ModioEx/Public/Library/ModioSubmissionExtensionLibrary.h +++ b/Source/ModioEx/Public/Library/ModioSubmissionExtensionLibrary.h @@ -24,7 +24,7 @@ class UModioSubmissionExtensionLibrary : public UBlueprintFunctionLibrary public: /** * @brief Queues the upload of a new mod file release for the specified mod, using the submitted parameters. This - * upload method accepts a a block of memory TArray rather than a file path. The upload's progress can be + * upload method accepts a block of memory `TArray` rather than a file path. The upload's progress can be * tracked in the same way as downloads; when completed, a Mod Management Event will be triggered with the result * code for the upload. * @param Target The ModioSubsystem object @@ -37,7 +37,7 @@ class UModioSubmissionExtensionLibrary : public UBlueprintFunctionLibrary /** * @brief Queues the upload of a new mod file release for the specified mod, using the submitted parameters. This upload method - * accepts a a block of memory TArray rather than a file path. The + * accepts a block of memory `TArray` rather than a file path. The * upload's progress can be tracked in the same way as downloads; when completed, a Mod Management Event will be * triggered with the result code for the upload. * @param Target The ModioSubsystem object diff --git a/Source/ThirdParty/NativeSDK b/Source/ThirdParty/NativeSDK index 3432e708..a0d800f0 160000 --- a/Source/ThirdParty/NativeSDK +++ b/Source/ThirdParty/NativeSDK @@ -1 +1 @@ -Subproject commit 3432e708a397d3786d7d52564d2943543d5fae1f +Subproject commit a0d800f06592878e15aa922b8a564ac1696efc4f