Files
MultiPar/alpha/tool/queue_create.py
2023-03-22 10:23:00 +09:00

880 lines
33 KiB
Python

import sys
import os
import subprocess
import stat
import tkinter as tk
from tkinter import ttk
from tkinter import filedialog
# Set path of MultiPar
client_path = "../par2j64.exe"
gui_path = "../MultiPar.exe"
# Set options for par2j
# Because /fe option is set to exclude .PAR2 files by default, no need to set here.
# Don't use /ss, /sn, /sr, or /sm here.
cmd_option = "/rr10 /rd2"
# How to set slices initially (either /ss, /sn, or /sr)
init_slice_option = "/sr10"
# Optimization settings. Refer "each_folder.py" for detail.
slice_size_multiplier = 4096
max_slice_count = 20000
max_slice_rate = 170
min_slice_count = 100
min_slice_rate = 30
min_efficiency_improvement = 0.3
# Initialize global variables
current_dir = "./"
sub_proc = None
# Read "Efficiency rate"
def read_efficiency(output_text):
# Find from the last
line_start = output_text.rfind("Efficiency rate\t\t:")
if line_start != -1:
line_start += 19
line_end = output_text.find("%\n", line_start)
return float(output_text[line_start:line_end])
else:
return -1
# Read "Input File Slice count"
def read_slice_count(output_text):
# Find from the top
line_start = output_text.find("Input File Slice count\t:")
if line_start != -1:
line_start += 25
line_end = output_text.find("\n", line_start)
return int(output_text[line_start:line_end])
else:
return -1
# Read "Input File Slice size"
def read_slice_size(output_text):
# Find from the top
line_start = output_text.find("Input File Slice size\t:")
if line_start != -1:
line_start += 24
line_end = output_text.find("\n", line_start)
return int(output_text[line_start:line_end])
else:
return -1
# Search setting of good efficiency
def test_efficiency(par_path, source_path):
min_size = 0
max_size = 0
best_count = 0
best_size = 0
best_efficiency = 0
best_efficiency_at_initial_count = 0
best_count_at_max_count = 0
best_size_at_max_count = 0
best_efficiency_at_max_count = 0
# First time to get initial value
cmd = "\"" + client_path + "\" t /uo /fe\"**.par2\" " + init_slice_option + " /sm" + str(slice_size_multiplier) + " " + cmd_option + " \"" + par_path + "\" \"" + source_path + "\""
res = subprocess.run(cmd, shell=True, capture_output=True, encoding='utf8')
if res.returncode != 0:
return 0
efficiency_rate = read_efficiency(res.stdout)
if efficiency_rate < 0:
return 0
initial_count = read_slice_count(res.stdout)
if initial_count <= 0:
return 0
initial_size = read_slice_size(res.stdout)
best_efficiency_at_initial_count = efficiency_rate
#print("initial_size =", initial_size, ", initial_count =", initial_count, ", efficiency =", efficiency_rate)
# Get min and max of slice count and size to be used at searching
# maximum slice count is co-related to minimum slice size
if max_slice_rate != 0:
if initial_count > max_slice_count:
if (initial_count * max_slice_rate / 100) > 32768:
max_count = 32768
else:
max_count = int(initial_count * max_slice_rate / 100)
else:
if (initial_count * max_slice_rate / 100) > max_slice_count:
max_count = max_slice_count
else:
max_count = int(initial_count * max_slice_rate / 100)
else:
max_count = 32768
cmd = "\"" + client_path + "\" t /uo /fe\"**.par2\" /sn" + str(max_count) + " /sm" + str(slice_size_multiplier) + " " + cmd_option + " \"" + par_path + "\" \"" + source_path + "\""
res = subprocess.run(cmd, shell=True, capture_output=True, encoding='utf8')
if res.returncode != 0:
return 0
efficiency_rate = read_efficiency(res.stdout)
if efficiency_rate < 0:
return 0
best_count_at_max_count = read_slice_count(res.stdout)
best_size_at_max_count = read_slice_size(res.stdout)
best_efficiency_at_max_count = efficiency_rate
max_count = read_slice_count(res.stdout)
min_size = read_slice_size(res.stdout)
#print("max_count =", max_count, ", min_size =", min_size, ", efficiency =", best_efficiency_at_max_count)
# Minimum slice count is co-related to maximum slice size
if min_slice_rate > 0 and (initial_count * min_slice_rate / 100) > min_slice_count:
min_count = int(initial_count * min_slice_rate / 100)
else:
min_count = min_slice_count
cmd = "\"" + client_path + "\" t /uo /fe\"**.par2\" /sn" + str(min_count) + " /sm" + str(slice_size_multiplier) + " " + cmd_option + " \"" + par_path + "\" \"" + source_path + "\""
res = subprocess.run(cmd, shell=True, capture_output=True, encoding='utf8')
if res.returncode != 0:
return 0
efficiency_rate = read_efficiency(res.stdout)
if efficiency_rate < 0:
return 0
min_count = read_slice_count(res.stdout)
max_size = read_slice_size(res.stdout)
best_count = read_slice_count(res.stdout)
best_size = read_slice_size(res.stdout)
best_efficiency = efficiency_rate
#print("min_count =", min_count, ", max_size =", max_size, ", efficiency =", best_efficiency)
# If the calculated maximum slice count is too small, no need to search (QUITE UNLIKELY to happen)
if max_slice_rate > 0 and (initial_count * max_slice_rate / 100) <= min_slice_count:
cmd = "\"" + client_path + "\" t /uo /fe\"**.par2\" /sn" + str(min_slice_count) + " /sm" + str(slice_size_multiplier) + " " + cmd_option + " \"" + par_path + "\" \"" + source_path + "\""
res = subprocess.run(cmd, shell=True, capture_output=True, encoding='utf8')
if res.returncode != 0:
return 0
efficiency_rate = read_efficiency(res.stdout)
if efficiency_rate < 0:
return 0
best_count = read_slice_count(res.stdout)
best_size = read_slice_size(res.stdout)
best_efficiency = efficiency_rate
#print("initial_count too small, best_count =", best_count, ", best_size =", best_size, ", best_efficiency =", best_efficiency)
# Return slice size to archive the best efficiency
return best_size
else:
# Try every (step) slice count between min_count and max_count
step_slice_count_int = int((min_count + 1) * 8 / 7)
while step_slice_count_int < max_count:
#print(f"Testing slice count: (around) {step_slice_count_int}, from {(step_slice_count_int - int(step_slice_count_int / 8))} to {int(step_slice_count_int * 17 / 16)}")
cmd = "\"" + client_path + "\" t /uo /fe\"**.par2\" /sn" + str(step_slice_count_int) + " /sm" + str(slice_size_multiplier) + " " + cmd_option + " \"" + par_path + "\" \"" + source_path + "\""
res = subprocess.run(cmd, shell=True, capture_output=True, encoding='utf8')
if res.returncode != 0:
break
efficiency_rate = read_efficiency(res.stdout)
if efficiency_rate < 0:
break
if efficiency_rate > best_efficiency + min_efficiency_improvement:
best_count = read_slice_count(res.stdout)
best_size = read_slice_size(res.stdout)
best_efficiency = efficiency_rate
# Next count should be more than 17/16 of the input count. (Range to +6.25% was checked already.)
step_slice_count_int = int((int(step_slice_count_int * 17 / 16) + 1) * 8 / 7)
# Evaluate slice count searched with initial_count
if initial_count < best_count and best_efficiency_at_initial_count > best_efficiency - min_efficiency_improvement:
best_count = initial_count
best_size = initial_size
best_efficiency = best_efficiency_at_initial_count
# Evaluate slice count searched with max_count.
if best_efficiency_at_max_count > best_efficiency + min_efficiency_improvement:
best_count = best_count_at_max_count
best_size = best_size_at_max_count
best_efficiency = best_efficiency_at_max_count
#print("best_count =", best_count, "best_size =", best_size, ", best_efficiency =", best_efficiency)
return best_size
# Return zero for empty folder
def check_empty(path='.'):
total = 0
with os.scandir(path) as it:
for entry in it:
if entry.is_file():
# Ignore hidden file
if entry.stat().st_file_attributes & stat.FILE_ATTRIBUTE_HIDDEN:
continue
# Ignore PAR file
entry_ext = os.path.splitext(entry.name)[1]
if entry_ext.lower() == ".par2":
continue
total += entry.stat().st_size
elif entry.is_dir():
total += check_empty(entry.path)
if total > 0:
break
return total
# Search children folders or files in a parent folder
def search_child_item(parent_path):
parent_name = os.path.basename(parent_path)
# Check the folder exists already
item_count = listbox_list1.size()
item_path = parent_path + "\\"
item_index = 0
while item_index < item_count:
index_path = listbox_list1.get(item_index)
if os.path.samefile(item_path, index_path):
label_status.config(text= "The folder \"" + parent_name + "\" is selected already.")
return
common_path = os.path.commonpath([item_path, index_path]) + "\\"
if os.path.samefile(common_path, item_path):
label_status.config(text= "The folder \"" + parent_name + "\" is parent of another selected item.")
return
if os.path.samefile(common_path, index_path):
label_status.config(text= "The folder \"" + parent_name + "\" is child of another selected item.")
return
item_index += 1
# Add found items
error_text = ""
add_count = 0
for item_name in os.listdir(parent_path):
# Ignore PAR files (extension ".par2")
item_ext = os.path.splitext(item_name)[1]
if item_ext.lower() == ".par2":
error_text += " PAR file \"" + item_name + "\" is ignored."
continue
# Ignore hidden item
item_path = os.path.join(parent_path, item_name)
if os.stat(item_path).st_file_attributes & stat.FILE_ATTRIBUTE_HIDDEN:
error_text += " Hidden \"" + item_name + "\" is ignored."
continue
# Distinguish folder or file
if os.path.isdir(item_path):
# Ignore empty folder
if check_empty(item_path) == 0:
continue
item_path += "\\"
listbox_list1.insert(tk.END, item_path)
add_count += 1
item_count = listbox_list1.size()
label_head1.config(text= str(item_count) + " items")
if item_count == 0:
label_status.config(text= "There are no items." + error_text)
button_start.config(state=tk.DISABLED)
elif add_count == 0:
label_status.config(text= "No items were found in folder \"" + parent_name + "\"." + error_text)
else:
label_status.config(text= str(add_count) + " items were found in folder \"" + parent_name + "\"." + error_text)
button_reset.config(state=tk.NORMAL)
button_start.config(state=tk.NORMAL)
button_remove.config(state=tk.NORMAL)
# Select a folder to search children folders or files
def button_parent_clicked():
global current_dir
if os.path.exists(current_dir) == False:
current_dir = "./"
search_dir = filedialog.askdirectory(initialdir=current_dir)
if search_dir == "":
return
current_dir = search_dir
search_child_item(search_dir)
# Reset lists and display status
def button_reset_clicked():
global current_dir
current_dir = "./"
# Clear list-box at first
listbox_list1.delete(0, tk.END)
listbox_list2.delete(0, tk.END)
# Reset statues text
label_head1.config(text= '0 items')
label_head2.config(text= '0 finished items')
label_status.config(text= 'Select folders and/or files to create PAR files.')
# Reset button state
button_parent.config(state=tk.NORMAL)
button_child.config(state=tk.NORMAL)
button_file.config(state=tk.NORMAL)
button_reset.config(state=tk.DISABLED)
button_start.config(state=tk.DISABLED)
button_stop.config(state=tk.DISABLED)
button_remove.config(state=tk.DISABLED)
button_open2.config(state=tk.DISABLED)
# Check and add items
def add_argv_item():
# Add specified items
error_text = ""
for one_path in sys.argv[1:]:
# Make sure to be absolute path
item_path = os.path.abspath(one_path)
if os.path.exists(item_path) == False:
error_text += " \"" + item_path + "\" doesn't exist."
continue
#print(item_path)
# Ignore PAR files (extension ".par2")
item_name = os.path.basename(item_path)
item_ext = os.path.splitext(item_name)[1]
if item_ext.lower() == ".par2":
error_text += " PAR file \"" + item_name + "\" is ignored."
continue
# Ignore hidden item
if os.stat(item_path).st_file_attributes & stat.FILE_ATTRIBUTE_HIDDEN:
error_text += " Hidden \"" + item_name + "\" is ignored."
continue
# Distinguish folder or file
if os.path.isdir(item_path):
# Ignore empty folder
if check_empty(item_path) == 0:
continue
item_path += "\\"
# Check the item exists already or duplicates
item_count = listbox_list1.size()
item_index = 0
while item_index < item_count:
index_path = listbox_list1.get(item_index)
if os.path.samefile(item_path, index_path):
error_text += " \"" + item_name + "\" is selected already."
item_count = -1
break
common_path = os.path.commonpath([item_path, index_path]) + "\\"
if os.path.samefile(common_path, item_path):
error_text += " \"" + item_name + "\" is parent of another selected item."
item_count = -1
break
if os.path.samefile(common_path, index_path):
error_text += " \"" + item_name + "\" is child of another selected item."
item_count = -1
break
item_index += 1
if item_count < 0:
continue
listbox_list1.insert(tk.END, item_path)
item_count = listbox_list1.size()
label_head1.config(text= str(item_count) + " items")
if item_count == 0:
label_status.config(text= "There are no items." + error_text)
else:
label_status.config(text= str(item_count) + " items were selected at first." + error_text)
button_reset.config(state=tk.NORMAL)
button_start.config(state=tk.NORMAL)
button_remove.config(state=tk.NORMAL)
# Cretae the first PAR set
def queue_run():
global sub_proc
if sub_proc != None:
return
if "disabled" in button_stop.state():
button_parent.config(state=tk.NORMAL)
button_child.config(state=tk.NORMAL)
button_file.config(state=tk.NORMAL)
button_start.config(state=tk.NORMAL)
button_remove.config(state=tk.NORMAL)
button_open2.config(state=tk.NORMAL)
label_status.config(text= "Stopped queue")
return
item_path = listbox_list1.get(0)
if item_path == "":
label_status.config(text= "There are no items.")
return
# When it's a folder, create PAR2 files for inner files.
if item_path[-1:] == "\\":
base_name = os.path.basename(item_path[:-1]) + ".par2"
source_path = item_path + "*"
par_path = item_path + base_name
# When it's a file, create PAR2 files for the file.
else:
base_name = os.path.basename(item_path) + ".par2"
source_path = item_path
par_path = item_path + ".par2"
label_status.config(text= "Creating \"" + base_name + "\"")
# Test setting for good efficiency
slice_size = test_efficiency(par_path, source_path)
if slice_size == 0:
label_status.config(text= "Failed to test options.")
return
# Set command-line
# Cover path by " for possible space
cmd = "\"" + client_path + "\" c /fe\"**.par2\" /ss" + str(slice_size) + " " + cmd_option + " \"" + par_path + "\" \"" + source_path + "\""
# If you want to see creating result only, use "t" command instead of "c".
#print(cmd)
# Run PAR2 client
sub_proc = subprocess.Popen(cmd, shell=True)
# Wait finish of creation
root.after(300, queue_result)
# Wait and read created result
def queue_result():
global sub_proc
# When sub-process was not started yet
if sub_proc == None:
return
# When sub-process is running still
exit_code = sub_proc.poll()
if exit_code == None:
# Call self again
root.after(300, queue_result)
return
sub_proc = None
item_path = listbox_list1.get(0)
# When fatal error happened in par2j
if exit_code == 1:
button_parent.config(state=tk.NORMAL)
button_child.config(state=tk.NORMAL)
button_file.config(state=tk.NORMAL)
button_reset.config(state=tk.NORMAL)
button_stop.config(state=tk.DISABLED)
button_remove.config(state=tk.NORMAL)
button_open2.config(state=tk.NORMAL)
label_status.config(text= "Failed queue")
return
# When you cancel par2j on Command Prompt
elif exit_code == 2:
button_parent.config(state=tk.NORMAL)
button_child.config(state=tk.NORMAL)
button_file.config(state=tk.NORMAL)
button_reset.config(state=tk.NORMAL)
button_start.config(state=tk.NORMAL)
button_stop.config(state=tk.DISABLED)
button_remove.config(state=tk.NORMAL)
button_open2.config(state=tk.NORMAL)
label_status.config(text= "Canceled queue")
return
# When par files were created successfully
else:
#print("exit code =", exit_code)
# Add to list of finished items
listbox_list2.insert(tk.END, item_path)
item_count = listbox_list2.size()
label_head2.config(text= str(item_count) + " finished items")
# Remove the first item from the list
listbox_list1.delete(0)
# Process next set
item_count = listbox_list1.size()
if item_count == 0:
button_parent.config(state=tk.NORMAL)
button_child.config(state=tk.NORMAL)
button_file.config(state=tk.NORMAL)
button_reset.config(state=tk.NORMAL)
button_stop.config(state=tk.DISABLED)
button_open2.config(state=tk.NORMAL)
label_status.config(text= "Created all items")
elif "disabled" in button_stop.state():
button_parent.config(state=tk.NORMAL)
button_child.config(state=tk.NORMAL)
button_file.config(state=tk.NORMAL)
button_reset.config(state=tk.NORMAL)
button_start.config(state=tk.NORMAL)
button_remove.config(state=tk.NORMAL)
button_open2.config(state=tk.NORMAL)
label_status.config(text= "Interrupted queue")
else:
root.after(100, queue_run)
# Select a child folder to add
def button_child_clicked():
global current_dir
if os.path.exists(current_dir) == False:
current_dir = "./"
one_path = filedialog.askdirectory(initialdir=current_dir)
if one_path == "":
return
current_dir = os.path.dirname(one_path)
# Check the folder has content
one_name = os.path.basename(one_path)
if check_empty(one_path) == 0:
label_status.config(text= "Selected folder \"" + one_name + "\" is empty.")
return
# Check the folder is new
one_path += "\\"
item_count = listbox_list1.size()
item_index = 0
while item_index < item_count:
index_path = listbox_list1.get(item_index)
if os.path.samefile(one_path, index_path):
label_status.config(text= "Folder \"" + one_name + "\" is selected already.")
return
common_path = os.path.commonpath([one_path, index_path]) + "\\"
if os.path.samefile(common_path, one_path):
label_status.config(text= "Folder \"" + one_name + "\" is parent of another selected item.")
return
if os.path.samefile(common_path, index_path):
label_status.config(text= "Folder \"" + one_name + "\" is child of another selected item.")
return
item_index += 1
listbox_list1.insert(tk.END, one_path)
item_count += 1
label_head1.config(text= str(item_count) + " items")
label_status.config(text= "Folder \"" + one_name + "\" was added.")
button_reset.config(state=tk.NORMAL)
button_start.config(state=tk.NORMAL)
button_remove.config(state=tk.NORMAL)
# Select multiple children files to add
def button_file_clicked():
global current_dir
if os.path.exists(current_dir) == False:
current_dir = "./"
multi_path = filedialog.askopenfilenames(initialdir=current_dir)
if len(multi_path) == 0:
return
one_path = multi_path[0]
current_dir = os.path.dirname(one_path)
# Add selected items
error_text = ""
add_count = 0
for one_path in multi_path:
# Ignore PAR file (extension ".par2")
one_name = os.path.basename(one_path)
item_ext = os.path.splitext(one_name)[1]
if item_ext.lower() == ".par2":
error_text += " PAR file \"" + one_name + "\" is ignored."
continue
# Ignore hidden file
if os.stat(one_path).st_file_attributes & stat.FILE_ATTRIBUTE_HIDDEN:
error_text += " Hidden \"" + one_name + "\" is ignored."
continue
# Check the file is new
item_count = listbox_list1.size()
item_index = 0
while item_index < item_count:
index_path = listbox_list1.get(item_index)
if os.path.samefile(one_path, index_path):
error_text += " \"" + one_name + "\" is selected already."
item_count = -1
break
common_path = os.path.commonpath([one_path, index_path]) + "\\"
if os.path.samefile(common_path, index_path):
error_text += " \"" + one_name + "\" is child of another selected item."
item_count = -1
break
item_index += 1
if item_count < 0:
continue
add_name = one_name
listbox_list1.insert(tk.END, one_path)
add_count += 1
item_count = listbox_list1.size()
label_head1.config(text= str(item_count) + " items")
if item_count == 0:
label_status.config(text= "There are no items." + error_text)
button_start.config(state=tk.DISABLED)
elif add_count == 0:
label_status.config(text= "No files were added." + error_text)
else:
if add_count == 1:
label_status.config(text= "File \"" + add_name + "\" was added." + error_text)
else:
label_status.config(text= str(add_count) + " files were added." + error_text)
button_reset.config(state=tk.NORMAL)
button_start.config(state=tk.NORMAL)
button_remove.config(state=tk.NORMAL)
# Select a child file to add
def button_file1_clicked():
global current_dir
if os.path.exists(current_dir) == False:
current_dir = "./"
one_path = filedialog.askopenfilename(initialdir=current_dir)
if one_path == "":
return
current_dir = os.path.dirname(one_path)
# Ignore PAR file (extension ".par2")
one_name = os.path.basename(one_path)
item_ext = os.path.splitext(one_name)[1]
# Compare filename in case insensitive
item_ext = item_ext.lower()
if item_ext == ".par2":
label_status.config(text= "PAR file \"" + one_name + "\" is ignored.")
return
# Ignore hidden file
if os.stat(one_path).st_file_attributes & stat.FILE_ATTRIBUTE_HIDDEN:
label_status.config(text= "Hidden \"" + one_name + "\" is ignored.")
return
# Check the file is new
item_count = listbox_list1.size()
item_index = 0
while item_index < item_count:
index_path = listbox_list1.get(item_index)
if os.path.samefile(one_path, index_path):
label_status.config(text= "File \"" + one_name + "\" is selected already.")
return
common_path = os.path.commonpath([one_path, index_path]) + "\\"
if os.path.samefile(common_path, index_path):
label_status.config(text= "File \"" + one_name + "\" is child of another selected item.")
return
item_index += 1
listbox_list1.insert(tk.END, one_path)
item_count += 1
label_head1.config(text= str(item_count) + " items")
label_status.config(text= "File \"" + one_name + "\" was added.")
button_reset.config(state=tk.NORMAL)
button_start.config(state=tk.NORMAL)
button_remove.config(state=tk.NORMAL)
# Resume stopped queue
def button_start_clicked():
global sub_proc
item_count = listbox_list1.size()
if item_count == 0:
label_status.config(text= "There are no items.")
return
button_parent.config(state=tk.DISABLED)
button_child.config(state=tk.DISABLED)
button_file.config(state=tk.DISABLED)
button_reset.config(state=tk.DISABLED)
button_start.config(state=tk.DISABLED)
button_stop.config(state=tk.NORMAL)
button_remove.config(state=tk.DISABLED)
button_open2.config(state=tk.DISABLED)
if sub_proc == None:
queue_run()
else:
queue_result()
# Stop running queue
def button_stop_clicked():
button_stop.config(state=tk.DISABLED)
if sub_proc != None:
label_status.config(text= "Waiting finish of current task")
# Remove items from list
def button_remove_clicked():
# It's possible to select multiple items.
selected_indices = listbox_list1.curselection()
selected_count = len(selected_indices)
if selected_count == 0:
label_status.config(text= "Select items to remove at first.")
return
label_status.config(text= "Removed " + str(selected_count) + " items.")
while selected_count > 0:
selected_count -= 1
selected_index = selected_indices[selected_count]
# Remove selected items at once
listbox_list1.delete(selected_index)
item_count = listbox_list1.size()
label_head1.config(text= str(item_count) + " items")
if item_count == 0:
button_start.config(state=tk.DISABLED)
button_remove.config(state=tk.DISABLED)
# Open a PAR set by MultiPar
def button_open2_clicked():
if os.path.exists(gui_path) == False:
label_status.config(text= "Cannot call \"" + gui_path + "\". Set path correctly.")
return
indices = listbox_list2.curselection()
if len(indices) == 1:
item_path = listbox_list2.get(indices[0])
if item_path[-1:] == "\\":
base_name = os.path.basename(item_path[:-1]) + ".par2"
par_path = item_path + base_name
else:
base_name = os.path.basename(item_path) + ".par2"
par_path = item_path + ".par2"
label_status.config(text= "Opening \"" + base_name + "\"")
# Set command-line
# Cover path by " for possible space
cmd = "\"" + gui_path + "\" /verify \"" + par_path + "\""
# Open MultiPar GUI to see details
# Because this doesn't wait finish of MultiPar, you may open some at once.
subprocess.Popen(cmd)
else:
label_status.config(text= "Select one item to open at first.")
# Window size and title
root = tk.Tk()
root.title('PAR Queue - Create')
root.minsize(width=480, height=200)
# Centering window
init_width = 640
init_height = 480
init_left = (root.winfo_screenwidth() - init_width) // 2
init_top = (root.winfo_screenheight() - init_height) // 2
root.geometry('{}x{}+{}+{}'.format(init_width, init_height, init_left, init_top))
#root.geometry("640x480")
root.columnconfigure(0, weight=1)
root.rowconfigure(1, weight=1)
# Control panel
frame_top = ttk.Frame(root, padding=(3,4,3,2))
frame_top.grid(row=0, column=0, sticky=(tk.E,tk.W))
button_parent = ttk.Button(frame_top, text="Search inner folder", width=18, command=button_parent_clicked)
button_parent.pack(side=tk.LEFT, padx=2)
button_child = ttk.Button(frame_top, text="Add single folder", width=16, command=button_child_clicked)
button_child.pack(side=tk.LEFT, padx=2)
button_file = ttk.Button(frame_top, text="Add multi files", width=14, command=button_file_clicked)
button_file.pack(side=tk.LEFT, padx=2)
button_reset = ttk.Button(frame_top, text="Reset lists", width=11, command=button_reset_clicked, state=tk.DISABLED)
button_reset.pack(side=tk.LEFT, padx=2)
# List
frame_middle = ttk.Frame(root, padding=(2,2,2,2))
frame_middle.grid(row=1, column=0, sticky=(tk.E,tk.W,tk.S,tk.N))
frame_middle.rowconfigure(0, weight=1)
frame_middle.columnconfigure(0, weight=1)
frame_middle.columnconfigure(1, weight=1)
# List of children items (folders and files)
frame_list1 = ttk.Frame(frame_middle, padding=(6,2,6,6), relief='groove')
frame_list1.grid(row=0, column=0, padx=4, sticky=(tk.E,tk.W,tk.S,tk.N))
frame_list1.columnconfigure(0, weight=1)
frame_list1.rowconfigure(2, weight=1)
frame_top1 = ttk.Frame(frame_list1, padding=(0,4,0,3))
frame_top1.grid(row=0, column=0, columnspan=2, sticky=(tk.E,tk.W))
button_start = ttk.Button(frame_top1, text="Start", width=6, command=button_start_clicked, state=tk.DISABLED)
button_start.pack(side=tk.LEFT, padx=2)
button_stop = ttk.Button(frame_top1, text="Stop", width=6, command=button_stop_clicked, state=tk.DISABLED)
button_stop.pack(side=tk.LEFT, padx=2)
button_remove = ttk.Button(frame_top1, text="Remove", width=8, command=button_remove_clicked, state=tk.DISABLED)
button_remove.pack(side=tk.LEFT, padx=2)
label_head1 = ttk.Label(frame_list1, text='0 items')
label_head1.grid(row=1, column=0, columnspan=2)
s_list1 = tk.StringVar()
listbox_list1 = tk.Listbox(frame_list1, listvariable=s_list1, activestyle='none', selectmode='extended')
listbox_list1.grid(row=2, column=0, sticky=(tk.E,tk.W,tk.S,tk.N))
scrollbar_list1 = ttk.Scrollbar(frame_list1, orient=tk.VERTICAL, command=listbox_list1.yview)
scrollbar_list1.grid(row=2, column=1, sticky=(tk.N, tk.S))
listbox_list1["yscrollcommand"] = scrollbar_list1.set
xscrollbar_list1 = ttk.Scrollbar(frame_list1, orient=tk.HORIZONTAL, command=listbox_list1.xview)
xscrollbar_list1.grid(row=3, column=0, sticky=(tk.E, tk.W))
listbox_list1["xscrollcommand"] = xscrollbar_list1.set
# List of finished items
frame_list2 = ttk.Frame(frame_middle, padding=(6,2,6,6), relief='groove')
frame_list2.grid(row=0, column=1, padx=4, sticky=(tk.E,tk.W,tk.S,tk.N))
frame_list2.columnconfigure(0, weight=1)
frame_list2.rowconfigure(2, weight=1)
frame_top2 = ttk.Frame(frame_list2, padding=(0,4,0,3))
frame_top2.grid(row=0, column=0, columnspan=2, sticky=(tk.E,tk.W))
button_open2 = ttk.Button(frame_top2, text="Open with MultiPar", width=20, command=button_open2_clicked, state=tk.DISABLED)
button_open2.pack(side=tk.LEFT, padx=2)
label_head2 = ttk.Label(frame_list2, text='0 finished items')
label_head2.grid(row=1, column=0, columnspan=2)
s_list2 = tk.StringVar()
listbox_list2 = tk.Listbox(frame_list2, listvariable=s_list2, activestyle='none')
listbox_list2.grid(row=2, column=0, sticky=(tk.E,tk.W,tk.S,tk.N))
scrollbar_list2 = ttk.Scrollbar(frame_list2, orient=tk.VERTICAL, command=listbox_list2.yview)
scrollbar_list2.grid(row=2, column=1, sticky=(tk.N, tk.S))
listbox_list2["yscrollcommand"] = scrollbar_list2.set
xscrollbar_list2 = ttk.Scrollbar(frame_list2, orient=tk.HORIZONTAL, command=listbox_list2.xview)
xscrollbar_list2.grid(row=3, column=0, sticky=(tk.E, tk.W))
listbox_list2["xscrollcommand"] = xscrollbar_list2.set
# Status text
frame_bottom = ttk.Frame(root)
frame_bottom.grid(row=2, column=0, sticky=(tk.E,tk.W))
label_status = ttk.Label(frame_bottom, text='Select folders and/or files to create PAR files.')
label_status.pack(side=tk.LEFT, padx=2)
# When a folder is specified in command-line
if len(sys.argv) == 2:
one_path = os.path.abspath(sys.argv[1])
if os.path.isdir(one_path):
search_child_item(one_path)
else:
add_argv_item()
# When multiple items are specified
elif len(sys.argv) > 2:
add_argv_item()
# If you want to start creation automatically, use below lines.
#if listbox_list1.size() > 0:
# button_parent.config(state=tk.DISABLED)
# button_child.config(state=tk.DISABLED)
# button_file.config(state=tk.DISABLED)
# button_stop.config(state=tk.NORMAL)
# root.after(100, queue_run)
# Show window
root.mainloop()