File size: 9,235 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
----- XEditorTool
-- Base class for all tools that supports:
--  1. Open/close tool setting using Ged (hosted in-game by default)
--  2. Restore focus back to the tool window when mouse leaves the settings
--  3. Context menu

DefineClass.XEditorTool = {
	__parents = { "XDialog", "XEditorToolSettings" },
	properties = {
		{ id = "DisplayInGame", editor = "bool", default = true, shared_setting = true, no_edit = true, },
	},
	HandleMouse = true,
	UsesCodeRenderables = false,
	
	-- refedine in child classes
	ToolTitle = "None",
	ToolSection = false,
	ToolKeepSelection = false, -- don't clear the selection when switching to this tool
	Description = false,
	ActionSortKey = "99",
	ActionIcon = false,
	ActionShortcut = false,
	ActionShortcut2 = false,
	FocusPropertyInSettings = false, -- text property to auto-focus in the settings, e.g. place object's Filter field
	FocusPropertySingleTime = false,
	PropertyTabs = false,
}

function XEditorTool:SetContext(context)
	-- set tool class members via the context, e.g. ToolTitle
	for key, value in pairs(context or empty_table) do
		if self:HasMember(key) then
			self[key] = value
		end
	end
end

function XEditorTool:OnMouseButtonDown(pos, button)
	if not camera.IsLocked(1) and button == "R" then
		-- right-click closes every tool other than "Select Objects"
		if not XEditorIsDefaultTool() then
			XEditorSetDefaultTool()
			return "break"
		end
		
		-- select object under cursor ONLY if there is no selection (motivation - some objects are hard to select by clicking on them)
		if #editor.GetSel() == 0 then
			local obj = GetObjectAtCursor()
			if obj then
				editor.AddToSel(editor.SelectionPropagate({obj}))
			end
		end
		
		-- open object properties or display context menu
		local sel_count = #editor.GetSel()
		local open_context_menu = terminal.IsKeyPressed(const.vkControl) == (EditorSettings:GetCtrlRightClickOpens() == "ContextMenu")
		if open_context_menu then
			local context = "NoSelection"
			if sel_count == 1 then
				context = "SingleSelection"
			elseif sel_count > 1 then
				context = "MultipleSelection"
			end
			XEditorOpenContextMenu(context, pos)
		elseif sel_count > 0 then
			OpenGedGameObjectEditor(editor.GetSel())
		end
		return "break"
	end
end

function XEditorTool:OnShortcut(shortcut, source, ...)
	if shortcut == "Escape" and not XEditorIsDefaultTool() then
		XEditorSetDefaultTool()
		return "break"
	end
	return XDialog.OnShortcut(self, shortcut, source, ...)
end


----- Editor settings panel update thread
-- 1. Opens/closes the GedApp for editing tool settings when necessary
-- 2. Reopens it when the DisplayInGame property changes
-- 3. Updates its root object when the editor tool changes
-- 4. Manages focus between the tool settings, the editor toolbar/statusbar, and the editor tool window

if FirstLoad then
	XEditorSettingsGed = false
	XEditorSettingsJustOpened = false
	XEditorSettingsUpdateThread = false
end

function OnMsg.GameEnterEditor()
	if not IsValidThread(XEditorSettingsUpdateThread) then
		XEditorSettingsUpdateThread = CreateRealTimeThread(function()
			while true do
				XEditorSettingsUpdate()
				Sleep(100)
			end
		end)
	end
end

local function editable_prop_count(class)
	local count = 0
	for _, prop in ipairs(class:GetProperties()) do
		if not prop_eval(prop.no_edit, class, prop) then
			count = count + 1
		end
	end
	return count
end

local function room_tools_visible(visible)
	local room_tools = GetDialog("XEditorRoomTools")
	if room_tools then
		room_tools:SetVisibleInstant(visible)
	end
end

function XEditorSettingsUpdate()
	local editor_tool = XEditorGetCurrentTool()
	
	local should_open = editor_tool and editable_prop_count(editor_tool) ~= 0
	if not XEditorSettingsGed and should_open then
		XEditorSettingsGed = OpenGedApp("XEditorToolSettingsPanel", editor_tool, nil, nil, editor_tool:GetDisplayInGame())
		room_tools_visible(not editor_tool:GetDisplayInGame())
	end
	
	if XEditorSettingsGed then
		if not should_open or editor_tool:GetDisplayInGame() ~= XEditorSettingsGed.context.in_game then
			local settings = XEditorSettingsGed
			XEditorSettingsGed = false
			CloseGedApp(settings, "wait")
			room_tools_visible(true)
		elseif XEditorSettingsGed:ResolveObj("root") ~= editor_tool then
			XEditorSettingsGed:BindObj("root", editor_tool)
		end
	end
	
	local desktop = terminal.desktop
	local focused_ctrl = desktop:GetKeyboardFocus()
	if desktop:GetMouseTarget(terminal.GetMousePos()) == editor_tool or XEditorSettingsJustOpened then
		if XEditorSettingsGed and XEditorSettingsGed.context.in_game then
			-- auto-focus a specified text field, e.g. the Filter text field for placing objects
			if editor_tool.FocusPropertyInSettings then
				local settings_dialog = GetDialog("XEditorToolSettingsPanel")
				local prop_editor = settings_dialog and settings_dialog.idPropPanel:LocateEditorById(editor_tool.FocusPropertyInSettings)
				if prop_editor and focused_ctrl ~= prop_editor.idEdit and (not editor_tool.FocusPropertySingleTime or XEditorSettingsJustOpened) then
					desktop:RemoveKeyboardFocus(settings_dialog, true)
					desktop:SetKeyboardFocus(prop_editor.idEdit)
				end
			else -- restore focus to XEditorTool when the mouse leaves the XEditorToolSettingsPanel window
				local focused_app = GetParentOfKind(focused_ctrl, "GedApp")
				if focused_app and focused_app.AppId == "XEditorToolSettingsPanel" then
					desktop:RemoveKeyboardFocus(focused_app, true)
				end
			end
		end
		
		XEditorRemoveFocusFromToolbars()
		XEditorSettingsJustOpened = false
	end
