forked from abrignoni/iLEAPP
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathileappGUI.py
552 lines (473 loc) · 24.4 KB
/
ileappGUI.py
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
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
import tkinter as tk
import typing
import json
import ileapp
import webbrowser
import scripts.plugin_loader as plugin_loader
from tkinter import ttk, filedialog as tk_filedialog, messagebox as tk_msgbox
from scripts.version_info import ileapp_version
from scripts.search_files import *
from scripts.tz_offset import tzvalues
from scripts.modules_to_exclude import modules_to_exclude
from scripts.lavafuncs import *
def pickModules():
'''Create a list of available modules:
- iTunesBackupInfo, lastBuild, Ph100-UFED-device-values-Plist that need to be executed first are excluded
- ones that take a long time to run are deselected by default'''
global mlist
for plugin in sorted(loader.plugins, key=lambda p: p.category.upper()):
if (plugin.name == 'iTunesBackupInfo'
or plugin.name == 'lastBuild'
or plugin.name == 'Ph100-UFED-device-values-Plist'):
continue
# Items that take a long time to execute are deselected by default
# and referenced in the modules_to_exclude list in an external file (modules_to_exclude.py).
plugin_enabled = tk.BooleanVar(value=False) if plugin.module_name in modules_to_exclude else tk.BooleanVar(value=True)
plugin_module_name = plugin.artifact_info.get('name', plugin.name) if hasattr(plugin, 'artifact_info') else plugin.name
mlist[plugin.name] = [plugin.category, plugin_module_name, plugin.module_name, plugin_enabled]
def get_selected_modules():
'''Update the number and return the list of selected modules'''
selected_modules = []
for artifact_name, module_infos in mlist.items():
if module_infos[-1].get():
selected_modules.append(artifact_name)
selected_modules_label.config(text=f'Number of selected modules: {len(selected_modules)} / {len(mlist)}')
return selected_modules
def select_all():
'''Select all modules in the list of available modules and execute get_selected_modules'''
for artifact_name in mlist:
main_window.nametowidget(f'f_modules.f_list.tbox.mcb_{artifact_name}').select()
get_selected_modules()
def deselect_all():
'''Unselect all modules in the list of available modules and execute get_selected_modules'''
for artifact_name in mlist:
main_window.nametowidget(f'f_modules.f_list.tbox.mcb_{artifact_name}').deselect()
get_selected_modules()
def load_profile():
'''Select modules from a profile file'''
global profile_filename
destination_path = tk_filedialog.askopenfilename(parent=main_window,
title='Load a profile',
filetypes=(('iLEAPP Profile', '*.ilprofile'),))
if destination_path and os.path.exists(destination_path):
profile_load_error = None
with open(destination_path, 'rt', encoding='utf-8') as profile_in:
try:
profile = json.load(profile_in)
except:
profile_load_error = 'File was not a valid profile file: invalid format'
if not profile_load_error:
if isinstance(profile, dict):
if profile.get('leapp') != 'ileapp' or profile.get('format_version') != 1:
profile_load_error = 'File was not a valid profile file: incorrect LEAPP or version'
else:
deselect_all()
ticked = set(profile.get('plugins', []))
for artifact_name in mlist:
if artifact_name in ticked:
main_window.nametowidget(f'f_modules.f_list.tbox.mcb_{artifact_name}').select()
get_selected_modules()
else:
profile_load_error = 'File was not a valid profile file: invalid format'
if profile_load_error:
tk_msgbox.showerror(title='Error', message=profile_load_error, parent=main_window)
else:
profile_filename = destination_path
tk_msgbox.showinfo(
title='Profile loaded', message=f'Loaded profile: {destination_path}', parent=main_window)
def save_profile():
'''Save selected modules in a profile file'''
destination_path = tk_filedialog.asksaveasfilename(parent=main_window,
title='Save a profile',
filetypes=(('iLEAPP Profile', '*.ilprofile'),))
if destination_path:
selected_modules = get_selected_modules()
with open(destination_path, 'wt', encoding='utf-8') as profile_out:
json.dump({'leapp': 'ileapp', 'format_version': 1, 'plugins': selected_modules}, profile_out)
tk_msgbox.showinfo(
title='Save a profile', message=f'Profile saved: {destination_path}', parent=main_window)
def scroll(event):
'''Continue to scroll the list with mouse wheel when cursor hover a checkbutton'''
parent = main_window.nametowidget(event.widget.winfo_parent())
parent.event_generate('<MouseWheel>', delta=event.delta, when='now')
def ValidateInput():
'''Returns tuple (success, extraction_type)'''
i_path = input_entry.get() # input file/folder
o_path = output_entry.get() # output folder
ext_type = ''
# check input
if len(i_path) == 0:
tk_msgbox.showerror(title='Error', message='No INPUT file or folder selected!', parent=main_window)
return False, ext_type
elif not os.path.exists(i_path):
tk_msgbox.showerror(title='Error', message='INPUT file/folder does not exist!', parent=main_window)
return False, ext_type
elif os.path.isdir(i_path) and (os.path.exists(os.path.join(i_path, 'Manifest.db')) or os.path.exists(os.path.join(i_path, 'Manifest.mbdb'))):
ext_type = 'itunes'
elif os.path.isdir(i_path):
ext_type = 'fs'
else:
ext_type = Path(i_path).suffix[1:].lower()
# check output now
if len(o_path) == 0: # output folder
tk_msgbox.showerror(title='Error', message='No OUTPUT folder selected!', parent=main_window)
return False, ext_type
# check if at least one module is selected
if len(get_selected_modules()) == 0:
tk_msgbox.showerror(title='Error', message='No module selected for processing!', parent=main_window)
return False, ext_type
return True, ext_type
def open_report(report_path):
'''Open report and Quit after processing completed'''
webbrowser.open_new_tab('file://' + report_path)
main_window.quit()
def process(casedata):
'''Execute selected modules and create reports'''
# check if selections made properly; if not we will return to input form without exiting app altogether
is_valid, extracttype = ValidateInput()
if is_valid:
GuiWindow.window_handle = main_window
input_path = input_entry.get()
output_folder = output_entry.get()
# ios file system extractions contain paths > 260 char, which causes problems
# This fixes the problem by prefixing \\?\ on each windows path.
if is_platform_windows():
if input_path[1] == ':' and extracttype == 'fs': input_path = '\\\\?\\' + input_path.replace('/', '\\')
if output_folder[1] == ':': output_folder = '\\\\?\\' + output_folder.replace('/', '\\')
# re-create modules list based on user selection
selected_modules = get_selected_modules()
if extracttype != 'itunes':
selected_modules.insert(0, 'lastBuild') # Force lastBuild as first item to be parsed
selected_modules = [loader[module] for module in selected_modules]
progress_bar.config(maximum=len(selected_modules))
casedata = {key: value.get() for key, value in casedata.items()}
out_params = OutputParameters(output_folder)
wrap_text = True
time_offset = timezone_set.get()
if time_offset == '':
time_offset = 'UTC'
logtext_frame.grid(row=1, column=0, rowspan=3, padx=14, pady=4, sticky='nswe')
bottom_frame.grid_remove()
progress_bar.grid(padx=16, sticky='we')
initialize_lava(input_path, out_params.report_folder_base, extracttype)
crunch_successful = ileapp.crunch_artifacts(
selected_modules, extracttype, input_path, out_params, wrap_text, loader,
casedata, time_offset, profile_filename)
lava_finalize_output(out_params.report_folder_base)
if crunch_successful:
report_path = os.path.join(out_params.report_folder_base, 'index.html')
if report_path.startswith('\\\\?\\'): # windows
report_path = report_path[4:]
if report_path.startswith('\\\\'): # UNC path
report_path = report_path[2:]
progress_bar.grid_remove()
open_report_button = ttk.Button(main_window, text='Open Report & Close', command=lambda: open_report(report_path))
open_report_button.grid(ipadx=8)
else:
log_path = out_params.screen_output_file_path
if log_path.startswith('\\\\?\\'): # windows
log_path = log_path[4:]
tk_msgbox.showerror(
title='Error',
message=f'Processing failed :( \nSee log for error details..\nLog file located at {log_path}',
parent=main_window)
def select_input(button_type):
'''Select source and insert its path into input field'''
if button_type == 'file':
input_filename = tk_filedialog.askopenfilename(parent=main_window,
title='Select a file',
filetypes=(('All supported files', '*.tar *.zip *.gz'),
('tar file', '*.tar'), ('zip file', '*.zip'),
('gz file', '*.gz')))
else:
input_filename = tk_filedialog.askdirectory(parent=main_window, title='Select a folder')
input_entry.delete(0, 'end')
input_entry.insert(0, input_filename)
def select_output():
'''Select target and insert its path into output field'''
output_filename = tk_filedialog.askdirectory(parent=main_window, title='Select a folder')
output_entry.delete(0, 'end')
output_entry.insert(0, output_filename)
def case_data():
# GUI layout
## Case Data
'''Add Case Data window'''
global casedata
def clear():
'''Remove the contents of all fields'''
case_number_entry.delete(0, 'end')
case_agency_entry.delete(0, 'end')
case_examiner_entry.delete(0, 'end')
def save_case():
'''Save case data in a Case Data file'''
destination_path = tk_filedialog.asksaveasfilename(parent=case_window,
title='Save a case data file',
filetypes=(('LEAPP Case Data', '*.lcasedata'),))
if destination_path:
json_casedata = {key: value.get() for key, value in casedata.items()}
with open(destination_path, 'wt', encoding='utf-8') as case_data_out:
json.dump({'leapp': 'case_data', 'case_data_values': json_casedata}, case_data_out)
tk_msgbox.showinfo(
title='Save Case Data', message=f'Case Data saved: {destination_path}', parent=case_window)
def load_case():
'''Import case data from a Case Data file'''
destination_path = tk_filedialog.askopenfilename(parent=case_window,
title='Load case data',
filetypes=(('LEAPP Case Data', '*.lcasedata'),))
if destination_path and os.path.exists(destination_path):
case_data_load_error = None
with open(destination_path, 'rt', encoding='utf-8') as case_data_in:
try:
case_data = json.load(case_data_in)
except:
case_data_load_error = 'File was not a valid case data file: invalid format'
if not case_data_load_error:
if isinstance(case_data, dict):
if case_data.get('leapp') != 'case_data':
case_data_load_error = 'File was not a valid case data file'
else:
casedata = case_data.get('case_data_values', {})
case_number_entry.delete(0, 'end')
case_number_entry.insert(0, casedata.get('Case Number', ''))
case_agency_entry.delete(0, 'end')
case_agency_entry.insert(0, casedata.get('Agency', ''))
case_examiner_entry.delete(0, 'end')
case_examiner_entry.insert(0, casedata.get('Examiner', ''))
else:
case_data_load_error = 'File was not a valid case data file: invalid format'
if case_data_load_error:
tk_msgbox.showerror(title='Error', message=case_data_load_error, parent=case_window)
else:
tk_msgbox.showinfo(
title='Load Case Data', message=f'Loaded Case Data: {destination_path}', parent=case_window)
### Case Data Window creation
case_window = tk.Toplevel(main_window)
case_window_width = 560
if is_platform_linux():
case_window_height = 290
else:
case_window_height = 270
#### Places Case Data window in the center of the screen
screen_width = main_window.winfo_screenwidth()
screen_height = main_window.winfo_screenheight()
margin_width = (screen_width - case_window_width) // 2
margin_height = (screen_height - case_window_height) // 2
#### Case Data window properties
case_window.geometry(f'{case_window_width}x{case_window_height}+{margin_width}+{margin_height}')
case_window.resizable(False, False)
case_window.configure(bg=theme_bgcolor)
case_window.title('Add Case Data')
case_window.grid_columnconfigure(0, weight=1)
#### Layout
case_title_label = ttk.Label(case_window, text='Add Case Data', font=('Helvetica 18'))
case_title_label.grid(row=0, column=0, padx=14, pady=7, sticky='w')
case_number_frame = ttk.LabelFrame(case_window, text=' Case Number ')
case_number_frame.grid(row=1, column=0, padx=14, pady=5, sticky='we')
case_number_entry = ttk.Entry(case_number_frame, textvariable=casedata['Case Number'])
case_number_entry.pack(padx=5, pady=4, fill='x')
case_number_entry.focus()
case_agency_frame = ttk.LabelFrame(case_window, text=' Agency ')
case_agency_frame.grid(row=2, column=0, padx=14, pady=5, sticky='we')
case_agency_entry = ttk.Entry(case_agency_frame, textvariable=casedata['Agency'])
case_agency_entry.pack(padx=5, pady=4, fill='x')
case_examiner_frame = ttk.LabelFrame(case_window, text=' Examiner ')
case_examiner_frame.grid(row=3, column=0, padx=14, pady=5, sticky='we')
case_examiner_entry = ttk.Entry(case_examiner_frame, textvariable=casedata['Examiner'])
case_examiner_entry.pack(padx=5, pady=4, fill='x')
modules_btn_frame = ttk.Frame(case_window)
modules_btn_frame.grid(row=4, column=0, padx=14, pady=16, sticky='we')
modules_btn_frame.grid_columnconfigure(2, weight=1)
load_case_button = ttk.Button(modules_btn_frame, text='Load Case Data File', command=load_case)
load_case_button.grid(row=0, column=0, padx=5)
save_case_button = ttk.Button(modules_btn_frame, text='Save Case Data File', command=save_case)
save_case_button.grid(row=0, column=1, padx=5)
ttk.Separator(modules_btn_frame, orient='vertical').grid(row=0, column=2, padx=20, sticky='ns')
clear_case_button = ttk.Button(modules_btn_frame, text='Clear', command=clear)
clear_case_button.grid(row=0, column=3, padx=5)
close_case_button = ttk.Button(modules_btn_frame, text='Close', command=case_window.destroy)
close_case_button.grid(row=0, column=4, padx=5)
case_window.grab_set()
## Main window creation
main_window = tk.Tk()
window_width = 890
window_height = 620
## Variables
icon = os.path.join(os.path.dirname(__file__), 'scripts', 'icon.png')
loader: typing.Optional[plugin_loader.PluginLoader] = None
loader = plugin_loader.PluginLoader()
mlist = {}
profile_filename = None
casedata = {'Case Number': tk.StringVar(),
'Agency': tk.StringVar(),
'Examiner': tk.StringVar(),
}
timezone_set = tk.StringVar()
pickModules()
## Theme properties
theme_bgcolor = '#2c2825'
theme_inputcolor = '#705e52'
theme_fgcolor = '#fdcb52'
if is_platform_macos():
mlist_window_height = 24
log_text_height = 36
elif is_platform_linux():
mlist_window_height = 16
log_text_height = 27
else:
mlist_window_height = 19
log_text_height = 29
## Places main window in the center
screen_width = main_window.winfo_screenwidth()
screen_height = main_window.winfo_screenheight()
margin_width = (screen_width - window_width) // 2
margin_height = (screen_height - window_height) // 2
## Main window properties
main_window.geometry(f'{window_width}x{window_height}+{margin_width}+{margin_height}')
main_window.title(f'iLEAPP version {ileapp_version}')
main_window.resizable(False, False)
main_window.configure(bg=theme_bgcolor)
logo_icon = tk.PhotoImage(file=icon)
main_window.iconphoto(True, logo_icon)
main_window.grid_columnconfigure(0, weight=1)
## Widgets default style
style = ttk.Style()
style.theme_use('default')
style.configure('.',
background=theme_bgcolor,
foreground=theme_fgcolor)
style.configure('TButton')
style.map('TButton',
background=[('active', 'black'), ('!disabled', theme_fgcolor)],
foreground=[('active', theme_fgcolor), ('!disabled', 'black')])
style.configure('TEntry', fieldbackground=theme_inputcolor, highlightthickness=0)
style.configure(
'TCombobox', selectforeground=theme_fgcolor,
selectbackground=theme_inputcolor, arrowcolor=theme_fgcolor)
style.map('TCombobox',
fieldbackground=[('active', theme_inputcolor), ('readonly', theme_inputcolor)],
)
style.configure('TScrollbar', background=theme_fgcolor, arrowcolor='black', troughcolor=theme_inputcolor)
style.configure('TProgressbar', thickness=4, background='DarkGreen')
## Main Window Layout
### Top part of the window
title_frame = ttk.Frame(main_window)
title_frame.grid(padx=14, pady=6, sticky='w')
title_label = ttk.Label(
title_frame,
text='iOS Logs, Events, And Plists Parser',
font=('Helvetica 22'))
title_label.pack(pady=4)
github_label = ttk.Label(
title_frame,
text='https://github.com/abrignoni/iLEAPP',
font=('Helvetica 14'))
github_label.pack(anchor='w')
### Input output selection
input_frame = ttk.LabelFrame(
main_window,
text=' Select the file (tar/zip/gz) or directory of the target iOS full file system extraction for parsing: ')
input_frame.grid(padx=14, pady=2, sticky='we')
input_frame.grid_columnconfigure(0, weight=1)
input_entry = ttk.Entry(input_frame)
input_entry.grid(row=0, column=0, padx=5, pady=4, sticky='we')
input_file_button = ttk.Button(input_frame, text='Browse File', command=lambda: select_input('file'))
input_file_button.grid(row=0, column=1, padx=5, pady=4)
input_folder_button = ttk.Button(input_frame, text='Browse Folder', command=lambda: select_input('folder'))
input_folder_button.grid(row=0, column=2, padx=5, pady=4)
output_frame = ttk.LabelFrame(main_window, text=' Select Output Folder: ')
output_frame.grid(padx=14, pady=5, sticky='we')
output_frame.grid_columnconfigure(0, weight=1)
output_entry = ttk.Entry(output_frame)
output_entry.grid(row=0, column=0, padx=5, pady=4, sticky='we')
output_folder_button = ttk.Button(output_frame, text='Browse Folder', command=select_output)
output_folder_button.grid(row=0, column=1, padx=5, pady=4)
### Modules
modules_frame = ttk.Frame(main_window, name='f_modules')
modules_frame.grid(padx=14, pady=4, sticky='we')
modules_frame.grid_columnconfigure(0, weight=1)
#### Buttons & Timezone
button_frame = ttk.Frame(modules_frame)
button_frame.grid(row=0, column=0, pady=4, sticky='we')
all_button = ttk.Button(button_frame, text='Select All', command=select_all)
all_button.grid(row=0, column=0, padx=5)
none_button = ttk.Button(button_frame, text='Deselect All', command=deselect_all)
none_button.grid(row=0, column=1, padx=5)
load_button = ttk.Button(button_frame, text='Load Profile', command=load_profile)
load_button.grid(row=0, column=2, padx=5)
save_button = ttk.Button(button_frame, text='Save Profile', command=save_profile)
save_button.grid(row=0, column=3, padx=5)
ttk.Separator(button_frame, orient='vertical').grid(row=0, column=4, padx=10, sticky='ns')
case_data_button = ttk.Button(button_frame, text='Case Data', command=case_data)
case_data_button.grid(row=0, column=5, padx=5)
ttk.Separator(button_frame, orient='vertical').grid(row=0, column=6, padx=10, sticky='ns')
ttk.Label(
button_frame, text='Timezone Offset: '
).grid(row=0, column=7)
timezone_offset = ttk.Combobox(
button_frame, textvariable=timezone_set, values=tzvalues, height=20, state='readonly')
timezone_offset.master.option_add('*TCombobox*Listbox.background', theme_inputcolor)
timezone_offset.master.option_add('*TCombobox*Listbox.foreground', theme_fgcolor)
timezone_offset.master.option_add('*TCombobox*Listbox.selectBackground', theme_fgcolor)
timezone_offset.grid(row=0, column=8)
#### List of modules
mlist_frame = ttk.LabelFrame(modules_frame, text=' Available Modules: ', name='f_list')
mlist_frame.grid(row=1, column=0, padx=4, pady=4, sticky='we')
mlist_frame.grid_columnconfigure(0, weight=1)
v = ttk.Scrollbar(mlist_frame, orient='vertical')
v.grid(row=0, column=1, sticky='ns')
mlist_text = tk.Text(mlist_frame, name='tbox', bg=theme_bgcolor, highlightthickness=0,
yscrollcommand=v.set, height=mlist_window_height)
mlist_text.grid(row=0, column=0, sticky='we')
v.config(command=mlist_text.yview)
for artifact_name, module_infos in mlist.items():
cb = tk.Checkbutton(mlist_text, name=f'mcb_{artifact_name}',
text=f'{module_infos[0]} [{module_infos[1]} | {module_infos[2]}.py]',
variable=module_infos[-1], onvalue=True, offvalue=False, command=get_selected_modules)
cb.config(background=theme_bgcolor, fg=theme_fgcolor, selectcolor=theme_inputcolor,
highlightthickness=0, activebackground=theme_bgcolor, activeforeground=theme_fgcolor)
mlist_text.window_create('insert', window=cb)
mlist_text.insert('end', '\n')
mlist_text.config(state='disabled')
main_window.bind_class('Checkbutton', '<MouseWheel>', scroll)
main_window.bind_class('Checkbutton', '<Button-4>', scroll)
main_window.bind_class('Checkbutton', '<Button-5>', scroll)
### Process
bottom_frame = ttk.Frame(main_window)
bottom_frame.grid(padx=16, pady=6, sticky='we')
bottom_frame.grid_columnconfigure(2, weight=1)
process_button = ttk.Button(bottom_frame, text='Process', command=lambda: process(casedata))
process_button.grid(row=0, column=0, rowspan=2, padx=5)
close_button = ttk.Button(bottom_frame, text='Close', command=main_window.quit)
close_button.grid(row=0, column=1, rowspan=2, padx=5)
selected_modules_label = ttk.Label(bottom_frame, text='Number of selected modules: ')
selected_modules_label.grid(row=0, column=2, padx=5, sticky='e')
auto_unselected_modules_text = '(Modules making some time to run were automatically unselected)'
if is_platform_macos():
auto_unselected_modules_label = ttk.Label(
bottom_frame,
text=auto_unselected_modules_text,
font=('Helvetica 10'))
else:
auto_unselected_modules_label = ttk.Label(bottom_frame, text=auto_unselected_modules_text)
auto_unselected_modules_label.grid(row=1, column=2, padx=5, sticky='e')
get_selected_modules()
#### Logs
logtext_frame = ttk.Frame(main_window, name='logs_frame')
logtext_frame.grid_columnconfigure(0, weight=1)
vlog = ttk.Scrollbar(logtext_frame, orient='vertical')
vlog.grid(row=0, column=1, pady=10, sticky='ns')
log_text = tk.Text(
logtext_frame, name='log_text', bg=theme_inputcolor, fg=theme_fgcolor,
highlightthickness=1, yscrollcommand=vlog.set, height=log_text_height)
log_text.grid(row=0, column=0, padx=4, pady=10, sticky='we')
vlog.config(command=log_text.yview)
### Progress bar
progress_bar = ttk.Progressbar(main_window, orient='horizontal')
### Push main window on top
def OnFocusIn(event):
if type(event.widget).__name__ == 'Tk':
event.widget.attributes('-topmost', False)
main_window.attributes('-topmost', True)
main_window.focus_force()
main_window.bind('<FocusIn>', OnFocusIn)
main_window.mainloop()