|
|
|
|
|
DefineClass.Achievement = { |
|
__parents = { "MsgReactionsPreset", }, |
|
__generated_by_class = "PresetDef", |
|
|
|
properties = { |
|
{ id = "display_name", name = "Display Name", |
|
editor = "text", default = false, translate = true, }, |
|
{ id = "description", name = "Description", |
|
editor = "text", default = false, translate = true, context = "(limited to 100 characters on XBOX)", }, |
|
{ id = "how_to", name = "How To", |
|
editor = "text", default = false, translate = true, context = "(limited to 100 characters on XBOX)", }, |
|
{ id = "image", name = "Image", |
|
editor = "ui_image", default = false, }, |
|
{ id = "secret", name = "Secret", |
|
editor = "bool", default = false, }, |
|
{ id = "target", name = "Target", |
|
editor = "number", default = 0, }, |
|
{ id = "time", name = "Time", |
|
editor = "number", default = 0, }, |
|
{ id = "save_interval", name = "Save Interval", |
|
editor = "number", default = false, }, |
|
{ category = "PS4", id = "ps4_trophy_group", name = "Trophy Group", |
|
editor = "preset_id", default = "Auto", preset_class = "TrophyGroup", extra_item = "Auto", }, |
|
{ category = "PS4", id = "ps4_used_trophy_group", name = "Used Trophy Group", |
|
editor = "preset_id", default = false, dont_save = true, read_only = true, no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, no_validate = true, preset_class = "TrophyGroup", }, |
|
{ category = "PS4", id = "ps4_duplicate", name = "Duplicate", |
|
editor = "buttons", default = false, dont_save = true, }, |
|
{ category = "PS4", id = "ps4_id", name = "Trophy Id", |
|
editor = "number", default = -1, no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, buttons = { {name = "Generate", func = "GenerateTrophyIDs"}, }, min = -1, max = 128, }, |
|
{ category = "PS4", id = "ps4_grade", name = "Grade", |
|
editor = "choice", default = "bronze", no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, items = function (self) return PlayStationTrophyGrades end, }, |
|
{ category = "PS4", id = "ps4_points", name = "Points", |
|
editor = "number", default = false, dont_save = true, read_only = true, no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, }, |
|
{ category = "PS4", id = "ps4_grouppoints", name = "Group Points", help = "Total sum for the base game should be 950 - 1050. For each expansion <= 200.", |
|
editor = "text", default = false, dont_save = true, read_only = true, no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, }, |
|
{ category = "PS4", id = "ps4_icon", name = "Icon", |
|
editor = "ui_image", default = "", dont_save = true, read_only = true, no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, |
|
no_validate = true, filter = "All files|*.png", }, |
|
{ category = "PS4", id = "ps4_platinum_linked", name = "Platinum Linked", |
|
editor = "bool", default = true, no_edit = function(self) local trophy_group_0 = GetTrophyGroupById("ps4", 0) |
|
return self:GetTrophyGroup("ps4") ~= trophy_group_0 or self.ps4_grade == "platinum" end, }, |
|
{ category = "PS5", id = "ps5_trophy_group", name = "Trophy Group", |
|
editor = "preset_id", default = "Auto", preset_class = "TrophyGroup", extra_item = "Auto", }, |
|
{ category = "PS5", id = "ps5_used_trophy_group", name = "Used Trophy Group", |
|
editor = "preset_id", default = false, dont_save = true, read_only = true, no_edit = function(self) return self:GetTrophyGroup("ps5") == "" end, no_validate = true, preset_class = "TrophyGroup", }, |
|
{ category = "PS5", id = "ps5_id", name = "Trophy Id", |
|
editor = "number", default = -1, no_edit = function(self) return self:GetTrophyGroup("ps5") == "" end, buttons = { {name = "Generate", func = "GenerateTrophyIDs"}, }, min = -1, max = 128, }, |
|
{ category = "PS5", id = "ps5_grade", name = "Grade", |
|
editor = "choice", default = "bronze", no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, items = function (self) return PlayStationTrophyGrades end, }, |
|
{ category = "PS5", id = "ps5_points", name = "Points", |
|
editor = "number", default = false, dont_save = true, read_only = true, no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, }, |
|
{ category = "PS5", id = "ps5_grouppoints", name = "Group Points", help = "Total sum for the base game should be 950 - 1050. For each expansion <= 200.", |
|
editor = "number", default = false, dont_save = true, read_only = true, no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, }, |
|
{ category = "PS5", id = "ps5_icon", name = "Icon", |
|
editor = "ui_image", default = "", dont_save = true, read_only = true, no_edit = function(self) return self:GetTrophyGroup("ps4") == "" end, |
|
no_validate = true, filter = "All files|*.png", }, |
|
{ category = "Xbox", id = "xbox_id", name = "Achievement Id", |
|
editor = "number", default = -1, }, |
|
{ category = "Steam", id = "steam_id", name = "Steam Id", help = "If not specified, the id of the preset will be used.", |
|
editor = "text", default = false, }, |
|
{ category = "Epic", id = "epic_id", name = "Epic Id", help = "If not specified, the id of the preset will be used.", |
|
editor = "text", default = false, }, |
|
{ category = "Epic", id = "flavor_text", name = "Flavor text", |
|
editor = "text", default = false, translate = true, }, |
|
}, |
|
HasSortKey = true, |
|
GlobalMap = "AchievementPresets", |
|
EditorMenubarName = "Achievements", |
|
EditorIcon = "CommonAssets/UI/Icons/top trophy winner.png", |
|
EditorMenubar = "Editors.Lists", |
|
} |
|
|
|
function Achievement:GetCompleteText() |
|
local unlocked = GetAchievementFlags(self.id) |
|
return unlocked and self.description or self.how_to |
|
end |
|
|
|
function Achievement:GetTrophyGroup(platform) |
|
|
|
local group_name_field = platform .. "_trophy_group" |
|
if self[group_name_field] ~= "Auto" then |
|
return self[group_name_field] |
|
end |
|
|
|
|
|
local trophies = PresetArray(Achievement, function(achievement) |
|
return achievement.SaveIn == self.save_in |
|
end) |
|
if #trophies ~= 0 then |
|
for i=#trophies,1,-1 do |
|
local group = trophies[i][group_name_field] |
|
if group ~= "" and group ~= "Auto" then |
|
return group |
|
end |
|
end |
|
end |
|
|
|
|
|
local group = FindPreset("TrophyGroup", self.save_in) |
|
if group then return group.id end |
|
|
|
|
|
local dlc = FindPreset("DLCConfig", self.save_in) |
|
local handling_field = platform .. "_handling" |
|
if dlc and dlc[handling_field] ~= "Embed" then |
|
|
|
|
|
return "" |
|
end |
|
|
|
return GetTrophyGroupById(platform, 0) |
|
end |
|
|
|
function Achievement:IsBaseGameTrophy(platform) |
|
local group = self:GetTrophyGroup(platform) |
|
return group ~= "" and TrophyGroupPresets[group]:IsBaseGameGroup(platform) |
|
end |
|
|
|
function Achievement:IsPlatinumLinked(platform) |
|
local trophy_group_0 = GetTrophyGroupById(platform, 0) |
|
local trophy_group = self:GetTrophyGroup(platform) |
|
local platinum_linked = trophy_group == trophy_group_0 and self[platform .. "_grade"] ~= "platinum" |
|
if platform == "ps4" then |
|
platinum_linked = platinum_linked and self.ps4_platinum_linked |
|
end |
|
return platinum_linked |
|
end |
|
|
|
function Achievement:IsCurrentlyUsed() |
|
return |
|
(Platform.steam and self.steam_id) or |
|
(Platform.epic and self.epic_id) or |
|
(Platform.ps4 and self.ps4_id >= 0) or |
|
(Platform.ps5 and self.ps5_id >= 0) or |
|
(Platform.xbox and self.xbox_id > 0) or |
|
(Platform.pc and (self.image or self.msg_reactions)) |
|
end |
|
|
|
function Achievement:GenerateTrophyIDs(root, prop_id) |
|
local platform = string.match(prop_id, "(.*)_id") |
|
local trophy_id_field = prop_id |
|
local group_id_field = platform .. "_gid" |
|
|
|
local trophies = PresetArray(Achievement, function(achievement) |
|
return achievement:GetTrophyGroup(platform) ~= "" |
|
end) |
|
|
|
local trophies_by_group = {} |
|
for _, trophy in ipairs(trophies) do |
|
local group = TrophyGroupPresets[trophy:GetTrophyGroup(platform)] |
|
local group_id = group[group_id_field] |
|
trophies_by_group[group_id] = trophies_by_group[group_id] or {} |
|
local group_trophies = trophies_by_group[group_id] |
|
group_trophies[#group_trophies + 1] = trophy |
|
end |
|
|
|
local trophy_id = 0 |
|
for _, group_trophies in sorted_pairs(trophies_by_group) do |
|
for _, trophy in ipairs(group_trophies) do |
|
if trophy[trophy_id_field] ~= trophy_id then |
|
trophy[trophy_id_field] = trophy_id |
|
trophy:MarkDirty() |
|
end |
|
trophy_id = trophy_id + 1 |
|
end |
|
end |
|
end |
|
|
|
function Achievement:Getps4_grouppoints() |
|
local group = self:GetTrophyGroup("ps4") |
|
local group_points = CalcTrophyGroupPoints(group, "ps4") |
|
local trophy_group_0 = GetTrophyGroupById("ps4", 0) |
|
if group == trophy_group_0 then |
|
local platinum_linked_points = CalcTrophyPlatinumLinkedPoints("ps4") |
|
if platinum_linked_points ~= group_points then |
|
return string.format("%d + %d", platinum_linked_points, group_points - platinum_linked_points) |
|
end |
|
end |
|
|
|
return tostring(group_points) |
|
end |
|
|
|
function Achievement:Getps4_used_trophy_group() |
|
return self:GetTrophyGroup("ps4") |
|
end |
|
|
|
function Achievement:Getps4_points() |
|
return TrophyGradesPlayStationPoints[self.ps4_grade] or 0 |
|
end |
|
|
|
function Achievement:Getps4_icon() |
|
local _, icon_path = GetPlayStationTrophyIcon(self, "ps4") |
|
return icon_path |
|
end |
|
|
|
function Achievement:Getps5_used_trophy_group() |
|
return self:GetTrophyGroup("ps4") |
|
end |
|
|
|
function Achievement:Getps5_points() |
|
return TrophyGradesPlayStationPoints[self.ps5_grade] or 0 |
|
end |
|
|
|
function Achievement:Getps5_grouppoints() |
|
return CalcTrophyGroupPoints(self:GetTrophyGroup("ps5"), "ps5") |
|
end |
|
|
|
function Achievement:Getps5_icon() |
|
local _, icon_path = GetPlayStationTrophyIcon(self, "ps5") |
|
return icon_path |
|
end |
|
|
|
function Achievement:GetError(platform) |
|
local errors = {} |
|
|
|
local ShouldTestPlatform = function(test_platform) |
|
return (not platform or platform == test_platform) |
|
end |
|
|
|
local trophies = PresetArray(Achievement) |
|
local GetPlayStationErrors = function(platform) |
|
local trophy_id_field = platform .. "_id" |
|
local self_trophy_id = self[trophy_id_field] |
|
if self.id == "PlatinumTrophy" and self_trophy_id ~= 0 then |
|
errors[#errors + 1] = string.format("%s platinum trophy's id must be 0!", string.upper(platform)) |
|
elseif self:GetTrophyGroup(platform) ~= "" and self_trophy_id < 0 then |
|
errors[#errors + 1] = string.format("Missing %s trophy id!", platform) |
|
elseif self_trophy_id >= 0 then |
|
table.sortby_field(trophies, trophy_id_field) |
|
local next_trophy_id = 0 |
|
local trophy_id_holes = {} |
|
for _, trophy in ipairs(trophies) do |
|
local curr_trophy_id = trophy[trophy_id_field] |
|
if next_trophy_id < self_trophy_id and curr_trophy_id >= 0 then |
|
if curr_trophy_id > next_trophy_id then |
|
if curr_trophy_id - next_trophy_id > 1 then |
|
trophy_id_holes[#trophy_id_holes + 1] = string.format("%d-%d", next_trophy_id, curr_trophy_id) |
|
else |
|
trophy_id_holes[#trophy_id_holes + 1] = next_trophy_id |
|
end |
|
end |
|
next_trophy_id = curr_trophy_id + 1 |
|
end |
|
if self ~= trophy and self_trophy_id == curr_trophy_id then |
|
errors[#errors + 1] = string.format("Duplicated %s trophy id (%s)!", platform, trophy.id) |
|
end |
|
end |
|
|
|
if #trophy_id_holes ~= 0 then |
|
errors[#errors + 1] = string.format("%s trophy ids are not consecutive, missing %s!", string.upper(platform), table.concat(trophy_id_holes, ", ")) |
|
end |
|
end |
|
end |
|
|
|
if ShouldTestPlatform("ps4") then GetPlayStationErrors("ps4") end |
|
if ShouldTestPlatform("ps5") then GetPlayStationErrors("ps5") end |
|
|
|
if ShouldTestPlatform("xbox_one") or ShouldTestPlatform("xbox_series") then |
|
if self.description and #TDevModeGetEnglishText(self.description) > 100 then |
|
errors[#errors + 1] = string.format("XBOX achievement description must be limited to 100 characters!") |
|
end |
|
if self.how_to and #TDevModeGetEnglishText(self.how_to) > 100 then |
|
errors[#errors + 1] = string.format("XBOX achievement how to must be limited to 100 characters!") |
|
end |
|
end |
|
|
|
return #errors ~= 0 and table.concat(errors, "\n") |
|
end |
|
|
|
function Achievement:GetWarning(platform) |
|
local warnings = {} |
|
|
|
local ShouldTestPlatform = function(test_platform) |
|
return (not platform or platform == test_platform) and self:GetTrophyGroup(test_platform) ~= "" |
|
end |
|
|
|
local GetPlayStationWarnings = function(platform) |
|
local is_placeholder, icon_path = GetPlayStationTrophyIcon(self, platform) |
|
if is_placeholder then |
|
warnings[#warnings + 1] = string.format("Missing %s trophy icon (placeholder used): %s", platform, icon_path) |
|
end |
|
|
|
local group = self:GetTrophyGroup(platform) |
|
local group_points = CalcTrophyGroupPoints(group, platform) |
|
local trophy_group_0 = GetTrophyGroupById(platform, 0) |
|
if trophy_group_0 == group then |
|
local platinum_linked_points = CalcTrophyPlatinumLinkedPoints(platform) |
|
local min, max = GetTrophyBaseGameNonPlatinumLinkedPointsRange(platform) |
|
local non_platinum_linked_points = group_points - platinum_linked_points |
|
if non_platinum_linked_points < min or max < non_platinum_linked_points then |
|
warnings[#warnings + 1] = string.format( |
|
"%s non platinum linked trophy points sum in base group is not between %d and %d.", |
|
string.upper(platform), min, max |
|
) |
|
end |
|
group_points = platinum_linked_points |
|
end |
|
|
|
local min, max = GetTrophyGroupPointsRange(group, platform) |
|
if group_points < min or max < group_points then |
|
warnings[#warnings + 1] = string.format( |
|
"%s trophy group points sum is not between %d and %d.", |
|
string.upper(platform), min, max |
|
) |
|
end |
|
end |
|
|
|
if ShouldTestPlatform("ps4") then GetPlayStationWarnings("ps4") end |
|
if ShouldTestPlatform("ps5") then GetPlayStationWarnings("ps5") end |
|
|
|
if ShouldTestPlatform("ps5") then |
|
if #self.id > 32 then |
|
warnings[#warnings + 1] = string.format("Trophy Id maximum length is 32 (< %d)!", #self.id) |
|
end |
|
if string.find(self.id, "[^%w]") then |
|
warnings[#warnings + 1] = "Trophy Id contains non-alphanumeric characters!" |
|
end |
|
end |
|
|
|
return #warnings ~= 0 and table.concat(warnings, "\n") |
|
end |
|
|
|
function Achievement:SaveAll(...) |
|
ForEachPreset(Achievement, function(trophy) |
|
if trophy:GetTrophyGroup("ps4") == "" then |
|
trophy:MarkDirty() |
|
trophy.ps4_id = -1 |
|
end |
|
if trophy:GetTrophyGroup("ps5") == "" then |
|
trophy:MarkDirty() |
|
trophy.ps5_id = -1 |
|
end |
|
end) |
|
Preset.SaveAll(self, ...) |
|
end |
|
|
|
DefineClass.DLCConfig = { |
|
__parents = { "Preset", }, |
|
__generated_by_class = "PresetDef", |
|
|
|
properties = { |
|
{ category = "General", id = "display_name", name = "Display Name", |
|
editor = "text", default = false, translate = true, }, |
|
{ category = "General", id = "description", name = "Description", |
|
editor = "text", default = false, translate = true, }, |
|
{ category = "General", id = "required_lua_revision", |
|
editor = "number", default = 237259, }, |
|
{ category = "General", id = "pre_load", |
|
editor = "func", default = function (self) |
|
if not IsDlcOwned(self) then |
|
return "remove" |
|
end |
|
end, }, |
|
{ category = "General", id = "load_anyway", name = "Enable Loading Saves When Missing", help = "Set to true if you want to be able to load a save with this dlc missing. If the dlc was deleted, but still present in the save's metadata - it is consider as load_anyway = true (Developer only)", |
|
editor = "bool", default = true, }, |
|
{ category = "General", id = "show_in_reports", name = "Show in Reports", |
|
editor = "bool", default = false, }, |
|
{ category = "General", id = "post_load", |
|
editor = "func", default = function (self) |
|
g_AvailableDlc[self.name] = true |
|
end, }, |
|
{ category = "Build Steam", id = "steam_dlc_id", name = "Steam DLC Id", |
|
editor = "number", default = false, }, |
|
{ category = "Build Pops", id = "pops_dlc_id", name = "Pops DLC Id", |
|
editor = "text", default = false, }, |
|
{ category = "Build Epic", id = "epic_dlc_id", name = "Artifact Id (Dev)", help = "Where the DLC pack will be pushed to. Can be looked up in the EpicGames dev portal.", |
|
editor = "text", default = false, }, |
|
{ category = "Build Epic", id = "epic_catalog_dlc_id", name = "Catalog Id (Live)", help = "Which catalog item (seen in the Epic Launcher) the game will check ownership for. Can be in looked up in the .mancpn files after downloading from the Epic Launcher (AppName corresponds to ArtifactId, we need CatalogItemId of the live item).", |
|
editor = "text", default = false, }, |
|
{ category = "Build", id = "generate_build_rule", name = "Generate Build Rule", help = "With name Dlc%Id%", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "deprecated", name = "Deprecated", help = "Used only for compatibility", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "dont_localize", name = "Dont localize contents", help = "Skip the DLC contents when running localization", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "localization", name = "Has localization packs", help = "If the Dlc should include latest localization packfiles", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "generate_art_folders", |
|
editor = "buttons", default = false, buttons = { {name = "Locate PS4 Art", func = "LocatePS4Art"}, {name = "LocateXboxArt", func = "LocateXboxArt"}, }, }, |
|
{ category = "Build", id = "ext_name", help = "(optional) name of executables", |
|
editor = "text", default = false, }, |
|
{ category = "Build", id = "lua", help = "If the Dlc should include Lua.hpk", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "data", help = "If the Dlc should include Data.hpk", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "nonentitytextures", help = "If the Dlc should include all post release non-entity textures, texture lists and bin assets", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "entitytextures", help = "If the Dlc should include all post release entity textures and texture lists", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "ui", help = "If the Dlc should include UI.hpk", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "shaders", help = "If the Dlc should include lastest shader packs.", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "sounds", help = "If the Dlc should include the latest Sounds.hpk", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "resource_metadata", help = "Generate ressource metadata", |
|
editor = "bool", default = true, }, |
|
{ category = "Build", id = "content_dep", help = "List of rules to be build before the content.hpk is packed (e.g. BinAssets)", |
|
editor = "prop_table", default = {}, }, |
|
{ category = "Build", id = "content_files", help = "Files added to the dlc content.hpk (e.g. PatchTextures.hpk, etc.)", |
|
editor = "nested_list", default = false, base_class = "DLCConfigContentFile", inclusive = true, }, |
|
{ category = "BuildPS4", id = "ps4_handling", name = "Handling", help = "Enable - Creates a full dlc package\nEmbed - Creates only a .hpk to be embedded in the main game package\nEmbedPatch - Creates a .hpk that gets embedded in the main game package and replaces an old dlc\nExclude - Not shipped on this platform", |
|
editor = "combo", default = "Exclude", items = function (self) return { "Enable", "Embed", "EmbedPatch", "Exclude" } end, }, |
|
{ category = "BuildPS4", id = "ps4_label", name = "Label", |
|
editor = "text", default = false, no_edit = function(self) return self.ps4_handling ~= "Enable" and self.ps4_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildPS4", id = "ps4_version", name = "Version", |
|
editor = "text", default = "01.00", no_edit = function(self) return self.ps4_handling ~= "Enable" and self.ps4_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildPS4", id = "ps4_entitlement_key", name = "Entitlement Key", help = "Unique 16 byte key. Will be automatically generated. Must be kept secret and not regenerated after certification.", |
|
editor = "text", default = false, read_only = true, no_edit = function(self) return self.ps4_handling ~= "Enable" and self.ps4_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildPS5", id = "ps5_handling", name = "Handling", help = "Enable - Creates a full dlc package\nEmbed - Creates only a .hpk to be embedded in the main game package\nEmbedPatch - Creates a .hpk that gets embedded in the main game package and replaces an old dlc\nExclude - Not shipped on this platform", |
|
editor = "combo", default = "Exclude", items = function (self) return { "Enable", "Embed", "EmbedPatch", "Exclude" } end, }, |
|
{ category = "BuildPS5", id = "ps5_label", name = "Label", |
|
editor = "text", default = false, no_edit = function(self) return self.ps5_handling ~= "Enable" and self.ps5_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildPS5", id = "ps5_master_version", name = "Master Version", |
|
editor = "text", default = "01.00", no_edit = function(self) return self.ps5_handling ~= "Enable" and self.ps5_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildPS5", id = "ps5_content_version", name = "Content Version", |
|
editor = "text", default = "01.000.000", no_edit = function(self) return self.ps5_handling ~= "Enable" and self.ps5_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildPS5", id = "ps5_entitlement_key", name = "Entitlement Key", help = "Unique 16 byte key. Will be automatically generated. Must be kept secret and not regenerated after certification.", |
|
editor = "text", default = false, read_only = true, no_edit = function(self) return self.ps5_handling ~= "Enable" and self.ps5_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildXbox", id = "xbox_handling", name = "Handling", help = "Enable - Creates a full dlc package\nEmbed - Creates only a .hpk to be embedded in the main game package\nEmbedPatch - Creates a .hpk that gets embedded in the main game package and replaces an old dlc\nExclude - Not shipped on this platform", |
|
editor = "combo", default = "Exclude", items = function (self) return { "Enable", "Embed", "EmbedPatch", "Exclude" } end, }, |
|
{ category = "BuildXbox", id = "xbox_name", |
|
editor = "text", default = false, no_edit = function(self) return self.xbox_handling ~= "Enable" and self.xbox_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildXbox", id = "xbox_store_id", |
|
editor = "text", default = false, no_edit = function(self) return self.xbox_handling ~= "Enable" and self.xbox_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildXbox", id = "xbox_display_name", |
|
editor = "text", default = false, no_edit = function(self) return self.xbox_handling ~= "Enable" and self.xbox_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildXbox", id = "xbox_identity", |
|
editor = "text", default = false, no_edit = function(self) return self.xbox_handling ~= "Enable" and self.xbox_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildXbox", id = "xbox_version", |
|
editor = "text", default = "1.0.0.0", no_edit = function(self) return self.xbox_handling ~= "Enable" and self.xbox_handling ~= "EmbedPatch" end, }, |
|
{ category = "BuildWindowsStore", id = "ws_identity_name", |
|
editor = "text", default = false, }, |
|
{ category = "BuildWindowsStore", id = "ws_version", |
|
editor = "text", default = "1.0.0.0", }, |
|
{ category = "BuildWindowsStore", id = "ws_store_id", |
|
editor = "text", default = false, }, |
|
{ id = "SaveIn", |
|
editor = "text", default = false, read_only = true, no_edit = true, }, |
|
{ category = "Build", id = "public", help = "information from/about this DLC can be made public", |
|
editor = "bool", default = false, }, |
|
{ category = "Build", id = "split_files", help = "These files will be split and added to the DLC.", |
|
editor = "string_list", default = {}, item_default = "", items = false, arbitrary_value = true, }, |
|
}, |
|
HasCompanionFile = true, |
|
SingleFile = false, |
|
EditorMenubarName = "DLC config", |
|
EditorIcon = "CommonAssets/UI/Icons/add buy cart plus.png", |
|
EditorMenubar = "DLC", |
|
save_in = "future", |
|
} |
|
|
|
function DLCConfig:GetEditorView() |
|
local str = self.id |
|
if self.generate_build_rule then |
|
str = str .. " <color 0 128 128>build</color>" |
|
end |
|
if self.deprecated then |
|
str = str .. " <color 128 128 0>deprecated</color>" |
|
end |
|
if self.Comment ~= "" then |
|
str = str .. " <color 0 128 0>" .. self.Comment .. "</color>" |
|
end |
|
return str |
|
end |
|
|
|
function DLCConfig:LocatePS4Art(root) |
|
local folder = "svnAssets/Source/ps4/" .. root.id .. "/" |
|
local files = { "icon0.png" } |
|
if not io.exists(folder) then |
|
io.createpath(folder) |
|
end |
|
for _, file in ipairs(files) do |
|
local path = folder .. file |
|
if not io.exists(path) then |
|
CopyFile("CommonAssets/Images/Achievements/PS4/ICON0.PNG", path) |
|
end |
|
end |
|
OS_LocateFile(folder) |
|
end |
|
|
|
function DLCConfig:LocateXboxArt(root) |
|
local folder = "svnAssets/Source/xbox/" .. root.id .. "/" |
|
local files = { "Logo.png", "SmallLogo.png", "WideLogo.png" } |
|
if not io.exists(folder) then |
|
io.createpath(folder) |
|
end |
|
for _, file in ipairs(files) do |
|
local path = folder .. file |
|
if not io.exists(path) then |
|
CopyFile("CommonAssets/Images/Achievements/PS4/ICON0.PNG", path) |
|
end |
|
end |
|
OS_LocateFile(folder) |
|
end |
|
|
|
function DLCConfig:GetCompanionFileSavePath(save_path) |
|
local dlc_id = string.match(save_path, "(%w+)%.lua") |
|
assert(dlc_id) |
|
if not dlc_id then dlc_id = "unknown" end |
|
return "svnProject/Dlc/" .. self.id .. "/autorun.lua" |
|
end |
|
|
|
function DLCConfig:GenerateCompanionFileCode(code) |
|
|
|
local autorun_template = { |
|
name = self.id, |
|
deprecated = self.deprecated or nil, |
|
display_name = self.display_name, |
|
required_lua_revision = self.required_lua_revision, |
|
ps4_trophy_group_description = self.ps4_trophy_group_description, |
|
ps5_trophy_group_description = self.ps5_trophy_group_description, |
|
steam_dlc_id = self.steam_dlc_id, |
|
pops_dlc_id = self.pops_dlc_id, |
|
epic_dlc_id = self.epic_dlc_id, |
|
epic_catalog_dlc_id = self.epic_catalog_dlc_id, |
|
ps4_label = self.ps4_label, |
|
ps5_label = self.ps5_label, |
|
xbox_store_id = self.xbox_store_id, |
|
ps4_gid = self.ps4_gid, |
|
ps5_gid = self.ps5_gid, |
|
pre_load = self.pre_load, |
|
post_load = self.post_load, |
|
} |
|
code:append("return ") |
|
code:append(TableToLuaCode(autorun_template)) |
|
end |
|
|
|
function DLCConfig:SaveAll(...) |
|
local class = self.PresetClass or self.class |
|
local dlcs = PresetArray(class) |
|
|
|
local PlayStationGenerateEntitlementKeys = function(additional_contents, platform) |
|
local handling = platform .. "_handling" |
|
local entitlement_key = platform .. "_entitlement_key" |
|
|
|
local used_entitlement_keys = {} |
|
for _, additional_content in ipairs(additional_contents) do |
|
if additional_content[entitlement_key] then |
|
used_entitlement_keys[additional_content[entitlement_key]] = true |
|
end |
|
end |
|
|
|
for _, additional_content in ipairs(additional_contents) do |
|
if additional_content[handling] == "Enable" and not additional_content[entitlement_key] then |
|
repeat |
|
additional_content[entitlement_key] = random_hex(128) |
|
until not used_entitlement_keys[additional_content[entitlement_key]] |
|
used_entitlement_keys[additional_content[entitlement_key]] = true |
|
additional_content:MarkDirty() |
|
end |
|
end |
|
end |
|
|
|
PlayStationGenerateEntitlementKeys(dlcs, "ps4") |
|
PlayStationGenerateEntitlementKeys(dlcs, "ps5") |
|
|
|
Preset.SaveAll(self, ...) |
|
|
|
local epic_ids = {} |
|
ForEachPreset(class, function(preset, group) |
|
local epic_catalog_dlc_id = preset.epic_catalog_dlc_id |
|
if (epic_catalog_dlc_id or "") ~= "" then |
|
epic_ids[#epic_ids + 1] = epic_catalog_dlc_id |
|
end |
|
end) |
|
local text = string.format("%sg_EpicDlcIds = %s", exported_files_header_warning, TableToLuaCode(epic_ids)) |
|
local path = "svnProject/Lua/EpicDlcIds.lua" |
|
local err = SaveSVNFile(path, text) |
|
if err then |
|
printf("Failed to save %s: %s", path, err); |
|
end |
|
end |
|
|
|
|
|
|
|
DefineClass.DLCConfigContentFile = { |
|
__parents = { "PropertyObject" }, |
|
properties = { |
|
{ id = "Source", editor = "text", default = ""}, |
|
{ id = "Destination", editor = "text", default = ""}, |
|
}, |
|
} |
|
|
|
DefineClass.GradingLUTSource = { |
|
__parents = { "Preset", }, |
|
__generated_by_class = "PresetDef", |
|
|
|
properties = { |
|
{ category = "Input", id = "src_path", name = "Path", |
|
editor = "browse", default = false, folder = "svnAssets/Source/Textures/LUTs", filter = "LUT (*.cube)|*.cube", force_extension = ".cube", }, |
|
{ category = "Input", id = "display_name", name = "Display name", |
|
editor = "text", default = false, translate = true, }, |
|
{ category = "Output", id = "size", name = "Size", |
|
editor = "number", default = false, dont_save = true, read_only = true, }, |
|
{ category = "Output", id = "color_space", name = "Color Space", |
|
editor = "text", default = false, dont_save = true, read_only = true, }, |
|
{ category = "Output", id = "color_gamma", name = "Color Gamma", |
|
editor = "text", default = false, dont_save = true, read_only = true, }, |
|
{ category = "Output", id = "dst_path", name = "Path", |
|
editor = "text", default = false, dont_save = true, read_only = true, buttons = { {name = "Locate", func = "LUT_LocateFile"}, }, }, |
|
}, |
|
HasSortKey = true, |
|
GlobalMap = "GradingLUTs", |
|
EditorMenubarName = "Grading LUTs", |
|
EditorMenubar = "Editors.Art", |
|
dst_dir = "Textures/LUTs/", |
|
} |
|
|
|
DefineModItemPreset("GradingLUTSource", { EditorName = "Photo Mode - Grading LUT", EditorSubmenu = "Other" }) |
|
|
|
function GradingLUTSource:OnPreSave() |
|
if self:IsDirty() or self:IsDataDirty() then |
|
self:OnSrcChange() |
|
end |
|
end |
|
|
|
function GradingLUTSource:Getsize() |
|
return hr.ColorGradingLUTSize |
|
end |
|
|
|
function GradingLUTSource:GetDstDir() |
|
if self:IsModItem() then |
|
return self.mod.content_path .. self.dst_dir |
|
end |
|
return self.dst_dir |
|
end |
|
|
|
function GradingLUTSource:Getcolor_space() |
|
return GetColorSpaceName(hr.ColorGradingLUTColorSpace) |
|
end |
|
|
|
function GradingLUTSource:Getcolor_gamma() |
|
return GetColorGammaName(hr.ColorGradingLUTColorGamma) |
|
end |
|
|
|
function GradingLUTSource:OnSrcChange() |
|
CreateRealTimeThread(function(self) |
|
local dst_dir = self:GetDstDir() |
|
if not io.exists(dst_dir) then |
|
local err = AsyncCreatePath(dst_dir) |
|
if err then |
|
print(string.format("Could not create path %s: err", dst_dir, err)) |
|
end |
|
if not self:IsModItem() then |
|
SVNAddFile(dst_dir) |
|
end |
|
end |
|
|
|
local dst_path = self:Getdst_path() |
|
ImportColorGradingLUT(self:Getsize(), dst_path, self.src_path) |
|
|
|
Sleep(3000) |
|
if not self:IsModItem() then |
|
SVNAddFile(dst_path) |
|
SVNAddFile(self.src_path) |
|
end |
|
end, self) |
|
end |
|
|
|
function GradingLUTSource:Getdst_path() |
|
return self:GetResourcePath() |
|
end |
|
|
|
function GradingLUTSource:GetResourcePath() |
|
return string.format("%s%s.dds", self:GetDstDir(), self.id) |
|
end |
|
|
|
function GradingLUTSource:GetError() |
|
local errors = {} |
|
if not self.src_path then |
|
errors[#errors + 1] = "Missing input path." |
|
elseif not io.exists(self.src_path) then |
|
errors[#errors + 1] = "Invalid input path." |
|
end |
|
return #errors ~= 0 and table.concat(errors, "\n") |
|
end |
|
|
|
function GradingLUTSource:GetDisplayName() |
|
return self.display_name |
|
end |
|
|
|
function GradingLUTSource:GetEditorView() |
|
if self:GetDisplayName() then |
|
return self.id .. ' <color 128 90 30>"' .. _InternalTranslate(self:GetDisplayName()) .. '"' |
|
else |
|
return self.id |
|
end |
|
end |
|
|
|
function GradingLUTSource:IsModItem() |
|
return config.Mods and self:IsKindOf("ModItem") |
|
end |
|
|
|
function GradingLUTSource:IsDataDirty() |
|
if not IsFSUnpacked() and not self:IsModItem() then |
|
return false |
|
end |
|
|
|
local src_timestamp, src_err = io.getmetadata(self.src_path, "modification_time") |
|
if src_err then |
|
print(string.format("[GradingLUTs] Failed checking %s for modification: %s", self.src_path, src_err)) |
|
return false |
|
end |
|
|
|
local dst_timestamp, dst_err = io.getmetadata(self:Getdst_path(), "modification_time") |
|
return dst_err or src_timestamp > dst_timestamp |
|
end |
|
|
|
|
|
|
|
if Platform.pc and Platform.developer then |
|
|
|
function CleanGradingLUTsDir(luts_dir) |
|
local err, processed_luts = AsyncListFiles(luts_dir, "*.dds", "relative") |
|
if err then |
|
print(string.format("[GradingLUTs] Failed listing processed LUTs: %s", err)) |
|
end |
|
for _,lut in pairs(GradingLUTs) do |
|
if lut:GetDstDir() == luts_dir then |
|
table.remove_entry(processed_luts, lut.id .. ".dds") |
|
end |
|
end |
|
for _,lut in ipairs(processed_luts) do |
|
local lut_path = luts_dir .. lut |
|
local err = AsyncFileDelete(lut_path) |
|
if err then |
|
print(string.format("[GradingLUTs] Failed deleting %s: %s", lut_path, err)) |
|
elseif luts_dir == GradingLUTSource.dst_dir then |
|
SVNDeleteFile(lut_path) |
|
end |
|
end |
|
end |
|
|
|
function CleanGradingLUTsDirs() |
|
if not IsFSUnpacked() then |
|
CleanGradingLUTsDir(GradingLUTSource.dst_dir) |
|
end |
|
for _, mod in ipairs(ModsList) do |
|
CleanGradingLUTsDir(mod.content_path .. GradingLUTSource.dst_dir) |
|
end |
|
end |
|
|
|
function OnMsg.PresetSave(class) |
|
if IsKindOf(class, "GradingLUTSource") then |
|
CleanGradingLUTsDirs() |
|
end |
|
end |
|
|
|
function OnMsg.DataLoaded() |
|
for _,lut in pairs(GradingLUTs) do |
|
if lut:IsDataDirty() then |
|
lut:OnSrcChange() |
|
end |
|
end |
|
CleanGradingLUTsDirs() |
|
end |
|
|
|
function LUT_LocateFile(preset) |
|
OS_LocateFile(preset:Getdst_path()) |
|
end |
|
|
|
end |
|
|
|
DefineClass.PlayStationActivities = { |
|
__parents = { "MsgReactionsPreset", }, |
|
__generated_by_class = "PresetDef", |
|
|
|
properties = { |
|
{ id = "title", name = "Title", help = "The name of the challenge. This field can be localized.", |
|
editor = "text", default = false, translate = true, }, |
|
{ id = "description", name = "Description", help = "The description of the challenge. This field can be localized.", |
|
editor = "text", default = false, translate = true, wordwrap = true, lines = 3, max_lines = 10, }, |
|
{ id = "_openEndedHelp", help = "An open-ended activity has no specific completion objective. It ends when the player chooses to end it. For example, batting practice in MLB® The Show™, build mode in Dreams, or realms in God of War.\n\nOpen-ended activities can contain tasks and subtasks that can be used to track optional objectives within the activity.\n\nThe system handles open-ended activities like progress activities, but when an open-ended activity ends, any result sent is ignored.\n\nJust like progress activities, results for single-player activities can be passed using UDS events. You must use the matches API to pass results for open-ended activities that are being played in multiplayer scenarios.", |
|
editor = "help", default = false, dont_save = true, read_only = true, no_edit = function(self) return self.category ~= "openEnded" end, }, |
|
{ id = "_progressHelp", help = "A progress activity is defined as any activity that requires the player to complete an objective or series of objectives in order to complete the activity. For example, chapters in Uncharted, or quests in Horizon Zero Dawn.\n\nProgress activities can optionally contain tasks and subtasks that players can use to understand what they should do next and track how close they are to completing an activity. See Tasks and Subtasks for more information on how these can be used.\n\nProgress activities must have a result when ended. For single-player progress activities, the game can set the result to COMPLETED, FAILED, or ABANDONED and pass this result back to the platform by means of the UDS activityEnd event. If you end a progress activity as COMPLETED or FAILED, it is written into the player's history and progress is reset for the next instance.\n\nNote:\nThese outcomes are automatically tagged on any publicly available UGC that is created. In the case of successful completion, this UGC can be surfaced to other players who are at the same point in the game as a form of help or walkthrough.\n\nFor the multiplayer match case, SUCCESS or FAILED are the only supported results. You must use the matches API to pass these results. If you want to make an activity no longer active while retaining its progress, you must move the match to ONHOLD through its status property.", |
|
editor = "help", default = false, dont_save = true, read_only = true, no_edit = function(self) return self.category ~= "progress" end, }, |
|
{ id = "category", name = "Category", |
|
editor = "choice", default = "openEnded", items = function (self) return { "openEnded", "progress" } end, }, |
|
{ id = "default_playtime_estimate", name = "Default Playtime Estimate (minutes)", help = 'The default playtime estimate is displayed in System UI when the system has not determined the estimated playtime. Once the system determines the estimated playtime, the value may be switched over from the default playtime that is specified. You can specify the time in minute at the activity, task and subtask level.\n• When the category is not "challenge", allow the value at 5-minute intervals. (e.g. 5, 10, 15);\n• When the category is "challenge", allow the value at 1-minute intervals (e.g. 1, 2, 3);\n• When the type is "task" or "subTask", allow the value at 1-minute intervals (e.g. 1, 2, 3).', |
|
editor = "number", default = false, step = 5, min = 0, }, |
|
{ id = "available_by_default", name = "Available By Default", help = 'When set to true, this automatically sets the availability of an activity to available. Use this for any activity that the player can play from the very first time they launch the game. For players who have the Spoiler Warning set to warn on "Everything You Haven\'t Seen Yet", this setting instructs the Spoiler service to ignore this activity as containing any spoilers, even when it hasn\'t yet been seen by the user.', |
|
editor = "bool", default = true, }, |
|
{ id = "hidden_by_default", name = "Hidden By Default", help = "When set to true, this activity, task, or subtask is considered a spoiler throughout the UX of the platform, until it becomes available, started, or ended for the player. This means that players see a spoiler flag on any user-generated content containing this activity, task, or subtask if they have not encountered it in the game yet. Additionally, if a friend is playing a hidden activity that the player hasn't encountered yet, the card is obscured for the player when viewed on the friend's profile.", |
|
editor = "bool", default = false, }, |
|
{ id = "is_required_for_completion", name = "Required For Completion", help = "This is used to determine if the player must complete the activity to complete the main story and to pass the activities TRC if your game has a main story. Primarily, this is used to determine the sorting of activities, as activities with isRequiredforCompletion set to true that the player has never completed are more likely to be suggested to the player. In addition, this can be set on tasks. When completed, those tasks are treated as part of the progress of the activity, ultimately controlling the completion percentage progress bars. If set to false, then tasks are ignored in the completion percentage progress bar giving you more granular control of those bars. You cannot set this value on subtasks. All subtasks are considered required for completion.", |
|
editor = "bool", default = false, no_edit = function(self) return self.category ~= "progress" end, }, |
|
{ id = "abandon_on_done_map", name = "Abandon on DoneMap", |
|
editor = "bool", default = false, }, |
|
{ id = "state", name = "Is Active", |
|
editor = "bool", default = false, dont_save = true, read_only = true, buttons = { {name = "Start", func = "Start"}, {name = "Abandon", func = "Abandon"}, {name = "Complete", func = "Complete"}, {name = "Fail", func = "Fail"}, }, }, |
|
{ id = "_test_buttons", |
|
editor = "buttons", default = false, buttons = { {name = "Test Launch Now", func = "Launch"}, {name = "Test Launch On Next Boot", func = "DbgLaunchOnBoot"}, }, }, |
|
{ id = "Launch", name = "Launch", |
|
editor = "func", default = function (self) end, }, |
|
{ id = "fullscreen_image", name = "Fullscreen Image", help = "• Dimension : 3840x2160 px\n• Image Format : PNG\n• 24-bit non-Interlaced\n• Full screen image used", |
|
editor = "ui_image", default = false, dont_save = true, read_only = true, no_validate = true, }, |
|
{ id = "card_image", name = "Card Image", help = "Image used on action cards representing the game or challenge and in notifications triggered for a challenge.\n• Dimension : 864x1040 px\n• Image Format : PNG\n• 24 bit non-Interlaced", |
|
editor = "ui_image", default = false, dont_save = true, read_only = true, no_validate = true, }, |
|
}, |
|
GlobalMap = "ActivitiesPresets", |
|
EditorMenubarName = "PlayStation Activities", |
|
EditorMenubar = "Editors.Other", |
|
} |
|
|
|
function PlayStationActivities:Getfullscreen_image() |
|
return string.format("svnAssets/Source/Images/Activities/%s_fullscreen.png", self.id) |
|
end |
|
|
|
function PlayStationActivities:Getcard_image() |
|
return string.format("svnAssets/Source/Images/Activities/%s_card.png", self.id) |
|
end |
|
|
|
function PlayStationActivities:Start() |
|
if Platform.ps5 then |
|
AsyncPlayStationActivityStart(self.id) |
|
end |
|
AccountStorage.PlayStationStartedActivities[self.id] = true |
|
SaveAccountStorage(5000) |
|
end |
|
|
|
function PlayStationActivities:DbgLaunchOnBoot() |
|
if not Platform.ps5 then |
|
AccountStorage.PlayStationActivityDbgLaunchOnBoot = self.id |
|
SaveAccountStorage(1000) |
|
end |
|
end |
|
|
|
function PlayStationActivities:IsActive() |
|
return AccountStorage.PlayStationStartedActivities[self.id] |
|
end |
|
|
|
function PlayStationActivities:Getstate() |
|
return AccountStorage.PlayStationStartedActivities[self.id] |
|
end |
|
|
|
function PlayStationActivities:Complete() |
|
if Platform.ps5 then |
|
AsyncPlayStationActivityEnd(self.id, const.PlayStationActivityOutcomeCompleted) |
|
end |
|
AccountStorage.PlayStationStartedActivities[self.id] = nil |
|
SaveAccountStorage(5000) |
|
end |
|
|
|
function PlayStationActivities:Fail() |
|
if Platform.ps5 then |
|
AsyncPlayStationActivityEnd(self.id, const.PlayStationActivityOutcomeFailed) |
|
end |
|
AccountStorage.PlayStationStartedActivities[self.id] = nil |
|
SaveAccountStorage(5000) |
|
end |
|
|
|
function PlayStationActivities:Abandon() |
|
if Platform.ps5 then |
|
AsyncPlayStationActivityEnd(self.id, const.PlayStationActivityOutcomeAbandoned) |
|
end |
|
AccountStorage.PlayStationStartedActivities[self.id] = nil |
|
SaveAccountStorage(5000) |
|
end |
|
|
|
function PlayStationActivities:GetWarning() |
|
local warnings = {} |
|
|
|
if not rawget(self, "Launch") then |
|
warnings[#warnings + 1] = "Missing launch procedure!" |
|
end |
|
|
|
return #warnings ~= 0 and table.concat(warnings, "\n") |
|
end |
|
|
|
|
|
|
|
if Platform.developer or Platform.ps5 then |
|
|
|
if FirstLoad then |
|
g_DelayedLaunchActivity = false |
|
g_PauseLaunchActivityReasons = { |
|
["EngineStarted"] = true, |
|
["AccountStorage"] = true |
|
} |
|
end |
|
|
|
function PlayStationLaunchActivity(activity_id) |
|
if g_PauseLaunchActivityReasons ~= empty_table then |
|
g_DelayedLaunchActivity = activity_id |
|
return |
|
end |
|
|
|
local activity = ActivitiesPresets[activity_id] |
|
if activity then |
|
assert(rawget(activity, "Launch"), "Activity not launchable!") |
|
activity:Launch() |
|
return |
|
end |
|
assert(false, string.format("Missing activity '%s'!", activity_id)) |
|
end |
|
|
|
function PauseLaunchActivity(reason) |
|
g_PauseLaunchActivityReasons[reason] = true |
|
end |
|
|
|
function ResumeLaunchActivity(reason) |
|
g_PauseLaunchActivityReasons[reason] = nil |
|
if g_DelayedLaunchActivity and g_PauseLaunchActivityReasons == empty_table then |
|
PlayStationLaunchActivity(g_DelayedLaunchActivity) |
|
g_DelayedLaunchActivity = false |
|
end |
|
end |
|
|
|
function PlayStationGetActiveActivities(account_ids) |
|
if not account_ids then |
|
local err, account_id = PlayStationGetUserAccountId() |
|
if err then return err end |
|
account_ids = { account_id } |
|
end |
|
|
|
account_ids = table.map(account_ids, tostring) |
|
local uri = string.format("/v1/users/activities?accountIds=%s&limit=10", table.concat(account_ids, ",")) |
|
local err, http_code, request_result_json = AsyncOpWait(PSNAsyncOpTimeout, nil, "AsyncPlayStationWebApiRequest", "activities", uri, "", "GET", "", {}) |
|
if err or http_code ~= 200 then |
|
return err or "Failed", http_code |
|
end |
|
|
|
local err, request_result = JSONToLua(request_result_json) |
|
if err then return err end |
|
|
|
local result = {} |
|
for _,account_id in ipairs(account_ids) do |
|
local user = table.find_value(request_result.users, "accountId", account_id) |
|
result[#result + 1] = user and user.activities or empty_table |
|
end |
|
|
|
return nil, result |
|
end |
|
|
|
function OnMsg.DoneMap() |
|
for activity_id,_ in pairs(AccountStorage.PlayStationStartedActivities) do |
|
if g_DelayedLaunchActivity == activity_id then |
|
goto continue |
|
end |
|
|
|
local activity = ActivitiesPresets[activity_id] |
|
if activity.abandon_on_done_map then |
|
activity:Abandon() |
|
end |
|
|
|
::continue:: |
|
end |
|
end |
|
|
|
function OnMsg.ChangeMap() |
|
PauseLaunchActivity("ChangeMap") |
|
end |
|
|
|
function OnMsg.ChangeMapDone() |
|
ResumeLaunchActivity("ChangeMap") |
|
end |
|
|
|
function OnMsg.EngineStarted() |
|
ResumeLaunchActivity("EngineStarted") |
|
CreateRealTimeThread(function() |
|
|
|
|
|
while not AccountStorage do |
|
WaitMsg("AccountStorageChanged") |
|
end |
|
|
|
|
|
AccountStorage.PlayStationStartedActivities = AccountStorage.PlayStationStartedActivities or {} |
|
|
|
|
|
if Platform.ps5 then |
|
local err, psn_activities = PlayStationGetActiveActivities() |
|
if not err and psn_activities[1] then |
|
table.clear(AccountStorage.PlayStationStartedActivities) |
|
for _,activity in ipairs(psn_activities[1]) do |
|
AccountStorage.PlayStationStartedActivities[activity.activityId] = true |
|
end |
|
end |
|
end |
|
|
|
if not Platform.ps5 and AccountStorage.PlayStationActivityDbgLaunchOnBoot then |
|
PlayStationLaunchActivity(AccountStorage.PlayStationActivityDbgLaunchOnBoot) |
|
AccountStorage.PlayStationActivityDbgLaunchOnBoot = false |
|
SaveAccountStorage(1000) |
|
end |
|
|
|
|
|
for activity_id,_ in pairs(AccountStorage.PlayStationStartedActivities) do |
|
local activity = ActivitiesPresets[activity_id] |
|
if not activity.abandon_on_done_map and g_DelayedLaunchActivity == activity_id then |
|
goto continue |
|
end |
|
|
|
if activity.abandon_on_done_map then |
|
activity:Abandon() |
|
end |
|
|
|
::continue:: |
|
end |
|
|
|
ResumeLaunchActivity("AccountStorage") |
|
end) |
|
end |
|
|
|
end |
|
|
|
DefineClass.RichPresence = { |
|
__parents = { "Preset", }, |
|
__generated_by_class = "PresetDef", |
|
|
|
properties = { |
|
{ id = "name", name = "Name", |
|
editor = "text", default = false, translate = true, }, |
|
{ id = "desc", name = "Description", |
|
editor = "text", default = false, translate = true, }, |
|
{ id = "xbox_id", name = "Xbox ID", |
|
editor = "text", default = false, }, |
|
}, |
|
GlobalMap = "RichPresencePresets", |
|
EditorMenubarName = "Rich Presence", |
|
EditorMenubar = "Editors.Lists", |
|
} |
|
|
|
DefineClass.TrophyGroup = { |
|
__parents = { "Preset", }, |
|
__generated_by_class = "PresetDef", |
|
|
|
properties = { |
|
{ category = "BuildPS4", id = "ps4_gid", name = "Group ID", help = "Those must be consecutive and unique.", |
|
editor = "number", default = -1, buttons = { {name = "Generate", func = "GenerateGroupIDs"}, }, min = -1, max = 128, }, |
|
{ category = "BuildPS4", id = "ps4_name", name = "Name", |
|
editor = "text", default = false, translate = true, }, |
|
{ category = "BuildPS4", id = "ps4_description", name = "Trophy Group Description", |
|
editor = "text", default = false, translate = true, }, |
|
{ category = "BuildPS4", id = "ps4_icon", name = "Icon", |
|
editor = "ui_image", default = "", dont_save = true, read_only = true, |
|
no_validate = true, filter = "All files|*.png", }, |
|
{ category = "BuildPS4", id = "ps4_trophies", name = "Trophies", |
|
editor = "preset_id_list", default = {}, dont_save = true, read_only = true, no_validate = true, preset_class = "Achievement", item_default = "", }, |
|
{ category = "BuildPS5", id = "ps5_gid", name = "Group ID", help = "Those must be consecutive and unique.", |
|
editor = "number", default = -1, buttons = { {name = "Generate", func = "GenerateGroupIDs"}, }, min = -1, max = 128, }, |
|
{ category = "BuildPS5", id = "ps5_name", name = "Name", |
|
editor = "text", default = false, translate = true, }, |
|
{ category = "BuildPS5", id = "ps5_description", name = "Description", |
|
editor = "text", default = false, translate = true, }, |
|
{ category = "BuildPS5", id = "ps5_icon", name = "Icon", |
|
editor = "ui_image", default = "", dont_save = true, read_only = true, |
|
no_validate = true, filter = "All files|*.png", }, |
|
{ category = "BuildPS5", id = "ps5_trophies", name = "Trophies", |
|
editor = "preset_id_list", default = {}, dont_save = true, read_only = true, no_validate = true, preset_class = "Achievement", item_default = "", }, |
|
}, |
|
GlobalMap = "TrophyGroupPresets", |
|
EditorMenubarName = "Trophy Groups", |
|
EditorIcon = "CommonAssets/UI/Icons/top trophy winner.png", |
|
EditorMenubar = "Editors.Lists", |
|
} |
|
|
|
function TrophyGroup:Getps4_icon() |
|
local _, icon_path = GetPlayStationTrophyGroupIcon(self.id, "ps4") |
|
return icon_path |
|
end |
|
|
|
function TrophyGroup:Getps4_trophies() |
|
return self:GetTrophies("ps4") |
|
end |
|
|
|
function TrophyGroup:Getps5_icon() |
|
local _, icon_path = GetPlayStationTrophyGroupIcon(self.id, "ps5") |
|
return icon_path |
|
end |
|
|
|
function TrophyGroup:Getps5_trophies() |
|
return self:GetTrophies("ps5") |
|
end |
|
|
|
function TrophyGroup:GenerateGroupIDs(root, prop_id) |
|
local platform = string.match(prop_id, "(.*)_gid") |
|
local group_id_field = prop_id |
|
local groups_counter = 0 |
|
ForEachPreset(TrophyGroup, function(group) |
|
local is_group_used = CalcTrophyGroupPoints(group.id, platform) ~= 0 |
|
|
|
local group_id = -1 |
|
if is_group_used then |
|
group_id = groups_counter |
|
groups_counter = groups_counter + 1 |
|
end |
|
|
|
if group[group_id_field] ~= group_id then |
|
group[group_id_field] = group_id |
|
group:MarkDirty() |
|
end |
|
end) |
|
end |
|
|
|
function TrophyGroup:GetTrophies(platform) |
|
local trophies = PresetArray(Achievement, function(achievement) |
|
return achievement:GetTrophyGroup(platform) == self.id |
|
end) |
|
return table.imap(trophies, function(trophy) |
|
return trophy.id |
|
end) |
|
end |
|
|
|
function TrophyGroup:IsBaseGameGroup(platform) |
|
if self[platform .. "_gid"] < 0 then return false end |
|
local dlc = FindPreset("DLCConfig", self.save_in) |
|
return not dlc or dlc[platform .. "_handling"] == "Embed" |
|
end |
|
|
|
function TrophyGroup:GetError(platform) |
|
local errors = {} |
|
|
|
local ShouldTestPlatform = function(test_platform) |
|
return (not platform or platform == test_platform) |
|
end |
|
|
|
local groups = PresetArray(TrophyGroup) |
|
local GetPlayStationErrors = function(platform) |
|
local group_id_field = platform .. "_gid" |
|
local self_group_id = self[group_id_field] |
|
if CalcTrophyGroupPoints(self.id, platform) > 0 and self_group_id < 0 then |
|
errors[#errors + 1] = string.format("Missing %s trophy group id!", platform) |
|
elseif self_group_id >= 0 then |
|
table.sortby_field(groups, group_id_field) |
|
local next_group_id = 0 |
|
local group_id_holes = {} |
|
for _, group in ipairs(groups) do |
|
local curr_group_id = group[group_id_field] |
|
if next_group_id < self_group_id and curr_group_id >= 0 then |
|
if curr_group_id > next_group_id then |
|
if curr_group_id - next_group_id > 1 then |
|
group_id_holes[#group_id_holes + 1] = string.format("%d-%d", next_group_id, curr_group_id) |
|
else |
|
group_id_holes[#group_id_holes + 1] = next_group_id |
|
end |
|
end |
|
next_group_id = curr_group_id + 1 |
|
end |
|
if self ~= group and self_group_id == curr_group_id then |
|
errors[#errors + 1] = string.format("Duplicated %s trophy group id (%s)!", platform, group.id) |
|
end |
|
end |
|
|
|
if #group_id_holes ~= 0 then |
|
errors[#errors + 1] = string.format("%s group ids are not consecutive, missing %s!", string.upper(platform), table.concat(group_id_holes, ", ")) |
|
end |
|
end |
|
end |
|
|
|
if ShouldTestPlatform("ps4") then GetPlayStationErrors("ps4") end |
|
if ShouldTestPlatform("ps5") then GetPlayStationErrors("ps5") end |
|
|
|
return #errors ~= 0 and table.concat(errors, "\n") |
|
end |
|
|
|
function TrophyGroup:GetWarning(platform) |
|
local warnings = {} |
|
|
|
local ShouldTestPlatform = function(test_platform) |
|
return (not platform or platform == test_platform) |
|
end |
|
|
|
local GetPlayStationWarnings = function(platform) |
|
local trophies = self:GetTrophies(platform) |
|
if self[platform .. "_gid"] >= 0 then |
|
if #trophies == 0 then |
|
warnings[#warnings + 1] = string.format("Has %s group id but no trophies assigned!", platform) |
|
end |
|
local is_placeholder, icon_path = GetPlayStationTrophyGroupIcon(self.id, platform) |
|
if is_placeholder then |
|
warnings[#warnings + 1] = string.format("Missing %s trophy group icon (placeholder used): %s", platform, icon_path) |
|
end |
|
end |
|
|
|
local is_base_game_group = self:IsBaseGameGroup(platform) |
|
for _, trophy_name in ipairs(trophies) do |
|
local trophy = FindPreset("Achievement", trophy_name) |
|
local is_base_game_trophy = trophy:IsBaseGameTrophy(platform) |
|
if trophy.save_in ~= self.save_in and not (is_base_game_group and is_base_game_trophy) then |
|
warnings[#warnings + 1] = string.format( |
|
"%s trophy %s saved in %s while the group is saved in %s.", |
|
string.upper(platform), trophy_name, trophy.save_in, self.save_in) |
|
end |
|
end |
|
end |
|
|
|
if ShouldTestPlatform("ps4") then GetPlayStationWarnings("ps4") end |
|
if ShouldTestPlatform("ps5") then GetPlayStationWarnings("ps5") end |
|
|
|
return #warnings ~= 0 and table.concat(warnings, "\n") |
|
end |
|
|
|
DefineClass.VideoDef = { |
|
__parents = { "Preset", }, |
|
__generated_by_class = "PresetDef", |
|
|
|
properties = { |
|
{ category = "Common", id = "source", |
|
editor = "browse", default = false, folder = "svnAssets/Source/Movies", filter = "Video Files|*.avi|All Files|*.*", }, |
|
{ category = "Common", id = "ffmpeg_input_pattern", |
|
editor = "text", default = '-i "$(source)"', }, |
|
{ category = "Common", id = "sound", |
|
editor = "browse", default = false, folder = "svnAssets/Source/Movies", filter = "Audio Files|*.wav", }, |
|
{ category = "Desktop", id = "present_desktop", |
|
editor = "bool", default = true, }, |
|
{ category = "Desktop", id = "extension_desktop", |
|
editor = "text", default = "ivf", |
|
no_edit = function(self) return not self.present_desktop end, }, |
|
{ category = "Desktop", id = "ffmpeg_commandline_desktop", |
|
editor = "text", default = "-c:v vp8 -preset veryslow", |
|
no_edit = function(self) return not self.present_desktop end, }, |
|
{ category = "Desktop", id = "bitrate_desktop", |
|
editor = "number", default = 8000, |
|
no_edit = function(self) return not self.present_desktop end, }, |
|
{ category = "Desktop", id = "framerate_desktop", |
|
editor = "number", default = 30, |
|
no_edit = function(self) return not self.present_desktop end, }, |
|
{ category = "Desktop", id = "resolution_desktop", |
|
editor = "point", default = point(1920, 1080), |
|
no_edit = function(self) return not self.present_desktop end, }, |
|
{ category = "PS4", id = "present_ps4", |
|
editor = "bool", default = true, }, |
|
{ category = "PS4", id = "extension_ps4", |
|
editor = "text", default = "bsf", |
|
no_edit = function(self) return not self.present_ps4 end, }, |
|
{ category = "PS4", id = "ffmpeg_commandline_ps4", |
|
editor = "text", default = "-c:v h264 -profile:v high422 -pix_fmt yuv420p -x264opts force-cfr -bsf h264_mp4toannexb -f h264 -r 30000/1001", |
|
no_edit = function(self) return not self.present_ps4 end, }, |
|
{ category = "PS4", id = "bitrate_ps4", |
|
editor = "number", default = 6000, |
|
no_edit = function(self) return not self.present_ps4 end, }, |
|
{ category = "PS4", id = "framerate_ps4", |
|
editor = "number", default = 30, |
|
no_edit = function(self) return not self.present_ps4 end, }, |
|
{ category = "PS4", id = "resolution_ps4", |
|
editor = "point", default = point(1920, 1080), |
|
no_edit = function(self) return not self.present_ps4 end, }, |
|
{ category = "PS5", id = "present_ps5", |
|
editor = "bool", default = true, }, |
|
{ category = "PS5", id = "extension_ps5", |
|
editor = "text", default = "bsf", |
|
no_edit = function(self) return not self.present_ps5 end, }, |
|
{ category = "PS5", id = "ffmpeg_commandline_ps5", |
|
editor = "text", default = "-c:v h264 -profile:v high422 -pix_fmt yuv420p -x264opts force-cfr -bsf h264_mp4toannexb -f h264 -r 30000/1001", |
|
no_edit = function(self) return not self.present_ps5 end, }, |
|
{ category = "PS5", id = "bitrate_ps5", |
|
editor = "number", default = 6000, |
|
no_edit = function(self) return not self.present_ps5 end, }, |
|
{ category = "PS5", id = "framerate_ps5", |
|
editor = "number", default = 30, |
|
no_edit = function(self) return not self.present_ps5 end, }, |
|
{ category = "PS5", id = "resolution_ps5", |
|
editor = "point", default = point(1920, 1080), |
|
no_edit = function(self) return not self.present_ps5 end, }, |
|
{ category = "Xbox One", id = "present_xbox_one", |
|
editor = "bool", default = true, }, |
|
{ category = "Xbox One", id = "extension_xbox_one", |
|
editor = "text", default = "mp4", |
|
no_edit = function(self) return not self.present_xbox_one end, }, |
|
{ category = "Xbox One", id = "ffmpeg_commandline_xbox_one", |
|
editor = "text", default = "-c:v h264 -preset veryslow -pix_fmt yuv420p", |
|
no_edit = function(self) return not self.present_xbox_one end, }, |
|
{ category = "Xbox One", id = "bitrate_xbox_one", |
|
editor = "number", default = 8000, |
|
no_edit = function(self) return not self.present_xbox_one end, }, |
|
{ category = "Xbox One", id = "framerate_xbox_one", |
|
editor = "number", default = 30, |
|
no_edit = function(self) return not self.present_xbox_one end, }, |
|
{ category = "Xbox One", id = "resolution_xbox_one", |
|
editor = "point", default = point(1920, 1080), |
|
no_edit = function(self) return not self.present_xbox_one end, }, |
|
{ category = "Xbox Series", id = "present_xbox_series", |
|
editor = "bool", default = true, }, |
|
{ category = "Xbox Series", id = "extension_xbox_series", |
|
editor = "text", default = "mp4", |
|
no_edit = function(self) return not self.present_xbox_series end, }, |
|
{ category = "Xbox Series", id = "ffmpeg_commandline_xbox_series", |
|
editor = "text", default = "-c:v libx265 -tag:v hvc1 -preset veryslow -pix_fmt yuv420p", |
|
no_edit = function(self) return not self.present_xbox_series end, }, |
|
{ category = "Xbox Series", id = "bitrate_xbox_series", |
|
editor = "number", default = 8000, |
|
no_edit = function(self) return not self.present_xbox_series end, }, |
|
{ category = "Xbox Series", id = "framerate_xbox_series", |
|
editor = "number", default = 30, |
|
no_edit = function(self) return not self.present_xbox_series end, }, |
|
{ category = "Xbox Series", id = "resolution_xbox_series", |
|
editor = "point", default = point(1920, 1080), |
|
no_edit = function(self) return not self.present_xbox_series end, }, |
|
{ category = "Switch", id = "present_switch", |
|
editor = "bool", default = true, }, |
|
{ category = "Switch", id = "extension_switch", |
|
editor = "text", default = "mp4", |
|
no_edit = function(self) return not self.present_switch end, }, |
|
{ category = "Switch", id = "ffmpeg_commandline_switch", |
|
editor = "text", default = "-c:v h264 -preset veryslow -pix_fmt yuv420p", |
|
no_edit = function(self) return not self.present_switch end, }, |
|
{ category = "Switch", id = "bitrate_switch", |
|
editor = "number", default = 700, |
|
no_edit = function(self) return not self.present_switch end, }, |
|
{ category = "Switch", id = "framerate_switch", |
|
editor = "number", default = 30, |
|
no_edit = function(self) return not self.present_switch end, }, |
|
{ category = "Switch", id = "resolution_switch", |
|
editor = "point", default = point(1280, 720), |
|
no_edit = function(self) return not self.present_switch end, }, |
|
}, |
|
HasCompanionFile = true, |
|
GlobalMap = "VideoDefs", |
|
EditorMenubarName = "Video defs", |
|
EditorIcon = "CommonAssets/UI/Icons/outline video.png", |
|
EditorMenubar = "Editors.Engine", |
|
} |
|
|
|
function VideoDef:GetPropsForPlatform(platform) |
|
assert(table.find({ "desktop", "ps4", "ps5", "xbox_one", "xbox_series", "switch" }, platform)) |
|
local result = {} |
|
local props = { "extension", "ffmpeg_commandline", "bitrate", "framerate", "resolution", "present" } |
|
for key, value in ipairs(props) do |
|
result[value] = self[value .. "_" .. platform] |
|
end |
|
local video_path = string.match(self.source or "", "svnAssets/Source/(.+)") |
|
if video_path then |
|
local dir, name, ext = SplitPath(video_path) |
|
result.video_game_path = dir .. name .. "." .. result.extension |
|
end |
|
|
|
local sound_path = string.match(self.sound or "", "svnAssets/Source/(.+)") |
|
if sound_path then |
|
local dir, name, ext = SplitPath(sound_path) |
|
result.sound_game_path = dir .. name |
|
end |
|
|
|
return result |
|
end |
|
|
|
DefineClass.VoiceActorDef = { |
|
__parents = { "Preset", }, |
|
__generated_by_class = "PresetDef", |
|
|
|
properties = { |
|
{ id = "VoiceId", name = "VoiceID", |
|
editor = "text", default = false, }, |
|
}, |
|
GlobalMap = "VoiceActors", |
|
EditorMenubarName = "Voice Actors", |
|
EditorIcon = "CommonAssets/UI/Icons/human male man people person.png", |
|
EditorMenubar = "Editors.Audio", |
|
EditorView = Untranslated("<u(Id)> <color 0 128 0><u(VoiceId)>"), |
|
} |
|
|
|
|