end

function OnMsg.GedClosing(id)
	-- settings closed out of game?
	if XEditorSettingsGed and GedConnections[id].app_template == "XEditorToolSettingsPanel" then
		local editor_tool = XEditorGetCurrentTool()
		if editor_tool then
			editor_tool:SetDisplayInGame(true)
		end
	end
end

function GedXEditorSettingsDescription()
	if XEditorIsDefaultTool() then return "" end
	local editor_tool = XEditorGetCurrentTool()
	local descr = editor_tool and editor_tool.Description
	descr = (type(descr) == "function") and descr(editor_tool) or descr
	descr = (type(descr) == "table") and descr or {descr}
	return descr and table.concat(descr, "\n\n") or ""
end


----- Toolbar

function OnMsg.ShortcutsReloaded()
	local action_toolbar_section = "No Section"
	for _, class_name in ipairs(ClassLeafDescendantsList("XEditorTool")) do
		local tool = g_Classes[class_name]
		if tool.ActionIcon then
			local shortcut = tool.ActionShortcut and " (" .. tool.ActionShortcut .. ")" or ""
			local shortcut2 = tool.ActionShortcut2 and " (" .. tool.ActionShortcut2 .. ")" or ""
			local description = type(tool.Description) == "function" and tool.Description() or tool.Description
			local titleStyle = GetDarkModeSetting() and "<style XEditorRolloverBoldDark>" or "<style XEditorRolloverBold>"
			local rolloverText = titleStyle .. tool.ToolTitle .. shortcut .. shortcut2 .. "</style>\n" .. table.concat(description, "\n\n") 
			XAction:new({
				ActionId = tool.ToolTitle,
				ActionMode = "Editor",
				ActionName = tool.ToolTitle,
				ActionTranslate = false,
				ActionIcon = tool.ActionIcon,
				ActionShortcut = tool.ActionShortcut,
				ActionShortcut2 = tool.ActionShortcut2,
				ActionSortKey = tool.ActionSortKey,
				ActionToolbarSection = tool.ToolSection or action_toolbar_section,
				RolloverText = rolloverText,
				ActionToolbar = "XEditorToolbar",
				ActionToggle = true,
				ActionToggled = function(self, host)
					return GetDialogMode("XEditor") == class_name
				end,
				ActionState = tool.ToolActionState or empty_func,
				OnAction = function(self, host)
					if GetDialogMode("XEditor") ~= class_name then
						SetDialogMode("XEditor", class_name)
					end
				end,
			}, XShortcutsTarget)
		end
	end
	
	for _, class_name in ipairs(ClassLeafDescendantsList("XEditorPlacementHelper")) do
		local tool = g_Classes[class_name]
		if tool.Title ~= "None" and tool.ActionIcon then
			local shortcut = tool.ActionShortcut and " (" .. tool.ActionShortcut .. ")" or ""
			local rolloverText = "<style XEditorRolloverBold>" .. tool.Title .. "</style>"
			XAction:new({
				ActionId = tool.Title,
				ActionMode = "Editor",
				ActionName = tool.Title,  
				ActionIcon = tool.ActionIcon, 
				ActionShortcut = tool.ActionShortcut,
				ActionShortcut2 = tool.ActionShortcut2,
				ActionSortKey = tool.ActionSortKey,
				RolloverText = rolloverText,
				ActionToolbar = "XEditorStatusbar",
				ActionToggle = true,
				ActionToggled = function(self, host)
					local dialog = GetDialog("XSelectObjectsTool") or GetDialog("XPlaceObjectTool")
					return dialog and dialog:GetHelperClass() == class_name
				end,
				ActionState = function(self, host)
					if GetDialog("XSelectObjectsTool") and tool.InXSelectObjectsTool then return true
					elseif GetDialog("XPlaceObjectTool") and tool.InXPlaceObjectTool then return true end
					return "hidden"
				end,
				OnAction = function(self, host)
					local dialog = GetDialog("XSelectObjectsTool") or GetDialog("XPlaceObjectTool")
					if not dialog then
						SetDialogMode("XEditor", "XSelectObjectsTool")
						dialog = GetDialog("XSelectObjectsTool")
					end
					GetDialog(dialog.class):SetHelperClass(class_name)
				end,
			}, XShortcutsTarget)
		end
	end	
end