diff --git a/README.md b/README.md
index 550d20b..ee3dc3a 100644
--- a/README.md
+++ b/README.md
@@ -6,8 +6,7 @@ Need python 3
# TODO
-- Add file rename functionality.
-- Attach copy, cut, paste, and delete signals to the controls menu.
+- Attach copy, cut, and paste signals to the controls menu.
- Add search functionality to the current directory.
- Add a settings file to store values.
- Improve icon detection for Steam.
diff --git a/bin/pytop-0-0-1-x64.deb b/bin/pytop-0-0-1-x64.deb
index 7d31965..7bbcac2 100644
Binary files a/bin/pytop-0-0-1-x64.deb and b/bin/pytop-0-0-1-x64.deb differ
diff --git a/src/debs/pytop-0-0-1-x64/opt/Pytop/PyTop.py b/src/debs/pytop-0-0-1-x64/opt/Pytop/PyTop.py
index 1cfdbc3..f427990 100755
--- a/src/debs/pytop-0-0-1-x64/opt/Pytop/PyTop.py
+++ b/src/debs/pytop-0-0-1-x64/opt/Pytop/PyTop.py
@@ -6,12 +6,13 @@ gi.require_version('Gtk', '3.0')
gi.require_version('WebKit2', '4.0')
from gi.repository import Gtk as gtk
+from gi.repository import Gdk as gdk
from gi.repository import WebKit2 as webkit
# Python imports
from utils import Settings, Events
-
+gdk.threads_init()
class Main:
def __init__(self):
webkit.WebView() # Needed for glade file to load...
diff --git a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Events.py b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Events.py
index b9f721e..3328d8d 100644
--- a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Events.py
+++ b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Events.py
@@ -1,25 +1,26 @@
# Gtk Imports
-import gi
+import gi, threading
+gi.require_version('Gtk', '3.0')
gi.require_version('Gdk', '3.0')
+from gi.repository import Gtk as gtk
from gi.repository import Gdk as gdk
+
# Python imports
-import threading
from .Grid import Grid
from .Dragging import Dragging
+
from threading import Thread
-
-gdk.threads_init()
class Events:
def __init__(self, settings):
- self.settings = settings
- self.builder = self.settings.returnBuilder()
- self.desktop = self.builder.get_object("Desktop")
- self.webview = self.builder.get_object("webview")
- self.desktopPath = self.settings.returnDesktopPath()
+ self.settings = settings
+ self.builder = self.settings.returnBuilder()
+ self.desktop = self.builder.get_object("Desktop")
+ self.webview = self.builder.get_object("webview")
+ self.desktopPath = self.settings.returnDesktopPath()
self.settings.setDefaultWebviewSettings(self.webview, self.webview.get_settings())
self.webview.load_uri(self.settings.returnWebHome())
@@ -30,19 +31,44 @@ class Events:
selectedDirDialog.add_filter(filefilter)
selectedDirDialog.set_filename(self.desktopPath)
+ self.grid = None
+ self.selectedFile = None
+
self.setDir(selectedDirDialog)
def setDir(self, widget, data=None):
- newPath = widget.get_filename()
- Thread(target=Grid(self.desktop, self.settings).generateDirectoryGrid, args=(newPath,)).start()
-
+ newPath = widget.get_filename()
+ self.grid = Grid(self.desktop, self.settings)
+ Thread(target=self.grid.generateDirectoryGrid, args=(newPath,)).start()
+ # Grid(self.desktop, self.settings).generateDirectoryGrid(newPath)
def showGridControlMenu(self, widget, data=None):
+ self.selectedFile = widget
popover = self.builder.get_object("gridControlMenu")
popover.show_all()
popover.popup()
+
+ # File control events
+ def renameFile(self, widget, data=None):
+ newName = widget.get_text().strip()
+ if data and data.keyval == 65293: # Enter key event
+ self.grid.renameFile(newName)
+ elif data == None: # Save button 'event'
+ self.grid.renameFile(newName)
+
+ def deleteFile(self, widget, data=None):
+ self.grid.deleteFile()
+
+ def copyFile(self):
+ pass
+
+ def cutFile(self):
+ pass
+
+
+ # Webview events
def showWebview(self, widget):
self.builder.get_object("webViewer").popup()
diff --git a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/FileHandler.py b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/FileHandler.py
index ac387e0..a1e80cb 100644
--- a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/FileHandler.py
+++ b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/FileHandler.py
@@ -23,20 +23,44 @@ class FileHandler:
self.MPLAYER_WH = " -xy 1600 -geometry 50%:50% ";
self.MPV_WH = " -geometry 50%:50% ";
+ self.selectedFile = None
+
+
def openFile(self, file):
print("Opening: " + file)
if file.lower().endswith(self.vids):
- subprocess.call([self.MEDIAPLAYER, self.MPV_WH, file])
+ subprocess.Popen([self.MEDIAPLAYER, self.MPV_WH, file])
elif file.lower().endswith(self.music):
- subprocess.call([self.MUSICPLAYER, file])
+ subprocess.Popen([self.MUSICPLAYER, file])
elif file.lower().endswith(self.images):
- subprocess.call([self.IMGVIEWER, file])
+ subprocess.Popen([self.IMGVIEWER, file])
elif file.lower().endswith(self.txt):
- subprocess.call([self.TEXTVIEWER, file])
+ subprocess.Popen([self.TEXTVIEWER, file])
elif file.lower().endswith(self.pdf):
- subprocess.call([self.PDFVIEWER, file])
+ subprocess.Popen([self.PDFVIEWER, file])
elif file.lower().endswith(self.office):
- subprocess.call([self.OFFICEPROG, file])
+ subprocess.Popen([self.OFFICEPROG, file])
else:
- subprocess.call(['xdg-open', file])
+ subprocess.Popen(['xdg-open', file])
+
+
+ def renameFile(self, oldFileName, newFileName):
+ try:
+ print("Renaming...")
+ print(oldFileName + " --> " + newFileName)
+ return 0
+ except Exception as e:
+ print("An error occured renaming the file:")
+ print(e)
+ return 1
+
+ def deleteFile(self, toDeleteFile):
+ try:
+ print("Deleting...")
+ print(toDeleteFile)
+ return 0
+ except Exception as e:
+ print("An error occured deleting the file:")
+ print(e)
+ return 1
diff --git a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Grid.py b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Grid.py
index 307dc7e..451a50b 100644
--- a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Grid.py
+++ b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Grid.py
@@ -9,47 +9,51 @@ from gi.repository import Gtk as gtk
from gi.repository import Gdk as gdk
from gi.repository import GObject as gobject
-
# Python imports
-from .Icon import Icon
-from .FileHandler import FileHandler
-import os, subprocess
+import os, subprocess, threading, hashlib
from os.path import isdir, isfile, join
from os import listdir
+from threading import Thread
+from .Icon import Icon
+from .FileHandler import FileHandler
+gdk.threads_init()
class Grid:
def __init__(self, desktop, settings):
self.desktop = desktop
self.settings = settings
self.filehandler = FileHandler()
- self.currentPath = ""
+ self.usrHome = settings.returnUserHome()
self.builder = self.settings.returnBuilder()
self.ColumnSize = self.settings.returnColumnSize()
+ self.thubnailGen = self.settings.getThumbnailGenerator()
+
+ self.currentPath = ""
+ self.selectedFile = ""
def generateDirectoryGrid(self, dirPath):
- dirPaths = ['.', '..']
- files = []
-
- # self.desktop.connect("button_press_event", self.showGridControlMenu, ())
+ loadProgress = self.builder.get_object('loadProgress')
+ loadProgress.set_text("Loading...")
+ loadProgress.set_fraction(0.0)
+ self.clearGrid(self.desktop)
+ self.currentPath = dirPath
+ dirPaths = ['.', '..']
+ files = []
for f in listdir(dirPath):
file = join(dirPath, f)
if self.settings.isHideHiddenFiles():
if f.startswith('.'):
continue
-
if isfile(file):
files.append(f)
else:
dirPaths.append(f)
- dirPaths.sort()
- files.sort()
-
- files = dirPaths + files
+ vidsList = ('.mkv', '.avi', '.flv', '.mov', '.m4v', '.mpg', '.wmv', '.mpeg', '.mp4', '.webm')
fractionTick = 1.0 / 1.0 if len(files) == 0 else len(files)
tickCount = 0.0
row = 0
@@ -57,17 +61,24 @@ class Grid:
x = 0
y = 0
- loadProgress = self.builder.get_object('loadProgress')
- loadProgress.set_text("Loading...")
- loadProgress.set_fraction(0.0)
- self.clearGrid(self.desktop)
+ dirPaths.sort()
+ files.sort()
+ files = dirPaths + files
for file in files:
- eveBox = Icon(self.settings).createIcon(dirPath, file)
- # self.drag.connectEvents(self.desktop, eveBox)
- eveBox.connect("button_press_event", self.iconClickEventManager, (eveBox,))
- eveBox.connect("enter_notify_event", self.settings.mouseOver, ())
- eveBox.connect("leave_notify_event", self.settings.mouseOut, ())
+ eveBox = gtk.EventBox()
+ # Generate any thumbnails beforehand...
+ if file.lower().endswith(vidsList):
+ fullPathFile = dirPath + "/" + file
+ fileHash = hashlib.sha256(str.encode(fullPathFile)).hexdigest()
+ hashImgpth = self.usrHome + "/.thumbnails/normal/" + fileHash + ".png"
+ if isfile(hashImgpth) == False:
+ self.generateVideoThumbnail(fullPathFile, hashImgpth)
+ eveBox = self.generateIcon(dirPath, file, col, row)
+ else:
+ eveBox = self.generateIcon(dirPath, file, col, row)
+
+ eveBox.show_all()
gobject.idle_add(self.addToGrid, (self.desktop, eveBox, col, row,))
tickCount = tickCount + fractionTick
loadProgress.set_fraction(tickCount)
@@ -77,9 +88,17 @@ class Grid:
col = 0
row += 1
- self.desktop.show_all()
loadProgress.set_text("Finished...")
+ def generateIcon(self, dirPath, file, col, row):
+ eveBox = Icon(self.settings).createIcon(dirPath, file)
+ # self.drag.connectEvents(self.desktop, eveBox)
+ eveBox.connect("button_press_event", self.iconClickEventManager, (eveBox,))
+ eveBox.connect("enter_notify_event", self.settings.mouseOver, ())
+ eveBox.connect("leave_notify_event", self.settings.mouseOut, ())
+ return eveBox
+
+
def addToGrid(self, args):
args[0].attach(args[1], args[2], args[3], 1, 1)
@@ -91,31 +110,63 @@ class Grid:
break
def iconClickEventManager(self, widget, eve, params):
- self.settings.setSelected(params[0])
- if eve.type == gdk.EventType.DOUBLE_BUTTON_PRESS:
+ try:
+ self.settings.setSelected(params[0])
children = widget.get_children()[0].get_children()
fileName = children[1].get_text()
dir = self.currentPath
file = dir + "/" + fileName
- if fileName == ".":
- self.generateDirectoryGrid(dir)
- elif fileName == "..":
- parentDir = os.path.abspath(os.path.join(dir, os.pardir))
- self.currentPath = parentDir
- self.generateDirectoryGrid(parentDir)
- elif isdir(file):
- self.currentPath = file
- Thread(target=self.generateDirectoryGrid, args=(self.currentPath,)).start()
- else:
- self.filehandler.openFile(file)
- elif eve.type == gdk.EventType.BUTTON_PRESS and eve.button == 3:
- children = widget.get_children()[0].get_children()
- input = self.builder.get_object("iconRenameInput")
- popover = self.builder.get_object("iconControlsWindow")
+ if eve.type == gdk.EventType.DOUBLE_BUTTON_PRESS:
+ if fileName == ".":
+ self.setNewPath(dir)
+ elif fileName == "..":
+ parentDir = os.path.abspath(os.path.join(dir, os.pardir))
+ self.currentPath = parentDir
+ self.setNewPath(parentDir)
+ elif isdir(file):
+ self.currentPath = file
+ self.setNewPath(self.currentPath)
+ elif isfile(file):
+ self.filehandler.openFile(file)
+ elif eve.type == gdk.EventType.BUTTON_PRESS and eve.button == 3:
+ input = self.builder.get_object("iconRenameInput")
+ popover = self.builder.get_object("iconControlsWindow")
+ self.selectedFile = file # Used for return to caller
- input.set_text(children[1].get_text())
- popover.set_relative_to(children[1])
- popover.set_position(gtk.PositionType.RIGHT)
- popover.show_all()
- popover.popup()
+ input.set_text(fileName)
+ popover.set_relative_to(children[1])
+ popover.set_position(gtk.PositionType.RIGHT)
+ popover.show_all()
+ popover.popup()
+ except Exception as e:
+ print(e)
+
+ def generateVideoThumbnail(self, fullPathFile, hashImgpth):
+ subprocess.call([self.thubnailGen, "-t", "65%", "-s", "300", "-c", "jpg", "-i", fullPathFile, "-o", hashImgpth])
+
+ def setNewPath(self, path):
+ self.generateDirectoryGrid(path)
+ # NOTE: Threading here causes seg faults. My theory is that
+ # the calling threading dies b/c the grid itself is cleard
+ # of children and the child is the one that has the 'thread'
+ #
+ # Thread(target=self.generateDirectoryGrid, args=(path,)).start()
+
+
+ # Pass through file control events
+ def renameFile(self, file):
+ newName = self.currentPath + "/" + file
+ status = self.filehandler.renameFile(self.selectedFile, newName)
+
+ if status == 0:
+ self.selectedFile = newName
+ self.generateDirectoryGrid(self.currentPath)
+
+
+ def deleteFile(self):
+ status = self.filehandler.deleteFile(self.selectedFile)
+
+ if status == 0:
+ self.selectedFile = ""
+ self.generateDirectoryGrid(self.currentPath)
diff --git a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Icon.py b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Icon.py
index 3288169..bd40516 100644
--- a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Icon.py
+++ b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Icon.py
@@ -8,7 +8,7 @@ from gi.repository import Gtk as gtk
from gi.repository import Gio as gio
from gi.repository import GdkPixbuf
-import os, subprocess, hashlib
+import os, hashlib
from os.path import isdir, isfile, join
@@ -51,10 +51,7 @@ class Icon:
if file.lower().endswith(vidsList):
fileHash = hashlib.sha256(str.encode(fullPathFile)).hexdigest()
hashImgpth = self.usrHome + "/.thumbnails/normal/" + fileHash + ".png"
- if isfile(hashImgpth) == False:
- self.generateVideoThumbnail(fullPathFile, hashImgpth)
-
- thumbnl = self.createIconImageFromBuffer(hashImgpth, self.viIconWxH)
+ thumbnl = self.createIconImageFromBuffer(hashImgpth, self.viIconWxH)
elif file.lower().endswith(imagesList):
thumbnl = self.createIconImageFromBuffer(fullPathFile, self.viIconWxH)
else:
@@ -76,9 +73,6 @@ class Icon:
return gtk.Image()
- def generateVideoThumbnail(self, fullPathFile, hashImgpth):
- subprocess.call(["ffmpegthumbnailer", "-t", "65%", "-s", "300", "-c", "jpg", "-i", fullPathFile, "-o", hashImgpth])
-
def getSystemThumbnail(self, filename,size):
final_filename = ""
if os.path.exists(filename):
diff --git a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Settings.py b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Settings.py
index 7d3fd0f..7a64ed4 100644
--- a/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Settings.py
+++ b/src/debs/pytop-0-0-1-x64/opt/Pytop/utils/Settings.py
@@ -16,16 +16,17 @@ class Settings:
self.DEFAULTCOLOR = gdk.RGBA(0.0, 0.0, 0.0, 0.0) # ~#00000000
self.MOUSEOVERCOLOR = gdk.RGBA(0.0, 0.9, 1.0, 0.64) # ~#00e8ff
self.SELECTEDCOLOR = gdk.RGBA(0.4, 0.5, 0.1, 0.84)
+ self.GTK_ORIENTATION = 1 # HORIZONTAL (0) VERTICAL (1)
+ self.THUMB_GENERATOR = "ffmpegthumbnailer"
- self.hideHiddenFiles = True
- self.usrHome = os.path.expanduser('~')
- self.desktopPath = self.usrHome + "/Desktop"
- self.ColumnSize = 10
- self.webHome = 'http://webfm.com/'
- self.GTK_ORIENTATION = 1 # HORIZONTAL (0) VERTICAL (1)
- self.iconContainerWxH = [128, -1]
+ self.hideHiddenFiles = True
+ self.webHome = 'http://webfm.com/'
+ self.usrHome = os.path.expanduser('~')
+ self.desktopPath = self.usrHome + "/Desktop"
+ self.iconContainerWxH = [128, 128]
self.systemIconImageWxH = [72, 72]
- self.viIconWxH = [128, -1]
+ self.viIconWxH = [256, 128]
+ self.ColumnSize = 8
def attachBuilder(self, builder):
@@ -70,8 +71,9 @@ class Settings:
def returnBuilder(self): return self.builder
def returnUserHome(self): return self.usrHome
def returnDesktopPath(self): return self.usrHome + "/Desktop"
- def returnColumnSize(self): return self.ColumnSize
def returnIconImagePos(self): return self.GTK_ORIENTATION
+ def getThumbnailGenerator(self): return self.THUMB_GENERATOR
+ def returnColumnSize(self): return self.ColumnSize
def returnContainerWH(self): return self.iconContainerWxH
def returnSystemIconImageWH(self): return self.systemIconImageWxH
def returnVIIconWH(self): return self.viIconWxH
diff --git a/src/versions/pytop-0.0.1/Pytop/PyTop.py b/src/versions/pytop-0.0.1/Pytop/PyTop.py
index 1cfdbc3..f427990 100755
--- a/src/versions/pytop-0.0.1/Pytop/PyTop.py
+++ b/src/versions/pytop-0.0.1/Pytop/PyTop.py
@@ -6,12 +6,13 @@ gi.require_version('Gtk', '3.0')
gi.require_version('WebKit2', '4.0')
from gi.repository import Gtk as gtk
+from gi.repository import Gdk as gdk
from gi.repository import WebKit2 as webkit
# Python imports
from utils import Settings, Events
-
+gdk.threads_init()
class Main:
def __init__(self):
webkit.WebView() # Needed for glade file to load...
diff --git a/src/versions/pytop-0.0.1/Pytop/utils/Events.py b/src/versions/pytop-0.0.1/Pytop/utils/Events.py
index b9f721e..3328d8d 100644
--- a/src/versions/pytop-0.0.1/Pytop/utils/Events.py
+++ b/src/versions/pytop-0.0.1/Pytop/utils/Events.py
@@ -1,25 +1,26 @@
# Gtk Imports
-import gi
+import gi, threading
+gi.require_version('Gtk', '3.0')
gi.require_version('Gdk', '3.0')
+from gi.repository import Gtk as gtk
from gi.repository import Gdk as gdk
+
# Python imports
-import threading
from .Grid import Grid
from .Dragging import Dragging
+
from threading import Thread
-
-gdk.threads_init()
class Events:
def __init__(self, settings):
- self.settings = settings
- self.builder = self.settings.returnBuilder()
- self.desktop = self.builder.get_object("Desktop")
- self.webview = self.builder.get_object("webview")
- self.desktopPath = self.settings.returnDesktopPath()
+ self.settings = settings
+ self.builder = self.settings.returnBuilder()
+ self.desktop = self.builder.get_object("Desktop")
+ self.webview = self.builder.get_object("webview")
+ self.desktopPath = self.settings.returnDesktopPath()
self.settings.setDefaultWebviewSettings(self.webview, self.webview.get_settings())
self.webview.load_uri(self.settings.returnWebHome())
@@ -30,19 +31,44 @@ class Events:
selectedDirDialog.add_filter(filefilter)
selectedDirDialog.set_filename(self.desktopPath)
+ self.grid = None
+ self.selectedFile = None
+
self.setDir(selectedDirDialog)
def setDir(self, widget, data=None):
- newPath = widget.get_filename()
- Thread(target=Grid(self.desktop, self.settings).generateDirectoryGrid, args=(newPath,)).start()
-
+ newPath = widget.get_filename()
+ self.grid = Grid(self.desktop, self.settings)
+ Thread(target=self.grid.generateDirectoryGrid, args=(newPath,)).start()
+ # Grid(self.desktop, self.settings).generateDirectoryGrid(newPath)
def showGridControlMenu(self, widget, data=None):
+ self.selectedFile = widget
popover = self.builder.get_object("gridControlMenu")
popover.show_all()
popover.popup()
+
+ # File control events
+ def renameFile(self, widget, data=None):
+ newName = widget.get_text().strip()
+ if data and data.keyval == 65293: # Enter key event
+ self.grid.renameFile(newName)
+ elif data == None: # Save button 'event'
+ self.grid.renameFile(newName)
+
+ def deleteFile(self, widget, data=None):
+ self.grid.deleteFile()
+
+ def copyFile(self):
+ pass
+
+ def cutFile(self):
+ pass
+
+
+ # Webview events
def showWebview(self, widget):
self.builder.get_object("webViewer").popup()
diff --git a/src/versions/pytop-0.0.1/Pytop/utils/FileHandler.py b/src/versions/pytop-0.0.1/Pytop/utils/FileHandler.py
index ac387e0..a1e80cb 100644
--- a/src/versions/pytop-0.0.1/Pytop/utils/FileHandler.py
+++ b/src/versions/pytop-0.0.1/Pytop/utils/FileHandler.py
@@ -23,20 +23,44 @@ class FileHandler:
self.MPLAYER_WH = " -xy 1600 -geometry 50%:50% ";
self.MPV_WH = " -geometry 50%:50% ";
+ self.selectedFile = None
+
+
def openFile(self, file):
print("Opening: " + file)
if file.lower().endswith(self.vids):
- subprocess.call([self.MEDIAPLAYER, self.MPV_WH, file])
+ subprocess.Popen([self.MEDIAPLAYER, self.MPV_WH, file])
elif file.lower().endswith(self.music):
- subprocess.call([self.MUSICPLAYER, file])
+ subprocess.Popen([self.MUSICPLAYER, file])
elif file.lower().endswith(self.images):
- subprocess.call([self.IMGVIEWER, file])
+ subprocess.Popen([self.IMGVIEWER, file])
elif file.lower().endswith(self.txt):
- subprocess.call([self.TEXTVIEWER, file])
+ subprocess.Popen([self.TEXTVIEWER, file])
elif file.lower().endswith(self.pdf):
- subprocess.call([self.PDFVIEWER, file])
+ subprocess.Popen([self.PDFVIEWER, file])
elif file.lower().endswith(self.office):
- subprocess.call([self.OFFICEPROG, file])
+ subprocess.Popen([self.OFFICEPROG, file])
else:
- subprocess.call(['xdg-open', file])
+ subprocess.Popen(['xdg-open', file])
+
+
+ def renameFile(self, oldFileName, newFileName):
+ try:
+ print("Renaming...")
+ print(oldFileName + " --> " + newFileName)
+ return 0
+ except Exception as e:
+ print("An error occured renaming the file:")
+ print(e)
+ return 1
+
+ def deleteFile(self, toDeleteFile):
+ try:
+ print("Deleting...")
+ print(toDeleteFile)
+ return 0
+ except Exception as e:
+ print("An error occured deleting the file:")
+ print(e)
+ return 1
diff --git a/src/versions/pytop-0.0.1/Pytop/utils/Grid.py b/src/versions/pytop-0.0.1/Pytop/utils/Grid.py
index 307dc7e..451a50b 100644
--- a/src/versions/pytop-0.0.1/Pytop/utils/Grid.py
+++ b/src/versions/pytop-0.0.1/Pytop/utils/Grid.py
@@ -9,47 +9,51 @@ from gi.repository import Gtk as gtk
from gi.repository import Gdk as gdk
from gi.repository import GObject as gobject
-
# Python imports
-from .Icon import Icon
-from .FileHandler import FileHandler
-import os, subprocess
+import os, subprocess, threading, hashlib
from os.path import isdir, isfile, join
from os import listdir
+from threading import Thread
+from .Icon import Icon
+from .FileHandler import FileHandler
+gdk.threads_init()
class Grid:
def __init__(self, desktop, settings):
self.desktop = desktop
self.settings = settings
self.filehandler = FileHandler()
- self.currentPath = ""
+ self.usrHome = settings.returnUserHome()
self.builder = self.settings.returnBuilder()
self.ColumnSize = self.settings.returnColumnSize()
+ self.thubnailGen = self.settings.getThumbnailGenerator()
+
+ self.currentPath = ""
+ self.selectedFile = ""
def generateDirectoryGrid(self, dirPath):
- dirPaths = ['.', '..']
- files = []
-
- # self.desktop.connect("button_press_event", self.showGridControlMenu, ())
+ loadProgress = self.builder.get_object('loadProgress')
+ loadProgress.set_text("Loading...")
+ loadProgress.set_fraction(0.0)
+ self.clearGrid(self.desktop)
+ self.currentPath = dirPath
+ dirPaths = ['.', '..']
+ files = []
for f in listdir(dirPath):
file = join(dirPath, f)
if self.settings.isHideHiddenFiles():
if f.startswith('.'):
continue
-
if isfile(file):
files.append(f)
else:
dirPaths.append(f)
- dirPaths.sort()
- files.sort()
-
- files = dirPaths + files
+ vidsList = ('.mkv', '.avi', '.flv', '.mov', '.m4v', '.mpg', '.wmv', '.mpeg', '.mp4', '.webm')
fractionTick = 1.0 / 1.0 if len(files) == 0 else len(files)
tickCount = 0.0
row = 0
@@ -57,17 +61,24 @@ class Grid:
x = 0
y = 0
- loadProgress = self.builder.get_object('loadProgress')
- loadProgress.set_text("Loading...")
- loadProgress.set_fraction(0.0)
- self.clearGrid(self.desktop)
+ dirPaths.sort()
+ files.sort()
+ files = dirPaths + files
for file in files:
- eveBox = Icon(self.settings).createIcon(dirPath, file)
- # self.drag.connectEvents(self.desktop, eveBox)
- eveBox.connect("button_press_event", self.iconClickEventManager, (eveBox,))
- eveBox.connect("enter_notify_event", self.settings.mouseOver, ())
- eveBox.connect("leave_notify_event", self.settings.mouseOut, ())
+ eveBox = gtk.EventBox()
+ # Generate any thumbnails beforehand...
+ if file.lower().endswith(vidsList):
+ fullPathFile = dirPath + "/" + file
+ fileHash = hashlib.sha256(str.encode(fullPathFile)).hexdigest()
+ hashImgpth = self.usrHome + "/.thumbnails/normal/" + fileHash + ".png"
+ if isfile(hashImgpth) == False:
+ self.generateVideoThumbnail(fullPathFile, hashImgpth)
+ eveBox = self.generateIcon(dirPath, file, col, row)
+ else:
+ eveBox = self.generateIcon(dirPath, file, col, row)
+
+ eveBox.show_all()
gobject.idle_add(self.addToGrid, (self.desktop, eveBox, col, row,))
tickCount = tickCount + fractionTick
loadProgress.set_fraction(tickCount)
@@ -77,9 +88,17 @@ class Grid:
col = 0
row += 1
- self.desktop.show_all()
loadProgress.set_text("Finished...")
+ def generateIcon(self, dirPath, file, col, row):
+ eveBox = Icon(self.settings).createIcon(dirPath, file)
+ # self.drag.connectEvents(self.desktop, eveBox)
+ eveBox.connect("button_press_event", self.iconClickEventManager, (eveBox,))
+ eveBox.connect("enter_notify_event", self.settings.mouseOver, ())
+ eveBox.connect("leave_notify_event", self.settings.mouseOut, ())
+ return eveBox
+
+
def addToGrid(self, args):
args[0].attach(args[1], args[2], args[3], 1, 1)
@@ -91,31 +110,63 @@ class Grid:
break
def iconClickEventManager(self, widget, eve, params):
- self.settings.setSelected(params[0])
- if eve.type == gdk.EventType.DOUBLE_BUTTON_PRESS:
+ try:
+ self.settings.setSelected(params[0])
children = widget.get_children()[0].get_children()
fileName = children[1].get_text()
dir = self.currentPath
file = dir + "/" + fileName
- if fileName == ".":
- self.generateDirectoryGrid(dir)
- elif fileName == "..":
- parentDir = os.path.abspath(os.path.join(dir, os.pardir))
- self.currentPath = parentDir
- self.generateDirectoryGrid(parentDir)
- elif isdir(file):
- self.currentPath = file
- Thread(target=self.generateDirectoryGrid, args=(self.currentPath,)).start()
- else:
- self.filehandler.openFile(file)
- elif eve.type == gdk.EventType.BUTTON_PRESS and eve.button == 3:
- children = widget.get_children()[0].get_children()
- input = self.builder.get_object("iconRenameInput")
- popover = self.builder.get_object("iconControlsWindow")
+ if eve.type == gdk.EventType.DOUBLE_BUTTON_PRESS:
+ if fileName == ".":
+ self.setNewPath(dir)
+ elif fileName == "..":
+ parentDir = os.path.abspath(os.path.join(dir, os.pardir))
+ self.currentPath = parentDir
+ self.setNewPath(parentDir)
+ elif isdir(file):
+ self.currentPath = file
+ self.setNewPath(self.currentPath)
+ elif isfile(file):
+ self.filehandler.openFile(file)
+ elif eve.type == gdk.EventType.BUTTON_PRESS and eve.button == 3:
+ input = self.builder.get_object("iconRenameInput")
+ popover = self.builder.get_object("iconControlsWindow")
+ self.selectedFile = file # Used for return to caller
- input.set_text(children[1].get_text())
- popover.set_relative_to(children[1])
- popover.set_position(gtk.PositionType.RIGHT)
- popover.show_all()
- popover.popup()
+ input.set_text(fileName)
+ popover.set_relative_to(children[1])
+ popover.set_position(gtk.PositionType.RIGHT)
+ popover.show_all()
+ popover.popup()
+ except Exception as e:
+ print(e)
+
+ def generateVideoThumbnail(self, fullPathFile, hashImgpth):
+ subprocess.call([self.thubnailGen, "-t", "65%", "-s", "300", "-c", "jpg", "-i", fullPathFile, "-o", hashImgpth])
+
+ def setNewPath(self, path):
+ self.generateDirectoryGrid(path)
+ # NOTE: Threading here causes seg faults. My theory is that
+ # the calling threading dies b/c the grid itself is cleard
+ # of children and the child is the one that has the 'thread'
+ #
+ # Thread(target=self.generateDirectoryGrid, args=(path,)).start()
+
+
+ # Pass through file control events
+ def renameFile(self, file):
+ newName = self.currentPath + "/" + file
+ status = self.filehandler.renameFile(self.selectedFile, newName)
+
+ if status == 0:
+ self.selectedFile = newName
+ self.generateDirectoryGrid(self.currentPath)
+
+
+ def deleteFile(self):
+ status = self.filehandler.deleteFile(self.selectedFile)
+
+ if status == 0:
+ self.selectedFile = ""
+ self.generateDirectoryGrid(self.currentPath)
diff --git a/src/versions/pytop-0.0.1/Pytop/utils/Icon.py b/src/versions/pytop-0.0.1/Pytop/utils/Icon.py
index 3288169..bd40516 100644
--- a/src/versions/pytop-0.0.1/Pytop/utils/Icon.py
+++ b/src/versions/pytop-0.0.1/Pytop/utils/Icon.py
@@ -8,7 +8,7 @@ from gi.repository import Gtk as gtk
from gi.repository import Gio as gio
from gi.repository import GdkPixbuf
-import os, subprocess, hashlib
+import os, hashlib
from os.path import isdir, isfile, join
@@ -51,10 +51,7 @@ class Icon:
if file.lower().endswith(vidsList):
fileHash = hashlib.sha256(str.encode(fullPathFile)).hexdigest()
hashImgpth = self.usrHome + "/.thumbnails/normal/" + fileHash + ".png"
- if isfile(hashImgpth) == False:
- self.generateVideoThumbnail(fullPathFile, hashImgpth)
-
- thumbnl = self.createIconImageFromBuffer(hashImgpth, self.viIconWxH)
+ thumbnl = self.createIconImageFromBuffer(hashImgpth, self.viIconWxH)
elif file.lower().endswith(imagesList):
thumbnl = self.createIconImageFromBuffer(fullPathFile, self.viIconWxH)
else:
@@ -76,9 +73,6 @@ class Icon:
return gtk.Image()
- def generateVideoThumbnail(self, fullPathFile, hashImgpth):
- subprocess.call(["ffmpegthumbnailer", "-t", "65%", "-s", "300", "-c", "jpg", "-i", fullPathFile, "-o", hashImgpth])
-
def getSystemThumbnail(self, filename,size):
final_filename = ""
if os.path.exists(filename):
diff --git a/src/versions/pytop-0.0.1/Pytop/utils/Settings.py b/src/versions/pytop-0.0.1/Pytop/utils/Settings.py
index 7d3fd0f..7a64ed4 100644
--- a/src/versions/pytop-0.0.1/Pytop/utils/Settings.py
+++ b/src/versions/pytop-0.0.1/Pytop/utils/Settings.py
@@ -16,16 +16,17 @@ class Settings:
self.DEFAULTCOLOR = gdk.RGBA(0.0, 0.0, 0.0, 0.0) # ~#00000000
self.MOUSEOVERCOLOR = gdk.RGBA(0.0, 0.9, 1.0, 0.64) # ~#00e8ff
self.SELECTEDCOLOR = gdk.RGBA(0.4, 0.5, 0.1, 0.84)
+ self.GTK_ORIENTATION = 1 # HORIZONTAL (0) VERTICAL (1)
+ self.THUMB_GENERATOR = "ffmpegthumbnailer"
- self.hideHiddenFiles = True
- self.usrHome = os.path.expanduser('~')
- self.desktopPath = self.usrHome + "/Desktop"
- self.ColumnSize = 10
- self.webHome = 'http://webfm.com/'
- self.GTK_ORIENTATION = 1 # HORIZONTAL (0) VERTICAL (1)
- self.iconContainerWxH = [128, -1]
+ self.hideHiddenFiles = True
+ self.webHome = 'http://webfm.com/'
+ self.usrHome = os.path.expanduser('~')
+ self.desktopPath = self.usrHome + "/Desktop"
+ self.iconContainerWxH = [128, 128]
self.systemIconImageWxH = [72, 72]
- self.viIconWxH = [128, -1]
+ self.viIconWxH = [256, 128]
+ self.ColumnSize = 8
def attachBuilder(self, builder):
@@ -70,8 +71,9 @@ class Settings:
def returnBuilder(self): return self.builder
def returnUserHome(self): return self.usrHome
def returnDesktopPath(self): return self.usrHome + "/Desktop"
- def returnColumnSize(self): return self.ColumnSize
def returnIconImagePos(self): return self.GTK_ORIENTATION
+ def getThumbnailGenerator(self): return self.THUMB_GENERATOR
+ def returnColumnSize(self): return self.ColumnSize
def returnContainerWH(self): return self.iconContainerWxH
def returnSystemIconImageWH(self): return self.systemIconImageWxH
def returnVIIconWH(self): return self.viIconWxH