-
-
Notifications
You must be signed in to change notification settings - Fork 24
/
Copy pathMainTabWindow_ResearchTree.cs
373 lines (305 loc) · 12.4 KB
/
MainTabWindow_ResearchTree.cs
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
// MainTabWindow_ResearchTree.cs
// Copyright Karel Kroeze, 2020-2020
using System.Collections.Generic;
using System.Linq;
using RimWorld;
using UnityEngine;
using Verse;
using static FluffyResearchTree.Constants;
namespace FluffyResearchTree
{
public class MainTabWindow_ResearchTree : MainTabWindow
{
internal static Vector2 _scrollPosition = Vector2.zero;
private static Rect _treeRect;
private Rect _baseViewRect;
private Rect _baseViewRect_Inner;
private bool _dragging;
private Vector2 _mousePosition = Vector2.zero;
private string _query = "";
private Rect _viewRect;
private Rect _viewRect_Inner;
private bool _viewRect_InnerDirty = true;
private bool _viewRectDirty = true;
private float _zoomLevel = 1f;
public MainTabWindow_ResearchTree()
{
closeOnClickedOutside = false;
Instance = this;
}
public static MainTabWindow_ResearchTree Instance { get; private set; }
public float ScaledMargin => Constants.Margin * ZoomLevel / Prefs.UIScale;
public float ZoomLevel
{
get => _zoomLevel;
set
{
_zoomLevel = Mathf.Clamp( value, 1f, MaxZoomLevel );
_viewRectDirty = true;
_viewRect_InnerDirty = true;
}
}
public Rect ViewRect
{
get
{
if ( _viewRectDirty )
{
_viewRect = new Rect(
_baseViewRect.xMin * ZoomLevel,
_baseViewRect.yMin * ZoomLevel,
_baseViewRect.width * ZoomLevel,
_baseViewRect.height * ZoomLevel
);
_viewRectDirty = false;
}
return _viewRect;
}
}
public Rect ViewRect_Inner
{
get
{
if ( _viewRect_InnerDirty )
{
_viewRect_Inner = _viewRect.ContractedBy( Margin * ZoomLevel );
_viewRect_InnerDirty = false;
}
return _viewRect_Inner;
}
}
public Rect TreeRect
{
get
{
if ( _treeRect == default )
{
var width = Tree.Size.x * ( NodeSize.x + NodeMargins.x );
var height = Tree.Size.z * ( NodeSize.y + NodeMargins.y );
_treeRect = new Rect( 0f, 0f, width, height );
}
return _treeRect;
}
}
public Rect VisibleRect =>
new Rect(
_scrollPosition.x,
_scrollPosition.y,
ViewRect_Inner.width,
ViewRect_Inner.height );
internal float MaxZoomLevel
{
get
{
// get the minimum zoom level at which the entire tree fits onto the screen, or a static maximum zoom level.
var fitZoomLevel = Mathf.Max( TreeRect.width / _baseViewRect_Inner.width,
TreeRect.height / _baseViewRect_Inner.height );
return Mathf.Min( fitZoomLevel, AbsoluteMaxZoomLevel );
}
}
public override void PreClose()
{
base.PreClose();
Log.Debug( "CloseOnClickedOutside: {0}", closeOnClickedOutside );
Log.Debug( StackTraceUtility.ExtractStackTrace() );
}
public void Notify_TreeInitialized()
{
SetRects();
}
public override void PreOpen()
{
base.PreOpen();
SetRects();
if ( !Tree.Initialized )
// initialize tree
Tree.Initialize();
// clear node availability caches
ResearchNode.ClearCaches();
_dragging = false;
closeOnClickedOutside = false;
}
private void SetRects()
{
// tree view rects, have to deal with UIScale and ZoomLevel manually.
_baseViewRect = new Rect(
StandardMargin / Prefs.UIScale,
( TopBarHeight + Constants.Margin + StandardMargin ) / Prefs.UIScale,
( Screen.width - StandardMargin * 2f ) / Prefs.UIScale,
( Screen.height - MainButtonDef.ButtonHeight - StandardMargin * 2f - TopBarHeight - Constants.Margin ) /
Prefs.UIScale );
_baseViewRect_Inner = _baseViewRect.ContractedBy( Constants.Margin / Prefs.UIScale );
// windowrect, set to topleft (for some reason vanilla alignment overlaps bottom buttons).
windowRect.x = 0f;
windowRect.y = 0f;
windowRect.width = UI.screenWidth;
windowRect.height = UI.screenHeight - MainButtonDef.ButtonHeight;
}
public override void DoWindowContents( Rect canvas )
{
if ( !Tree.Initialized )
return;
// top bar
var topRect = new Rect(
canvas.xMin,
canvas.yMin,
canvas.width,
TopBarHeight );
DrawTopBar( topRect );
ApplyZoomLevel();
// draw background
GUI.DrawTexture( ViewRect, Assets.SlightlyDarkBackground );
// draw the actual tree
// TODO: stop scrollbars scaling with zoom
_scrollPosition = GUI.BeginScrollView( ViewRect, _scrollPosition, TreeRect );
GUI.BeginGroup(
new Rect(
ScaledMargin,
ScaledMargin,
TreeRect.width + ScaledMargin * 2f,
TreeRect.height + ScaledMargin * 2f
)
);
Tree.Draw( VisibleRect );
Queue.DrawLabels( VisibleRect );
HandleZoom();
GUI.EndGroup();
GUI.EndScrollView( false );
HandleDragging();
HandleDolly();
// reset zoom level
ResetZoomLevel();
// cleanup;
GUI.color = Color.white;
Text.Anchor = TextAnchor.UpperLeft;
}
private void HandleDolly()
{
var dollySpeed = 10f;
if ( KeyBindingDefOf.MapDolly_Left.IsDown )
_scrollPosition.x -= dollySpeed;
if ( KeyBindingDefOf.MapDolly_Right.IsDown )
_scrollPosition.x += dollySpeed;
if ( KeyBindingDefOf.MapDolly_Up.IsDown )
_scrollPosition.y -= dollySpeed;
if ( KeyBindingDefOf.MapDolly_Down.IsDown )
_scrollPosition.y += dollySpeed;
}
private void HandleZoom()
{
// handle zoom
if ( Event.current.isScrollWheel )
{
// absolute position of mouse on research tree
var absPos = Event.current.mousePosition;
// Log.Debug( "Absolute position: {0}", absPos );
// relative normalized position of mouse on visible tree
var relPos = ( Event.current.mousePosition - _scrollPosition ) / ZoomLevel;
// Log.Debug( "Normalized position: {0}", relPos );
// update zoom level
ZoomLevel += Event.current.delta.y * ZoomStep * ZoomLevel;
// we want to keep the _normalized_ relative position the same as before zooming
_scrollPosition = absPos - relPos * ZoomLevel;
Event.current.Use();
}
}
private void HandleDragging()
{
if ( Event.current.type == EventType.MouseDown )
{
_dragging = true;
_mousePosition = Event.current.mousePosition;
Event.current.Use();
}
if ( Event.current.type == EventType.MouseUp )
{
_dragging = false;
_mousePosition = Vector2.zero;
}
if ( Event.current.type == EventType.MouseDrag )
{
var _currentMousePosition = Event.current.mousePosition;
_scrollPosition += _mousePosition - _currentMousePosition;
_mousePosition = _currentMousePosition;
}
}
private void ApplyZoomLevel()
{
GUI.EndClip(); // window contents
GUI.EndClip(); // window itself?
GUI.matrix = Matrix4x4.TRS( new Vector3( 0f, 0f, 0f ), Quaternion.identity,
new Vector3( Prefs.UIScale / ZoomLevel, Prefs.UIScale / ZoomLevel, 1f ) );
}
private void ResetZoomLevel()
{
// dummies to maintain correct stack size
// TODO; figure out how to get actual clipping rects in ApplyZoomLevel();
UI.ApplyUIScale();
GUI.BeginClip( windowRect );
GUI.BeginClip( new Rect( 0f, 0f, UI.screenWidth, UI.screenHeight ) );
}
private void DrawTopBar( Rect canvas )
{
var searchRect = canvas;
var queueRect = canvas;
searchRect.width = 200f;
queueRect.xMin += 200f + Constants.Margin;
GUI.DrawTexture( searchRect, Assets.SlightlyDarkBackground );
GUI.DrawTexture( queueRect, Assets.SlightlyDarkBackground );
DrawSearchBar( searchRect.ContractedBy( Constants.Margin ) );
Queue.DrawQueue( queueRect.ContractedBy( Constants.Margin ), !_dragging );
}
private void DrawSearchBar( Rect canvas )
{
Profiler.Start( "DrawSearchBar" );
var iconRect = new Rect(
canvas.xMax - Constants.Margin - 16f,
0f,
16f,
16f )
.CenteredOnYIn( canvas );
var searchRect = new Rect(
canvas.xMin,
0f,
canvas.width,
30f )
.CenteredOnYIn( canvas );
GUI.DrawTexture( iconRect, Assets.Search );
var query = Widgets.TextField( searchRect, _query );
if ( query != _query )
{
_query = query;
Find.WindowStack.FloatMenu?.Close( false );
if ( query.Length > 2 )
{
// open float menu with search results, if any.
var options = new List<FloatMenuOption>();
foreach ( var result in Tree.Nodes.OfType<ResearchNode>()
.Select( n => new {node = n, match = n.Matches( query )} )
.Where( result => result.match > 0 )
.OrderBy( result => result.match ) )
options.Add( new FloatMenuOption( result.node.Label, () => CenterOn( result.node ),
MenuOptionPriority.Default, () => CenterOn( result.node ) ) );
if ( !options.Any() )
options.Add( new FloatMenuOption( "Fluffy.ResearchTree.NoResearchFound".Translate(), null ) );
Find.WindowStack.Add( new FloatMenu_Fixed( options,
UI.GUIToScreenPoint(
new Vector2(
searchRect.xMin, searchRect.yMax ) ) ) );
}
}
Profiler.End();
}
public void CenterOn( Node node )
{
var position = new Vector2(
( NodeSize.x + NodeMargins.x ) * ( node.X - .5f ),
( NodeSize.y + NodeMargins.y ) * ( node.Y - .5f ) );
node.Highlighted = true;
position -= new Vector2( UI.screenWidth, UI.screenHeight ) / 2f;
position.x = Mathf.Clamp( position.x, 0f, TreeRect.width - ViewRect.width );
position.y = Mathf.Clamp( position.y, 0f, TreeRect.height - ViewRect.height );
_scrollPosition = position;
}
}
}