File size: 14,774 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
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
DefineClass.XFloatingText = {
	__parents = { "XText" },
	properties = {
		{ category = "Floating Text", id = "expire_time", name = "Expire time", editor = "number", default = 800 },
		{ category = "Floating Text", id = "life_time", name = "Life time", editor = "number", default = 0, scale = "ms",
			help = "The life time of a floating text. If it is not shown in that period of time, it will never be shown. Set this to 0 for an endless life time.", },
		{ category = "Floating Text", id = "fade_start", name = "Fade start time", editor = "number", default = 400 },
		{ category = "Floating Text", id = "transparency_start", name = "Transparency start", editor = "number", default = 255, min = 0, max = 255 },
		{ category = "Floating Text", id = "transparency_end", name = "Transparency end", editor = "number", default = 0, min = 0, max = 255 },
		{ category = "Floating Text", id = "offset_start_time", name = "Offset start time", editor = "number", default = 0 },
		{ category = "Floating Text", id = "offset_amount", name = "Offset amount", editor = "number", default = 100 },
		{ category = "Floating Text", id = "z_offset", name = "Z offset", editor = "number", default = 30, scale = "m" },
		{ category = "Floating Text", id = "randomize_x", name = "Randomize X", editor = "bool", default = true },
		{ category = "Floating Text", id = "exclusive", name = "Exclusive", editor = "bool", default = false,
			help = "Only this floating text can be active on the target. Removes all previously spawned." },
		{ category = "Floating Text", id = "same_exclusive", name = "Same Exclusive", editor = "choice", items = {"False", "Same Time", "All"}, default = "False",
			help = "The same floating texts cannot coexist. Removes the new one. Same Time - removes same floating texts if added at the same time, All - removes same floating texts." },
		{ category = "Floating Text", id = "exclusive_discard", name = "Exclusive Discard", editor = "bool", default = false,
			help = "Like exclusive, but in reverse - prevents other texts from spawning, instead destroying them." },
		{ category = "Floating Text", id = "exclusive_by_type", name = "Exclusive By Type", editor = "bool", default = false,
			help = "Like exclusive, but only removes previously spawned texts of the same type." },
		{ category = "Floating Text", id = "interpolate_opacity", name = "Interpolate opacity", editor = "bool", default = false },
		{ category = "Floating Text", id = "interpolate_pos", name = "Interpolate position", editor = "bool", default = true },
		{ category = "Floating Text", id = "always_show_on_distance", name = "doe snot hide on distance", editor = "bool", default = false },
	},
	TextStyle = "EditorText",
	
	target = false,
	
	HandleMouse = false,
	ChildrenHandleMouse = false,
	UseClipBox = false,
	Clip = false,
	Translate = true,
	default_spot = false,
	prevent_overlap = true,
	stagger_spawn = true, -- Prevent texts from overlapping by staggering texts on the same target.
	
	spawn_stagger = false,
	dbg_removed_by = false
}

function XFloatingText:OnDelete()
	Msg(self)
end

MapVar("FloatingTexts", {}, weak_keys_meta)
PersistableGlobals.FloatingTexts = false

local function lFindTextPos(target, z_offset)
	local prev = FloatingTexts and FloatingTexts[target]
	prev = prev and prev[1]
	local tx, ty, tz = target:GetVisualPosXYZ()
	local z = tz + z_offset*guic
	return tx, ty, z
end

local function lIntersectionDepth(box1, box2)
	local halfWidthA = box1:sizex() / 2
	local halfHeightA = box1:sizey() / 2
	local halfWidthB = box2:sizex() / 2
	local halfHeightB = box2:sizey() / 2
	local centerA = box1:Center()
	local centerB = box2:Center()

	-- calculate current and minimum-non-intersecting distances between centers
	local distanceX = centerA:x() - centerB:x()
	local distanceY = centerA:y() - centerB:y()
	local minDistanceX = halfWidthA + halfWidthB
	local minDistanceY = halfHeightA + halfHeightB

	-- if we are not intersecting at all, return (0, 0)
	if abs(distanceX) >= minDistanceX or abs(distanceY) >= minDistanceY then return 0, 0 end
		
	local depthX
	if distanceX == 0 then
		depthX = 0
	elseif distanceX > 0 then
		depthX = minDistanceX - distanceX
	else
		depthX = -minDistanceX - distanceX
	end
	
	local depthY
	if distanceY == 0 then
		depthY = 0
	elseif distanceY > 0 then
		depthY = minDistanceY - distanceY
	else
		depthY = -minDistanceY - distanceY
	end
	
	return depthX, depthY
