diff --git a/gui/history.txt b/gui/history.txt
index c5e77ff..6e439a2 100644
--- a/gui/history.txt
+++ b/gui/history.txt
@@ -7,18 +7,16 @@
``````````````````````
-
-
-
-
-----------------------------
Version : 0.1.0
Date:
-----------------------------
- * Added auto-detect mode. (auto-pilot)
+ * Added auto-detect mode.
* Added a save feature.
- * Added the ability to select a file using directory navigation.
- * Added settings for you to adjust preferences
+ * Added a feature which allows you to select a file using directory navigation.
+ * Added settings for you to adjust preferences.
+ * Added a feature which allows you to load either a file or text from clipboard into the application.
+
-----------------------------
Version : Beta Release
diff --git a/gui/main.py b/gui/main.py
index b61ef8b..e1c6b82 100644
--- a/gui/main.py
+++ b/gui/main.py
@@ -21,7 +21,7 @@
import pathlib
from os import path
from subprocess import Popen, PIPE
-from os.path import getsize, splitext
+from os.path import getsize, splitext, realpath
import tempfile
from configparser import ConfigParser
import sys
@@ -44,10 +44,18 @@
drop_type = None #In saving, this checks whether a file or text was dropped
temp_files = [] #This is an array which contains a list of temporary files created.
checked = False #This is used in auto-detect mode to check if a files mode has been detected.
+file_path = ''
+#=====================================================================================
+# Constants
+PICTURES = ['.jpg','.jif','jfif','.jpe', '.png', '.ico', '.jpeg','.gif', '.tiff','.vda','.vst','.tif','.icb','.icns','.webp', '.ppm','.pgm', '.pnm', '.pbm', '.tga', '.pcx', '.jp2','.j2k', '.jpf','jpx','jpm','mj2', 'raw', 'bmp', 'xpm']
+AUDIOS = ['.mp3','.m4r', '.wav','.m4p','m4a', '.pcm','.aiff','.aif', '.aac', '.ogg', '.wma', '.flac', '.alac','.aifc', '.3ga', '.aa','.act', '.al','.bap','.cda','.mp2','.mpa','.oga']
+VIDEOS = ['.mp4','.m4b','.avi','.qt','.mov','.wmv','.flv','.avchd','.webm','.mkv','.ogv','.dirac', '.mpg', '.mpeg','.3gp','.nsv','.3g2','.svi','.m4v']
+#=====================================================================================
configurations = {
'default_mode':'encode',
'sys_argv_mode': None,
- 'output_dir' : 'output'
+ 'output_dir' : 'output',
+ 'temp_dir' : tempfile.gettempdir()
} #The default settings for configuration
#======================================
#Functions and classes
@@ -82,9 +90,13 @@ def show():
def main(self):
menu = Menu(self.menubar, tearoff = 0)
- menu.add_command(label = 'Settings', command = lambda : Menubar.settings(self))
menu.add_command(label = 'About', command = lambda: Menubar.about(self))
menu.add_separator()
+ menu.add_command(label = 'New Window', command = new_window)
+ menu.add_command(label = 'Restart Window', command = restart)
+ menu.add_separator()
+ menu.add_command(label = 'Settings', command = lambda : Menubar.settings(self))
+ menu.add_separator()
menu.add_command(label = 'Quit', command = lambda: root.quit())
self.menubar.add_cascade(label = 'More...', menu = menu)
@@ -95,10 +107,15 @@ def save_command():
configurations['default_mode'] = default_mode.get()
configurations['sys_argv_mode'] = sys_argv_variable.get()
configurations['output_dir'] = path_entry_variable.get()
+ configurations['temp_dir'] = temp_dir_variable.get()
settings_menu.destroy()
def select_dir():
path = askdirectory()
path_entry_variable.set(path)
+ def select_temp_dir():
+ #Function for choosing temporary tempfiles directory.
+ path = askdirectory()
+ temp_dir_variable.set(path)
#======================================
settings_menu = Toplevel()
settings_menu.transient(root)
@@ -110,10 +127,12 @@ def select_dir():
default_mode = StringVar()
path_entry_variable = StringVar()
sys_argv_variable = IntVar()
+ temp_dir_variable = StringVar()
#----------------------------------
default_mode.set(configurations['default_mode'])
sys_argv_variable.set(configurations['sys_argv_mode'])
path_entry_variable.set(configurations['output_dir'])
+ temp_dir_variable.set(configurations['temp_dir'])
#==============================================
style = ttk.Style()
#==============================================
@@ -134,9 +153,22 @@ def select_dir():
#
general_settings.grid(row = 0, column = 0)
#=============================================
-
+ path_settings = ttk.Frame(notebook)
+ #
+ output_dir_frame = ttk.LabelFrame(path_settings, text = 'Output directory')
+ path_entry = ttk.Entry(output_dir_frame, textvariable = path_entry_variable, state = 'disabled', width = 40).grid(row = 0, column = 0)
+ browse_button = ttk.Button(output_dir_frame, text = 'Browse...', command = select_dir).grid(row = 0, column = 1)
+ output_dir_frame.grid(row = 0, column = 0)
+ #
+ temp_folder_frame = ttk.LabelFrame(path_settings, text = 'Temporary folder path')
+ temp_path_entry = ttk.Entry(temp_folder_frame, textvariable = temp_dir_variable, state = 'disabled', width = 40).grid(row = 0, column = 0)
+ browse_button = ttk.Button(temp_folder_frame, text = 'Browse...', command = select_temp_dir).grid(row = 0, column = 1)
+ temp_folder_frame.grid(row = 1, column = 0)
+ #
+ path_settings.grid(row = 0, column = 0)
#=============================================
notebook.add(general_settings,text = 'General')
+ notebook.add(path_settings, text = 'Paths')
confirmation_frame = Frame(settings_menu)
button_ok = ttk.Button(confirmation_frame, text = 'OK', command = save_command).grid(row = 0, column = 0)
button_cancel = ttk.Button(confirmation_frame, text = 'Cancel', command = lambda: settings_menu.destroy()).grid(row = 0, column = 1)
@@ -157,14 +189,32 @@ def select_dir():
def rightclick_menu_results(event):
save_state = DISABLED
- if temp_file_name != '':
+ copy_html_state = DISABLED
+ if temp_file_name != '': #If the temp_file_name is empty.
save_state = NORMAL
+ copy_html_state = NORMAL
+ copy_css_state = NORMAL
+ if drop_type == 'text' or mode_variable.get() == 'decode': #If drop_type is text or mode is decode
+ copy_html_state = DISABLED
+ copy_css_state = DISABLED
elif drop_type == 'text':
save_state = NORMAL
+
+ suffix = str(pathlib.Path(file_path).suffix).lower()
+ if (suffix in PICTURES) or (suffix in AUDIOS) or (suffix in VIDEOS):
+ copy_html_state = NORMAL
+ else:
+ copy_html_state = DISABLED
+ if suffix in PICTURES:
+ copy_css_state = NORMAL
+ else:
+ copy_css_state = DISABLED
menu = Menu(menubar, tearoff = 0)
menu.add_command(label = 'Save', command = save_file, state = save_state)
menu.add_command(label = 'Save (Give ext.)', command = save_file_manual, state = save_state)
menu.add_command(label = 'Copy', command = copy_to_clipboard, state = save_state)
+ menu.add_command(label = 'Copy (HTML)', command = copy_html, state = copy_html_state)
+ menu.add_command(label = 'Copy (CSS)', command = copy_css, state = copy_css_state)
menu.tk_popup(event.x_root, event.y_root)
def rightclick_menu_file(event):
@@ -184,6 +234,7 @@ def rightclick_menu_file(event):
menu.add_command(label = 'Clipboard (Text)', state = text_state, command = load_text_clipboard)
menu.tk_popup(event.x_root, event.y_root)
+
def copy_to_clipboard():
if type(results) == str:
pyperclip.copy(results)
@@ -204,21 +255,34 @@ def save_file_manual():
with open(filename, 'wb') as file:
file.write(results)
-
+def restart():
+ os.startfile(__file__)
+ root.quit()
+
+def new_window():
+ os.startfile(__file__)
def save_file():
#Just saves a file
mode = mode_variable.get()
- if mode == 'encode':
+ directory = pathlib.Path(configurations['output_dir'])
+ try:
+ if str(directory) == 'output':
+ os.mkdir('output')
+ except FileExistsError:
+ pass
+ if mode == 'encode':
if drop_type == 'file' or drop_type == 'load':
- p = pathlib.Path(file_path).name
- directory = pathlib.Path(configurations['output_dir'])
- with open(p+'.txt', 'w') as file:
+ p = pathlib.Path(file_path).name
+ path = str(directory.joinpath(p))
+ with open(path + '.txt', 'w') as file:
file.write(results)
elif drop_type == 'text':
- p = asksaveasbfilename(filetypes = (("Text files", "*.txt"), ("All files", "*.*")))
+ p = asksaveasfilename(filetypes = (("Text files", "*.txt"), ("All files", "*.*")))
if p == '':
return
+ print(p)
+ path = str(directory.joinpath(p))
with open(p, 'w') as file:
file.write(results)
elif mode == 'decode':
@@ -226,7 +290,8 @@ def save_file():
if save_extension == None:
return
p = pathlib.Path(file_path).name
- with open(p+save_extension, 'wb') as file:
+ path = str(directory.joinpath(p))
+ with open(path+save_extension, 'wb') as file:
file.write(results)
elif drop_type == 'text':
@@ -275,7 +340,7 @@ def load_text_clipboard():
write_to_temp()
checked = False
else:
- insert(results[1:20000])
+ insert(results[:20000])
statusbar_label.config(fg ='black')
status_variable.set("Text too large to display all output.")
write_to_temp()
@@ -441,52 +506,72 @@ def get_file_type(filename):
print("File type is ", file_type)
if file_type == 'text/plain':
save_extension = '.txt'
+ return file_type
elif file_type == 'audio/mpeg':
save_extension = '.mp3'
+ return file_type
elif file_type == 'application/x-dosexec':
save_extension = '.exe'
+ return file_type
elif file_type == 'application/x-font-tff':
save_extension = '.ttf'
+ return file_type
elif file_type == 'image/png':
save_extension = '.png'
+ return file_type
elif file_type == 'image/gif':
save_extension = '.gif'
+ return file_type
elif file_type == 'application/octet-stream':
save_extension = '.txt'
+ return file_type
elif file_type == 'text/html':
save_extension = '.html'
+ return file_type
elif file_type == 'text/rtf':
save_extension = '.rtf'
+ return file_type
elif file_type == 'text/x-c++':
save_extension = '.cpp'
+ return file_type
elif file_type == 'text/x-python':
save_extension = '.py'
+ return file_type
elif file_type == 'text/x-msdos-batch':
save_extension = '.bat'
+ return file_type
elif file_type == 'text/x-php':
save_extension = '.php'
+ return file_type
elif file_type == 'application/zip':
save_extension = '.zip'
+ return file_type
elif file_type == 'application/x-gzip':
save_extension = '.tar'
+ return file_type
elif file_type == 'text/xml':
save_extension = '.xml'
+ return file_type
def write_to_temp():
global temp_file_name
- #This function writes the data to a temporary file.
- __, temp_file_name = tempfile.mkstemp()
- if mode_variable.get() == 'encode':
- with open(temp_file_name, 'w') as file:
- file.write(results)
- os.close(__)
- temp_files.append(temp_file_name)
- elif mode_variable.get() == 'decode':
- with open(temp_file_name, 'wb') as file:
- file.write(results)
- os.close(__)
- temp_files.append(temp_file_name)
- get_file_type(temp_file_name)
+ try:
+ #This function writes the data to a temporary file.
+ __, temp_file_name = tempfile.mkstemp(dir = configurations['temp_dir'])
+ if mode_variable.get() == 'encode':
+ with open(temp_file_name, 'w') as file:
+ file.write(results)
+ os.close(__)
+ temp_files.append(temp_file_name)
+ elif mode_variable.get() == 'decode':
+ with open(temp_file_name, 'wb') as file:
+ file.write(results)
+ os.close(__)
+ temp_files.append(temp_file_name)
+ get_file_type(temp_file_name)
+ except FileNotFoundError:
+ os.mkdir(configurations['output'])
+ write_to_temp()
def load_file(filename = None):
#In auto-detect mode, if the filename == None, it means the file's mode
@@ -573,6 +658,36 @@ def file_contents(file_path):
with open(file_path, 'rb') as file:
contents = file.read()
return contents, 'binary'
+
+def copy_html():
+ #Copy to clipboard for embeddding in HTML
+ path_suffix = str(pathlib.Path(file_path).suffix).lower()
+ if drop_type == 'file' or drop_type == 'load':
+ if path_suffix in PICTURES:
+ code = '"data:image/%s;base64,%s"' % (path_suffix[1:], results)
+ pyperclip.copy(code)
+ elif path_suffix in AUDIOS:
+ code = '"data:audio/%s;base64,%s"' % (path_suffix[1:], results)
+ pyperclip.copy(code)
+ elif path_suffix in VIDEOS:
+ code = '"data:video/%s;base64,%s"' % (path_suffix[1:], results)
+ pyperclip.copy(code)
+
+
+def copy_css():
+ #Copy to clipboard for embeddding in CSS
+ path_suffix = str(pathlib.Path(file_path).suffix).lower()
+ if drop_type == 'file' or drop_type == 'load':
+ if path_suffix in PICTURES:
+ code = 'url(data:image/%s;base64,%s)' % (path_suffix[1:], results)
+ pyperclip.copy(code)
+ elif path_suffix in AUDIOS:
+ code = 'url(data:audio/%s;base64,%s)' % (path_suffix[1:], results)
+ pyperclip.copy(code)
+ elif path_suffix in VIDEOS:
+ code = 'url(data:video/%s;base64,%s)' % (path_suffix[1:], results)
+ pyperclip.copy(code)
+
def drop_file(event):
global drop_type
global results
@@ -668,7 +783,7 @@ def drop_text(event):
insert(results)
write_to_temp()
else:
- insert(results[1:20000])
+ insert(results[:20000])
statusbar_label.config(fg ='black')
status_variable.set("Text too large to display all output.")
write_to_temp()
@@ -684,7 +799,7 @@ def drop_text(event):
write_to_temp()
checked = False
else:
- insert(results[1:20000])
+ insert(results[:20000])
statusbar_label.config(fg ='black')
status_variable.set("Text too large to display all output.")
write_to_temp()
@@ -799,7 +914,7 @@ def sys_argv():
sys_argv()
def init_configuration():
- # Note:
+ # This function is for starting and reading the configuration.
global mode_variable
global mode
try:
@@ -807,20 +922,25 @@ def init_configuration():
config.read('configuration.ini')
mode = config['general'].get('default_mode')
sys_argv_mode = config['general'].get('sys_argv_mode')
+ output_dir = config['paths'].get('output_dir')
+ temp_dir = config['paths'].get('temp_dir')
#____________________________________________________________
configurations['default_mode'] = mode
configurations['sys_argv_mode'] = sys_argv_mode
+ configurations['output_dir'] = output_dir
+ configurations['temp_dir'] = temp_dir
if mode == 'detect':
auto_detect_variable.set(1)
mode_variable.set(mode)
except:
mode = 'encode'
configurations['default_mode'] = mode
+ configurations['output_dir'] = 'output'
configurations['sys_argv_mode'] = 1
+ configurations['temp_dir'] = tempfile.gettempdir()
mode_variable.set('encode')
init_configuration()
-
#===========================================================
#Frames
#===================================
@@ -849,14 +969,14 @@ def init_configuration():
text_frame.grid(row = 0, column = 1, sticky = E+W+N+S)
controls_frame.grid(row = 1, column = 0, columnspan = 2, sticky = W+E+N)
statusbar_frame.grid(row = 2, column = 0, sticky = W+E, columnspan = 2)
-#========================================
+#==================================================================
# Widgets
mode_encode.grid(row = 0, column = 0, sticky = W)
mode_auto_detect.grid(row = 0, column = 0)
mode_decode.grid(row = 0, column = 2)
text_area.grid(row = 0, column = 0, sticky = N+E+W+S)
statusbar_label.grid(row = 0, column = 0)
-#=====================================
+#==========================================================
# Making widgets stretchable
root.grid_rowconfigure(0, weight = 1)
root.grid_columnconfigure(0, weight = 1, minsize = 150)
@@ -891,9 +1011,9 @@ def init_configuration():
#===========================================
sys_argv()
root.mainloop()
-#============================================================
+#==========================================================================
#End
-#============================================================
+#==========================================================================
try:
#This statement deletes all the temporary files created
#by the application.
@@ -912,5 +1032,9 @@ def init_configuration():
'default_mode' : configurations['default_mode'],
'sys_argv_mode' : configurations['sys_argv_mode'],
}
+configuration['paths'] = {
+ 'output_dir' : configurations['output_dir'],
+ 'temp_dir' : configurations['temp_dir']
+}
with open('configuration.ini', 'w') as file:
configuration.write(file)
diff --git a/gui/readme.txt b/gui/readme.txt
index 786f1ed..4fc78c9 100644
--- a/gui/readme.txt
+++ b/gui/readme.txt
@@ -8,9 +8,92 @@ to base64 format and can decode base64 contents back to it's original format.
It's cool auto-detecting feature allows you to encode and decode with ease. Just drop a file, it will detect the mode
and perform the action.
+
+MODES
+-------
+Encode : This mode encodes an object into base64 format.
+Decode : This function decodes an object from base64 format.
+auto-detect : This function on it's own set the mode to encode if your file needs to be encoded.
+It will also set the mode to decode if your file needs to be decoded.
+
HOW TO USE
-------------
-1. Set a mode, either 'encode' or 'decode'
-2. Drop a file onto the surface of the program.
-3. You then recieve contents in the results panel.
+-------------
+1. Set a mode,
+NB: If auto-detect is active, it will overide your default mode and choose a mode depending on the object.
+2. Drag and drop a file onto the application's left pane,
+the results will appear on the right side pane of the application.
+
+HOW TO SAVE
+------------
+The application allows you to save either the encoded or the decoded contents.
+After either encoding or decoding, right-click on the right side pane, a pop-up menu will apppear,
+select save.
+
+NB: If you are saving a decoded file, the application automatically use the object's mime-type to
+generate an extension for your file, meaning, you don't have to know the file type of the contents
+you are decoding.
+To overide this feature, select the option 'Save (Give ext.)', this options allows you yourself
+to generate a filename with your prefered extension.
+
+SETTINGS
+-----------
+
+General settings
+..................
+
+Default mode: This allows you to set the mode you want on application start.
+This mode is very useful if you want to encode or decode a file without opening the application.
+The best mode for this is 'auto-detect'. For instance, if the mode is on auto-detect, and your drop
+a file on Quick Base64's exe, it will determine the mode and encode/decode. This way, the application
+will start with the results, saving you a bunch of time.
+
+Drop file and work on application start? : This option allows you to disable the ability of a file being encoded or decoded when dropped on Quick Base64's exe while's it's closed.
+If set to 'yes', when a file is dropped on Quick Base64's exe, a command will be executed, command being
+either encode or decode.
+If set to 'no', when a file is dropped on Quick Base64's exe, it will just open the application.
+
+Paths
+..............
+
+Output directory : This enables you to specify a directory where all encoded or decoded files will
+be saved.
+
+Temporary file directory : This allows you to specify the temporary directory Quick Base64 will use.
+Whenever you close Quick Base64, the temporary files are deleted, but on a case of an application crash,
+these temporary files, wont be deleted, specifying the temporary folder is useful as it gives you control over the temporary files.
+
+Right-Click Context Menu on Right Panel.
+ 1. Save
+ This option saves the encoded/decoded contents to a file.
+ 2. Save (Give ext.)
+ This option allows you to determine the filename and extension of the file to be saved.
+ 3. Copy
+ Copies the raw decoded/encoded contents to clipboard.
+ 4. Copy (HTML)
+ Converts the encoded results into base64 format which you can embed in html.
+ For example, if you encode a file 'picture.png', this option will produce the following
+ results, "".
+ NB: The results is already wrapped in quotation marks so when coding it in HTML,
+ just write the tag to paste the results,
+ wrong : url(data:mime/type;base64, base64contents)
+ This command works for files which are images only.
+
+ Right-Click Context Menu on Left Panel.
+ 1.Load file:
+ This option allows you to select a file using the windows file selector dialog.
+ 2. Clipboard (File):
+ This enables you to embed a file which you have copied to clipboard into the application.
+ NB: Quick Base64 is able to differentiate between text and files in clipboard.
+ 3. Clipboard (Text):
+ This allows you to embed text in your clipboard into the application.
+
+
+============================================================================================
+Report bugs to support@eakloe.com or make an issue on github at github.com/biah/quickbase64
\ No newline at end of file