File size: 13,322 Bytes
b6a38d7 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 |
AppendClass.EntitySpecProperties = {
properties = {
{ id = "env_colorized", name = "EnvColorized Group", editor = "combo", items = GetEnvColorizedGroups, category = "Misc", default = "", entitydata = true, },
{ id = "default_colors", name = "Default colors", editor = "nested_obj", base_class = "ColorizationPropSet", inclusive = true, default = false, },
},
}
local function InsertClass(class_parent_str, new_class)
if not new_class or new_class == "" then return class_parent_str end
if not class_parent_str or class_parent_str == "" then
class_parent_str = new_class
elseif not string.find(class_parent_str, "EnvColorized") then
class_parent_str = class_parent_str .. "," .. new_class
end
return class_parent_str
end
function OnMsg.ClassesGenerate()
local old_ExportEntityDataForSelf = EntitySpecProperties.ExportEntityDataForSelf
function EntitySpecProperties:ExportEntityDataForSelf()
local data = old_ExportEntityDataForSelf(self)
if self.env_colorized and self.env_colorized ~= "" and not self:IsPropertyDefault("env_colorized") then
data.entity.class_parent = InsertClass(data.entity.class_parent, "EnvColorized")
end
return data
end
end
--------- Environment ------------
EnvColorizedGroups = false
local function CollectGroups()
if not EnvColorizedGroups then
EnvColorizedGroups = { "EnvColorized" }
local class_list = ClassDescendantsListInclusive("EnvColorized")
local groups = { ["EnvColorized"] = true }
for _, class in ipairs(class_list) do
local class_table = _G[class]
if not groups[class_table.env_colorized] then
groups[class_table.env_colorized] = true
table.insert(EnvColorizedGroups, class_table.env_colorized)
end
end
end
return EnvColorizedGroups
end
function GetEnvColorizedGroups()
local list = table.copy(CollectGroups())
table.insert(list, 1, "")
return list
end
function GetEnvColorizedFilters(obj, prop_meta, validate_fn)
if validate_fn == "validate_fn" then
-- function for preset validation, checks whether the property value is from "items"
return "validate_fn", function(value, obj, prop_meta)
return table.find(CollectGroups(), value) or IsKindOf(g_Classes[value], "EnvColorized")
end
end
local list = { }
for _, group in ipairs(CollectGroups()) do
local text = "GROUP " .. group
if group == "EnvColorized" then
text = "All EnvColorized Objects"
end
table.insert(list, { text = text, value = group } )
end
local class_list = ClassDescendantsList("EnvColorized")
for _, class in ipairs(class_list) do
table.insert(list, { text = class, value = class } )
end
return list
end
DefineClass.EnvColorized = {
__parents = {"ColorizableObject", "CObject" },
properties = {
{ id = "env_colorized", name = "EnvColorized Group", editor = "text", read_only = true, dont_save = true},
},
flags = { cfEditorCallback = true, },
env_colorized = "EnvColorized",
}
function EnvColorized:ColorizationReadOnlyReason()
return "Object is EnvColorized. Colorization for such objects is controlled by EnvironmentColorPalette Editor."
end
function EnvColorized:ColorizationPropsDontSave(i)
return true
end
------------------------------ EnvironmentColorEntryBase ------------------------------
DefineClass.EnvironmentColorEntryBase = {
__parents = {"ColorizationPropSet"},
properties = {},
EditorExcludeAsNested = true,
}
function EnvironmentColorEntryBase:AcceptsClass(obj_class)
return false
end
function EnvironmentColorEntryBase:AcceptsTerrain(terrain_id)
return false
end
function EnvironmentColorEntryBase:__eq(b)
return rawequal(self, b)
end
------------------------------ EnvironmentColorEntry (Entity colorization) ------------------------------
DefineClass.EnvironmentColorEntry = {
__parents = {"EnvironmentColorEntryBase"},
properties = {
{ id = "filter_class", editor = "choice", items = GetEnvColorizedFilters, default = "", },
{ id = "hue_variation1", editor = "number", slider = true, min = 0, default = 0, max = 900, scale = 10, },
{ id = "hue_variation2", editor = "number", slider = true, min = 0, default = 0, max = 900, scale = 10, },
{ id = "hue_variation3", editor = "number", slider = true, min = 0, default = 0, max = 900, scale = 10, },
},
EditorExcludeAsNested = true,
}
function EnvironmentColorEntry:GetEditorView()
local filter_name = ""
if self.filter_class == "EnvColorized" then filter_name = "All EnvColorized Objects"
elseif rawget(_G, self.filter_class) then filter_name = "Entity " .. self.filter_class
else filter_name = "Group " .. self.filter_class end
return Untranslated(filter_name .. " " .. _InternalTranslate(ColorizationPropSet.GetEditorView(self)))
end
local IsKindOf = IsKindOf
function EnvironmentColorEntry:AcceptsClass(obj_class)
local filter_value = self.filter_class
if not filter_value or filter_value == "" then return false end
local class_table = _G[obj_class]
if not IsKindOf(class_table, "EnvColorized") then return false end
if filter_value == "EnvColorized" then return true end
if filter_value == obj_class then return true end
if filter_value == class_table.env_colorized then return true end
return false
end
------------------------------ EnvironmentTerrainColorEntry (Terrain colorization) ------------------------------
DefineClass.EnvironmentTerrainColorEntry = {
__parents = {"EnvironmentColorEntryBase"},
properties = {
{ id = "terrain_id", editor = "choice", items = PresetsCombo("TerrainObj"), default = "", },
},
EditorExcludeAsNested = true,
}
function EnvironmentTerrainColorEntry:GetEditorView()
local filter_name = string.format("Terrain %s - %s", self.terrain_id, _InternalTranslate(ColorizationPropSet.GetEditorView(self)))
return filter_name
end
function EnvironmentTerrainColorEntry:AcceptsTerrain(terrain_id)
return terrain_id == self.terrain_id
end
DefineClass.EnvironmentColorPalette = {
__parents = { "Preset" },
properties = {
{ category = "Match (AND)", id = "regions", editor = "string_list", default = false, items = function (self) return PresetsCombo("GameStateDef", "region") end, help = "Match if current region is any of the list. Leave empty to always match." },
{ category = "Match (AND)", id = "lightmodels", editor = "preset_id_list", preset_class = "LightmodelPreset", default = false, help = "Match if current lightmodel is any of the list. Leave empty to always match." },
{ category = "Match (AND)", id = "enabled", editor = "bool", default = true, help = "Should match?" },
},
GlobalMap = "EnvironmentColorPalettes",
ContainerClass = "EnvironmentColorEntryBase",
HasSortKey = true,
HasGroups = false,
EditorCustomActions = {
{
FuncName = "ApplyOnCurrentMap",
Icon = "CommonAssets/UI/Ged/play",
Menubar = "Test",
Name = "Apply",
Toolbar = "main",
},
},
Documentation = "Changes the color of various aspects of the environment like vegetation, terrains or rocks.",
}
function EnvironmentColorPalette:GetEditorView()
local regions = "any"
if self.regions and #self.regions > 0 and self.regions[1] then
regions = table.concat(table.map(self.regions or {}, function(v) return v or "" end ), ", ")
end
local lightmodels = "any"
if self.lightmodels and #self.lightmodels > 0 and self.lightmodels[1] then
lightmodels = table.concat(table.map(self.lightmodels or {}, function(v) return v or "" end ), ", ")
end
local act_string = false
if not self.enabled then
act_string = "disabled"
elseif regions == "any" and lightmodels == "any" and self.enabled then
act_string = "always matched"
else
act_string = "[RG] " .. regions .. " [LM] " .. lightmodels
end
local preset_name = self.id
local is_active = LastEnvColorizedCache and LastEnvColorizedCache.EnvColorSource == self.id
if is_active then
preset_name = "<color 89 192 98>" .. preset_name .. "</color>"
end
return Untranslated(preset_name .. "<color 128 128 168> - " .. act_string .. "</color>")
end
if FirstLoad then
LastEnvColorizedCache = false
end
envpalette_print = CreatePrint{
"envpalette",
format = "printf",
output = function() end,
}
function EnvironmentColorPalette:CalcEnvCache()
local class_list = ClassDescendantsListInclusive("EnvColorized")
local class_to_color = {}
for _, class in ipairs(class_list) do
class_to_color[class] = false
end
local terrain_to_color = {}
ForEachPreset("TerrainObj", function(preset) terrain_to_color[preset.id] = false end)
local IsKindOf = IsKindOf
for _, child in ipairs(self) do
for _, class in ipairs(class_list) do
if child:AcceptsClass(class) then
class_to_color[class] = child
end
end
ForEachPreset("TerrainObj", function(preset)
if child:AcceptsTerrain(preset.id) then
terrain_to_color[preset.id] = child
end
end)
end
return {
EnvColorizedToColor = class_to_color,
TerrainToColor = terrain_to_color,
EnvColorSource = self.id,
EnvColorizedHash = table.hash(class_to_color),
TerrainHash = table.hash(terrain_to_color),
}
end
function ModifyHueByOffset(color, offset)
if offset == 0 then return color end
local r, g, b = GetRGB(color)
local h, s, v = UIL.RGBtoHSV(r,g,b)
h = h + offset
if h < 0 then
h = h + 256
else
h = h % 256
end
return RGB(UIL.HSVtoRGB(h, s, v))
end
local xxhash = xxhash
local MulDivRound = MulDivRound
local function ApplyToObject(class_to_color, obj)
local palette = class_to_color[obj:GetEntity()] or class_to_color[obj.class]
if not palette then return end
obj:SetColorization(palette)
local x, y = obj:GetPosXYZ()
local seed = xxhash(x, y)
local offset1 = palette.hue_variation1 - MulDivRound((seed >> 0 ) & 0xFF, palette.hue_variation1 * 2, 0xFF)
local offset2 = palette.hue_variation2 - MulDivRound((seed >> 8 ) & 0xFF, palette.hue_variation2 * 2, 0xFF)
local offset3 = palette.hue_variation3 - MulDivRound((seed >> 16) & 0xFF, palette.hue_variation3 * 2, 0xFF)
obj:SetEditableColor1(ModifyHueByOffset(obj:GetEditableColor1(), offset1 / 10))
obj:SetEditableColor2(ModifyHueByOffset(obj:GetEditableColor2(), offset2 / 10))
obj:SetEditableColor3(ModifyHueByOffset(obj:GetEditableColor3(), offset3 / 10))
return true
end
function ApplyCurrentEnvColorizedToObj(obj)
if not LastEnvColorizedCache or not IsKindOf(obj, "EnvColorized") then
return false
end
return ApplyToObject(LastEnvColorizedCache.EnvColorizedToColor, obj)
end
function EnvironmentColorPalette:ApplyOnCurrentMap(force)
local oldEnvCache = LastEnvColorizedCache
local envcache = self:CalcEnvCache()
LastEnvColorizedCache = envcache
if force or not oldEnvCache or oldEnvCache.EnvColorizedHash ~= envcache.EnvColorizedHash then
MapForEach("map", "EnvColorized", function(obj, envcache)
ApplyToObject(envcache.EnvColorizedToColor, obj)
end, envcache)
end
if force or not oldEnvCache or oldEnvCache.TerrainHash ~= envcache.TerrainHash then
ReloadTerrains() -- Moves terrain data form lua to C
hr.TR_ForceReloadNoTextures = 1 -- Updates the terrain itself
end
ObjModified(Presets.EnvironmentColorPalette)
end
function EnvColorizedTerrainColor(terrain_obj) -- Called from C
local color_mod = terrain_obj.color_modifier
if LastEnvColorizedCache then
local override_value = LastEnvColorizedCache.TerrainToColor[terrain_obj.id]
if override_value then
color_mod = override_value:GetEditableColor1()
end
end
return color_mod
end
local ApplyCurrentEnvColorizedToObj = ApplyCurrentEnvColorizedToObj
function OnMsg.EditorCallback(id, objects, ...)
if id == "EditorCallbackPlace" or id == "EditorCallbackPlaceCursor" or id == "EditorCallbackClone" then
for i = 1, #objects do
local obj = objects[i]
ApplyCurrentEnvColorizedToObj(obj)
for _, attach in ipairs(obj:GetAttaches()) do
ApplyCurrentEnvColorizedToObj(attach)
end
end
end
end
local ignore_lightmodels = {"SatelliteView"}
local function FindEnvColorPalette(region, lightmodel)
for _, lm_name in ipairs(ignore_lightmodels) do
if string.find(lightmodel, lm_name) then
return false
end
end
local best_match = false
ForEachPreset(EnvironmentColorPalette, function(preset)
local lm_found = not preset.lightmodels or #preset.lightmodels == 0 or table.find(preset.lightmodels, lightmodel)
local region_found = not preset.regions or #preset.regions == 0 or table.find(preset.regions, region)
if lm_found and region_found and preset.enabled and not best_match then
best_match = preset
end
end)
return best_match
end
function ApplyCurrentEnvironmentColorPalette(force)
local lightmodel_id = CurrentLightmodel and CurrentLightmodel[1] and CurrentLightmodel[1].id
local region_id = CurrentMap and CurrentMap ~= "" and MapData[CurrentMap] and MapData[CurrentMap].Region
local envpalette = FindEnvColorPalette(region_id, lightmodel_id)
envpalette_print("Applying palette '%s' from region '%s' and lightmodel '%s', forced '%s'. Previous '%s'",
envpalette and envpalette.id or "none", region_id, lightmodel_id, not not force, LastEnvColorizedCache and LastEnvColorizedCache.EnvColorSource or "none")
if envpalette then
envpalette:ApplyOnCurrentMap(force)
return true
end
end
function OnMsg.LightmodelChange(view, lightmodel, time, prev_lm)
if lightmodel then
if not ChangingMap then
ApplyCurrentEnvironmentColorPalette()
end
end
end
function OnMsg.NewMapLoaded()
LastEnvColorizedCache = false
ApplyCurrentEnvironmentColorPalette(true)
end |