end

local ShowFloatingTextDist = const.Camera and const.Camera.ShowFloatingTextDist or 500 * guim
local IsPoint = IsPoint
local IsValid = IsValid
local GameToScreenXY = GameToScreenXY
local IsCloser2D = IsCloser2D

function ShouldShowFloatingText(target, text, always_show_on_distance)
	if not text or text == "" or not target or not config.FloatingTextEnabled then return end
	-- invalid pos
	if not IsValidPos(target) then
		return
	end
	-- do not show at game start
	if GameInitAfterLoading or GameTime() == 0 then return end
	if always_show_on_distance then
		return true
	end
	-- show only texts visible on-screen
	local front, sx, sy = GameToScreenXY(target)
	if not front or terminal.desktop.box:Dist2D2(sx, sy) > 200*200 then
		return
	end
	-- show only texts close to the camera
	local cam_pos = camera.GetPos()
	return IsValidPos(cam_pos) and IsCloser2D(cam_pos, target, ShowFloatingTextDist)
end

function CreateCustomFloatingText(ftext, target, text, style, spot, stagger_spawn, params, game_time)
	if not ShouldShowFloatingText(target, text, ftext and ftext.always_show_on_distance) then
		if ftext then
			ftext:delete()
		end
		return
	end
	
	if not IsT(text) then
		text = Untranslated(text)
	end
	
	local timeNow = game_time and GameTime() or GetPreciseTicks()
	local target_key = IsPoint(target) and xxhash(target) or target
	local list = FloatingTexts and FloatingTexts[target_key]
	local prev = list and list[#list]
	if prev and prev.window_state ~= "destroying" then
		if prev.exclusive_discard then
			if ftext then
				ftext.dbg_removed_by = "exclusive discard ftext on target"
				ftext:delete()
			end
			return
		end
		local ttext = _InternalTranslate(text)
		for _, previ in ipairs(list) do
			if (previ.same_exclusive == "Same Time" and previ.timeNow == timeNow or previ.same_exclusive == "All") and previ.text == ttext then
				if ftext then
					ftext.dbg_removed_by = "exclusive same ftext on target"
					ftext:delete()
				end
				return
			end
		end
	end
	

	-- don't show the same texts on nearby targets
	if config.RemoveSameFTextNearby then
		local ttext = _InternalTranslate(text)
		local _, sx, sy = GameToScreenXY(target)
		for ftarget, ftext_list in pairs(FloatingTexts) do
			if target.show_same_floating_texts_nearby or target == ftarget or not IsValid(ftarget) then goto skip end
			
			local _, t_sx, t_sy = GameToScreenXY(ftarget)
			if IsCloser2D(sx, sy , t_sx, t_sy, 30) then
				for _, prev_ftext in ipairs(ftext_list) do
					if prev_ftext.text == ttext then
						if ftext then
							ftext.dbg_removed_by = "exclusive same ftext on targets nearby"
							ftext:delete()
						end
						return
					end
				end
			end
			
			::skip::
		end
	end
	
	if not ftext then
		ftext = XTemplateSpawn(config.FloatingTextClass or "XFloatingText", EnsureDialog("FloatingTextDialog"), false)
		table.overwrite(ftext, params or empty_table)
	end
	
	if ftext.window_state == "new" then ftext:Open() end
	spot = spot or ftext.default_spot
	stagger_spawn = stagger_spawn == nil and ftext.stagger_spawn or stagger_spawn
	
	if list then
		-- Close any previously open floating texts on this target.
		-- We don't want them overlapping in some cases.
		if (ftext.exclusive or ftext.exclusive_by_type) and prev then
			local byType = ftext.exclusive_by_type
			local myType = ftext.class
			for i, t in ipairs(list) do
				if t.window_state == "open" and (not byType or IsKindOf(t, myType)) then
					t.dbg_removed_by = "exclusive ftext, byType:" .. tostring(byType) .. " and of type " .. tostring(myType)
					t:Close()
				end
			end
			prev = false
		end
		
		list[#list + 1] = ftext
	else
		list = { ftext }
		FloatingTexts[target_key] = list
	end
	
	ftext.timeNow = timeNow
	ftext.target = target_key
	ftext:SetText(text)
	if type(style) == "number" then
		ftext:SetTextColor(style)
	elseif type(style) == "string" then
		ftext:SetTextStyle(style)
	end
	local backupPosition
	if IsValid(target) then
		local x, y, z = target:GetVisualPosXYZ()
		backupPosition = point(x, y, z + ftext.z_offset*guic)
	end
	
	CreateMapRealTimeThread(WaitStartFloatingText, ftext, prev, stagger_spawn, spot, target, backupPosition, game_time)
	
	return ftext
end

local function RemoveFloatingTextReason(ftext, reason)
	table.remove_entry(FloatingTexts[ftext.target], ftext)
	if #(FloatingTexts[ftext.target] or "") == 0 then
		FloatingTexts[ftext.target] = nil
	end
	if ftext.window_state == "open" then
		ftext.dbg_removed_by = reason
		ftext:Close()
	end
end

function WaitStartFloatingText(ftext, prev, stagger_spawn, spot, target, backupPosition, game_time)
	-- Center the text above the spot. To do this get the text size...
	local width, height = ftext:Measure(ftext.MaxWidth, ftext.MaxHeight)
	local minx, miny, maxx, maxy = ftext:GetEffectiveMargins()
	local xLoc = -(width / 2) + minx
	local yLoc = -height + miny
	
	-- If the height of the window is smaller than the offset up, it will be considered off screen (pos is 0,0) and clipped
	height = height + -yLoc

	if ftext.OffsetBox then
		xLoc, yLoc = ftext:OffsetBox(xLoc, yLoc)
	end

	if prev and ftext.prevent_overlap and prev.expire_time then
		if ftext.randomize_x then
			xLoc = xLoc + AsyncRand(-100, 50) -- Make it more interesting
		end
		if stagger_spawn then
			-- Stagger this text a bit to let the previous one pass
			local timeNow = game_time and GameTime() or RealTime()
			if prev.spawn_stagger and (prev.spawn_stagger - timeNow > 0) then
				ftext.spawn_stagger = prev.spawn_stagger
			else
				ftext.spawn_stagger = timeNow
			end
			ftext:SetVisible(false)
			ftext.spawn_stagger = ftext.spawn_stagger + prev.offset_start_time + prev.expire_time / 3
			while prev.window_state ~= "destroying" do
				local sleep_time = ftext.spawn_stagger - (game_time and GameTime() or RealTime())
				if game_time then
					sleep_time = MulDivRound(sleep_time, 1000, Max(GetTimeFactor(), 1000))
				end
				if sleep_time <= 1 then
					break
				end
				local textDeleted = WaitMsg(prev, sleep_time)
				if textDeleted then
					break
				end
			end
			if ftext.window_state == "destroying" then return end -- Was deleted while waiting.
			
			-- Check if this ftext's lifetime is over and if so - remove it
			if ftext.life_time ~= 0 then
				timeNow = game_time and GameTime() or RealTime()
				if ftext.timeNow + ftext.life_time - timeNow < 0 then
					RemoveFloatingTextReason(ftext, "lifetime")
					return
				end
			end
			ftext:SetVisible(true)
		end
	end

	-- Set box directly outside of the layout system
	local x1, y1, x2, y2 = ScaleXY(ftext.scale, ftext.Padding:xyxy())
	ftext:SetBox(xLoc - x1, yLoc - y1, width + maxx + x1 + x2, height + maxy + y1 + y2, false)
	ftext.Dock = "ignore"
	local max_cam_dist_m = config.FloatingTextMaxDist_m
	local targetIsPoint = IsPoint(target)
	if not targetIsPoint and not (IsValid(target) and target:IsValidPos()) then
		target = backupPosition
		targetIsPoint = true
	end
	if spot and not targetIsPoint then
		ftext:AddDynamicPosModifier{
			id = "attached_ui",
			target = target,
			spot_type = EntitySpots[spot],
			max_cam_dist_m = max_cam_dist_m,
		}
	else
		if targetIsPoint then
			ftext:AddDynamicPosModifier{
				id = "attached_ui",
				target = ValidateZ(target),
				max_cam_dist_m = max_cam_dist_m,
			}
		else
			-- If a target, but without a specified spot, randomize a position around.
			local posx, posy, posz = lFindTextPos(target, ftext.z_offset)
			ftext:AddDynamicPosModifier{
				id = "attached_ui",
				target = point(posx, posy, posz),
				max_cam_dist_m = max_cam_dist_m,
			}
		end
	end
	
	if ftext.expire_time then
		ftext:CreateThread("floating_text_interp", function()
			ftext:StartInterpolation(game_time)
			Sleep(ftext.expire_time)
			RemoveFloatingTextReason(ftext, "expired")
		end)
	end
end

--- Create floating text at the specified pos.
-- @param target point || CObject The point or object to place the text above.
-- @param text T || string The text to display.
-- @param style string The text style to use, this will override the one specified in the template.
-- @param spot string Optional, the target's spot to attach to.
function CreateFloatingText(target, text, style, spot, stagger_spawn, params, game_time)
	return CreateCustomFloatingText(nil, target, text, style, spot, stagger_spawn, params, game_time)
end

local b = box(0, 0, 1, 1)

function XFloatingText:StartInterpolation(game_time)
	if self.interpolate_opacity then
		local transInter = {
			id = "transparency",
			type = const.intAlpha,
			startValue = self.transparency_start,
			endValue = self.transparency_end,
			start = (game_time and GameTime() or GetPreciseTicks()) + self.fade_start,
			duration = self.expire_time - self.fade_start,
			flags = game_time and const.intfGameTime or nil,
		}
		self:AddInterpolation(transInter)
	end
	if self.interpolate_pos then
		local _, offset_y = ScaleXY(self.scale, 0, self.offset_amount)
		local moveInterp = {
			id = "movement",
			type = const.intRect,
			originalRect = b,
			targetRect = box(b:minx(), b:miny() - offset_y, b:maxx(), b:maxy() - offset_y),
			start = (game_time and GameTime() or GetPreciseTicks()) + self.offset_start_time,
			duration = self.expire_time - self.offset_start_time,
			flags = game_time and const.intfGameTime or nil,
		}
		self:AddInterpolation(moveInterp)
	end
end

DefineClass.FloatingTextDialog = {
	__parents = { "XDialog" },
	UseClipBox = false,
	ZOrder = 0,
	FocusOnOpen = false,
}

function FloatingTextDialog:Measure(max_width, max_height)
	return 0, 0 -- prevent calls to measure all children
end

function FloatingTextDialog:UpdateLayout()
	self.layout_update = false -- do nothing, all floating text set their boxes manually
end

function ShowFloatingTextNoExpire(actor, text, style)
	return CreateFloatingText(actor, text, style, nil, nil, { expire_time = false })
end

function OnMsg.DoneMap()
	for _, texts in pairs(FloatingTexts or empty_table) do
		for _, text in ipairs(texts) do
			if text.window_state ~= "destroying" then
				text:delete()
			end
		end
	end
end

function RemoveFloatingTextsFrom(obj, except)
	local list = FloatingTexts[obj]
	if not list then return end
	for i = #list, 1, -1 do
		local text = list[i]
		if (not except or not IsKindOf(text, except)) and text.window_state ~= "destroying" then
			text:delete()
			text.dbg_removed_by = "call to RemoveFloatingTextsFrom"
			table.remove(list, i)
		end
	end
end