-
Notifications
You must be signed in to change notification settings - Fork 0
/
EBT2.ks
435 lines (344 loc) · 13 KB
/
EBT2.ks
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
423
424
425
426
427
428
429
430
431
432
433
434
435
//**************************************************************************************************
//
// kOS ENGINE BALANCING TOOL
//
// =================================================================================================
//
// History
// --------
// v0.81 Comments now all in english. Minor code improvements. Thresholds for acceptable
// leveragae magnitude changed so that %thrustlimit update steps.
// v0.8 First public release.
//
//**************************************************************************************************
CLEARSCREEN.
CLEARVECDRAWS().
GLOBAL PROGRAM_RELEASE to "v0.81".
GLOBAL GUI_WIDTH to 330.
GLOBAL GUI_BOX2_WIDTH to 313.
GLOBAL GUI_BOX2_HEIGTH to 70.
GLOBAL GUI_MAX_MESSAGES to 10.
GLOBAL GUI_FONT_MESSAGES to "Courier New".
GLOBAL GUI_ON_LABEL to "<color=#00ff00><b>ACTIVE</b></color>".
GLOBAL GUI_OFF_LABEL to "<color=#880000><b>OFF</b></color>".
GLOBAL GUI_TITLE_LABEL to "<b><size=14><color=#ffaa00FF>Engine Balancing Tool</color></size></b>".
GLOBAL GUI_LEGEND_LABEL to "<color=#ffffff>©Kerbal Foundries</color>".
// GLOBAL for Balancing behaviour Active / OFF
GLOBAL isActive to false.
// GLOBAL for Exit behaviour
GLOBAL isFinished to false.
// Messages Management
set message_list to list().
// Defining the GUI
set mygui to GUI(GUI_WIDTH).
set box1 to mygui:ADDHLAYOUT().
set mytitle to box1:addlabel(GUI_TITLE_LABEL).
set button_onoff to box1:addbutton(GUI_OFF_LABEL).
set box2 to mygui:ADDHLAYOUT().
set box2:style:width to GUI_BOX2_WIDTH.
set box2:style:height to GUI_BOX2_HEIGTH.
set sb to box2:addscrollbox().
set sb:style:margin:H to 0.
set sb:style:margin:V to 0.
set sb:style:padding:H to 0.
set sb:style:padding:V to 0.
set sb:VALWAYS to true.
set box3 to mygui:ADDHLAYOUT().
set button_hide to box3:addbutton("Hide").
set mylegend to box3:addlabel(GUI_LEGEND_LABEL).
set mylegend:STYLE:ALIGN TO "CENTER".
set button_exit to box3:addbutton("Exit").
// The second window
GLOBAL MINIGUI_WIDTH to 62.
set myminigui to GUI(MINIGUI_WIDTH).
set button_show to myminigui:addbutton("Show").
// --------------------------------------------------------------------------
//
// GUI Functions
//
//
function addNewMessage
{
parameter message.
set nb to message_list:length.
message_list:add(sb:addlabel(message)).
set message_list[nb]:style:font to GUI_FONT_MESSAGES.
if message_list:length > GUI_MAX_MESSAGES
{
message_list[0]:dispose().
message_list:remove(0).
}
} // addNewMessage()
function cbOnOffButton
{
if isActive
{
set isActive to false.
set button_onoff:text to GUI_OFF_LABEL.
}
else
{
set isActive to true.
set button_onoff:text to GUI_ON_LABEL.
}
} // cbOnOffButton()
function cbExitButton
{
set isFinished to true.
} // cbExitButton()
function cbHideButton
{
set myminigui:x to mygui:x.
set myminigui:y to mygui:y.
set mygui:x to myminigui:x.
set mygui:y to myminigui:y.
myminigui:SHOW().
mygui:HIDE().
} // cbHideButton()
function cbShowButton
{
set mygui:x to myminigui:x.
set mygui:y to myminigui:y.
myminigui:HIDE().
mygui:SHOW().
} // cbShowButton()
SET button_onoff:ONCLICK TO cbOnOffButton@.
SET button_exit:ONCLICK TO cbExitButton@.
SET button_hide:ONCLICK TO cbHideButton@.
SET button_show:ONCLICK TO cbShowButton@.
// ----------------- END GUI DESIGNING -----------------
// --------------------------------------------------------------------------
//
// BALANCING Variables
//
//
global partInCommandUID to 0.
global CoM to SHIP:position.
GLOBAL MAX_ENGINE_ANGLE to 45.
GLOBAL MAX_DEVIATON TO 0.50. // meters
GLOBAL AVG_DEVIATON TO 0.10. // meters
GLOBAL MIN_DEVIATON TO 0.01. // meters
GLOBAL THRUST_PCT_MAXSTEP TO 2. // %thrustlimit is updated per 2% step
GLOBAL THRUST_PCT_AVGSTEP TO 0.5. // %thrustlimit is updated per 0.5% step
GLOBAL THRUST_PCT_MINSTEP TO 0.25. // %thrustlimit is updated per 0.25% step
global current_thrust_pct_step to THRUST_PCT_MAXSTEP.
// --------------------------------------------------------------------------
//
// BALANCING Functions
//
//
function identify_engines {
// Returns a list of usable engines for thrust balancing purposes.
// Conditions for an engine being accepted
// 1/ Engine has been activated
// 2/ Engine thrust is aligned (-45° < x < 45°) with the part in control facing vector.
// Retuns an empty list is no engine has been found
if ( SHIP:CONTROLPART:UID <> partInCommandUID )
{
addNewMessage("Part in command is " + SHIP:CONTROLPART:name).
set partInCommandUID to SHIP:CONTROLPART:UID.
}
set partInCommand to SHIP:CONTROLPART.
list ENGINES in initial_list.
local final_list to list().
local current_max_thrust to 0.
for i_engine in initial_list
{
// Condition is Engine has been activated AND orientation is within a 45° range to the direction pointed by the part in controll
if ( i_engine:ignition and abs(vectorangle(i_engine:facing:forevector, partInCommand:facing:forevector)) < MAX_ENGINE_ANGLE)
{
final_list:add(i_engine).
set current_max_thrust to current_max_thrust + i_engine:possiblethrust.
}
}
if ( final_list:length() = 0 )
{
addNewMessage("<color=#ff0000><b>Can't work with these " + engineList:length() + " engine</b></color>").
return list().
}
if ( current_max_thrust = 0 )
// If there is no thrust available: no balancing could be done.
{
addNewMessage("<color=#ff0000><b>Total thrust is 0!</b></color>").
addNewMessage("<color=#ff0000><b>Check your engines!</b></color>").
return list().
}
return final_list.
} // identify_engines()
function getThrustVector {
// Returns a vec (relative xyz) of the global Thrust
// engines are considered active
// value is calculated upon maximum possible thrust per engine with %throttle limit
// Must be called with at least 2 engines in engineList
local v_g_thrust to engineList[0]:facing:forevector * engineList[0]:possiblethrust.
local sublistengines to engineList:sublist(2, engineList:length()).
for i_engine in sublistengines {
set v_g_thrust to v_g_thrust + i_engine:facing:forevector * i_engine:possiblethrust.
}
return v_g_thrust.
} // getThrustVector()
function getCoT {
// Returns a vec of CoT coordonates
// The CoT is like a weighted average calculation based on
// - Engine positions
// - Effective thrust of engines regarding the global Thrust
// Must be called with a VGThrust:MAG > 0
local tCoT to v(0,0,0).
for i_engine in engineList {
// engine points towards i_engine:facing:forevector
// What we want is the angle between this vector and the global thrust vector
set i_engine_angle to vectorangle(VGThrust, i_engine:facing:forevector).
// Now the effective thrust of this engine regarding the GThrust Vector is
// its own thrust * the cosinus of this angle = effective weight
set tCoT to tCoT + i_engine:position * i_engine:possiblethrust * cos(i_engine_angle).
//print "DEBUG angle of engine VS VGThrust = " + i_engine_angle.
}
// We have now to divide tCoT by VGThrust:MAG which is the sum of every effective weight
set tCoT to tCoT / VGThrust:MAG.
return tCot.
} // getCoT()
function getVLeverage {
// X is the nearest point to CoM on the Thrust line passing through the CoT
// The function return the Vector from CoM to X
// X:MAG is the distance in meters between CoM and the GThrust Vector
local X to CoT + (CoM-CoT):MAG * cos (CoMToThrustAngle) * VGThrust:normalized.
return (X - CoM).
} // getVLeverage()
function strenghtenEngineIfPossible {
// Find an engine which thrust can be increased.
// Performs the operation if doable
// Return a True or False accordingly to what it did
local max_angle to 0.
local target_engine to 0.
local was_set to false.
for i_engine in engineList
{
// calculate angle from CoM->X and CoM->i_engine
local i_engine_angle to vectorangle(VLeverage, (i_engine:position - CoM)).
// We have to identify the engine at the "most" opposite to the thrust vector
if i_engine_angle > max_angle {
set target_engine to i_engine.
set max_angle to i_engine_angle.
}
}
if target_engine:thrustlimit < 100
// Engine identified AND tweakable
{
set target_engine:thrustlimit to target_engine:thrustlimit + current_thrust_pct_step.
set was_set to TRUE.
print "Engine " + target_engine:tag + " ++ to " + target_engine:thrustlimit.
}
return was_set.
} // getIndexOfEngineToBeStrenghtened()
function weakenEngineIfPossible {
// Find an engine which thrust can be decreased.
// Performs the operation if doable
// Return a True or False accordingly to what it did
local max_angle to 360.
local target_engine to 0.
local was_set to false.
for i_engine in engineList
{
// calculate angle from CoM->X and CoM->i_engine
local i_engine_angle to vectorangle(VLeverage, (i_engine:position - CoM)).
// We have to identify the engine the closest to the thrust vector
if i_engine_angle < max_angle {
set target_engine to i_engine.
set max_angle to i_engine_angle.
}
}
if target_engine:thrustlimit > 0
// Engine identified AND tweakable
{
set target_engine:thrustlimit to target_engine:thrustlimit - current_thrust_pct_step.
set was_set to TRUE.
print "Engine " + target_engine:tag + " -- to " + target_engine:thrustlimit.
}
return was_set.
} // getIndexOfEngineToBeStrenghtened()
// MAIN
myGui:show().
addNewMessage("Program loaded " + PROGRAM_RELEASE).
// Engine Balancing Algorithm to align Thrust on CoM
//
// 1/ a/ Find the CoM (that's should be ok :o)
// b/ Calculate the Global Thrust Vector based on individual engine thrust, position and facing
// c/ Calculate the center of Thrust
// d/ Find what I call the Leverage Vector who is responsible for the leverage behaviour.
// The Leverage Vector is orthogonal to the Global Thrust Vector and starts at the CoM.
// To avoid leverage, this vector should have a magnitude of 0
//
// 2/ In a loop which shall not exceed a maximum of iterations do
// if the deviation is > MAXIMAL_DEVIATION_ALLOWED
// a/ Does an engine exist in an opposit direction of the Deviation Vector ?
// Does its thrustlimit can be increased ?
// b/ Increase its %thrustlimit a little bit
// c/ Recalculate the data found in 1/
// d/ loop on 2a
//
// e/ If no engine can be boosted, is there an engine in the direction of the leverage vector whose %thrustlimit can be lowered ?
// f/ Lower the %thrustlimit a little bit
// g/ Recalculate the data found in 1/
// h/ loop on 2e
//
// 3/ wait 0.1s
// 4/ loop on 2/
global engineList to list().
global VGThrust to v(0,0,0).
global CoT to v(0,0,0).
global CoMToThrustAngle to v(0,0,0).
global VLeverage to v(0,0,0).
function update_globals {
set CoM to ship:position.
set VGThrust to getThrustVector().
set CoT to getCoT().
set CoMToThrustAngle to vectorangle(CoM-CoT, VGThrust).
set VLeverage to getVLeverage().
if VLeverage:MAG < MIN_DEVIATON
{
set current_thrust_pct_step to THRUST_PCT_MINSTEP.
//print "MINSTEP".
}
else if VLeverage:MAG < AVG_DEVIATON
{
set current_thrust_pct_step to THRUST_PCT_AVGSTEP.
//print "AVGSTEP".
}
else
{
set current_thrust_pct_step to THRUST_PCT_MAXSTEP.
//print "MAXSTEP".
}
} // update_globals()
// THE LOOP
UNTIL isFinished
{
UNTIL isFinished or not isActive
// ending condition is isFinished & not active
{
// Get fresh information about the current situation.
set engineList to identify_engines().
if ( engineList:length() <= 1)
{
cbOnOffButton().
break.
}
update_globals().
// 2 - Loop on engines to be boosted
UNTIL not isActive or (VLeverage:MAG < MIN_DEVIATON) or NOT strenghtenEngineIfPossible() {
update_globals().
//print "DEBUG : less".
print "VLeverage:MAG=" + VLeverage:MAG*100 + "cm".
}
// 3 - Loop on engines to be nerfed
UNTIL not isActive or (VLeverage:MAG < MIN_DEVIATON) or NOT weakenEngineIfPossible() {
update_globals().
//print "DEBUG : more".
print "VLeverage:MAG=" + VLeverage:MAG*100 + "cm".
}
wait 0.1.
} // Engine balancing loop
wait 1.
} // THE PROGRAM LOOP
mygui:dispose().
myminigui:dispose().