Skip to main content

Opening Files with your Python application

For a long time now I couldn't find any help online for my problem. I want to make my own image viewer, since the one that comes with Windows 10 is so bad, but I want to be able to right click on any image, and select open with my Python application. .
Lets take a look at my simple program:

1:  import sys  
2:    
3:  try:  
4:    print(sys.argv[1])  
5:  except IndexError:  
6:    print("Being run without file selected.")  
7:    
8:  input("\n\nPress any key to exit...")  

I want to just get a simple program working first, i'll build the application later.

Normally argv would be used to parse command line arguments into a program but it turns out that when your program is compiled, when you use 'open with' your program, the second argument will be set to the path to which the file is located. The first argument is the name of the program. So the program prints the second argument. If there isn't one, if the program is just run, then this will throw an IndexError. This is then caught using the try/except block. Then I just do a simple input statement so that the window doesn't just instantly close.

Now I compile this Python script. To do this I use cx_Freeze. Here's the setup script that I used:

1:  from cx_Freeze import setup, Executable  
2:    
3:  setup(  
4:    name = "test_reg",  
5:    version = "0.0.2",  
6:    description = "test_reg",  
7:    executables = [Executable("test_reg.py", base = None)]  
8:  )  

I use base = None so that the console isn't removed. Now I run setup.py

1:  python setup.py build_exe  

The resulting application is really ugly, but it's just a proof of concept. Here's what happens when you run the resulting executable:


Ok, lets try running on an image. First I select 'Choose another app':


Then I selected 'Look for another app on this PC' (test_reg.exe didn't appear the first time) and navigated to the directory in which I had made the executable. The user could select the check box the second time they opened the application to always use my program.



Success! The name of the file was parsed into the application. Now I can create my own GUI to show the image.


Edit: After doing this method with a tkinter application, I kept getting dll errors. This is because cx_Freeze with tkinter requires a couple of dlls, namely tcl86t.dll and tk86t.dll. Normally this is fine because they are in the same directory as the .exe file. But when you run the .exe on an image file. the current working directory (cwd) will be directory in which the image is located, not the directory of the .exe.

I solved this by having the main program a simple python script that changes the cwd to the one in which the one with the dlls in, by using

1:  os.chdir()  

then running the tkinter app as a module. Then switch back to the original cwd.

The hard part is knowing where the application is installed, which is hard to know if the user installed the app using an msi or something.


Comments

Popular posts from this blog

FPS counter with OpenCV

Showing FPS on the screen is an important part of benchmarking your app. In this post we're going to look at how to calculate the number of frames per second and show it on screen. The theory is that we keep track of the last 5 or so timestamps and average over to get the FPS. We keep deleting the last item so the list only has 5 times in it. Lets have a look at the code: 1: import time 2: 3: times = [] 4: count = 5 5: 6: times.append(time.time()) 7: if len(times) >= count: 8: times = times[-count:] 9: fps = "%.2f FPS" % (count/(times[-1] - times[0])) 10: cv2.putText(frame, fps, (0, 20), cv2.FONT_HERSHEY_SIMPLEX, 1, (0, 255, 0), 1, cv2.LINE_AA) First import time, which is required for this to work. Then I define two constants which will be used in the function. Remember this code will run every time the videoloop runs, it adds the time to the list. If the list gets too long, get the last bit using indexing. Then work out the av...

Getting the icon association for any file or folder with the Python Windows API

I wanted to make a file browser which has the correct file icons for each file type, instead of just getting them from the file name. This is a bad solution, since it has to call images from disc and may be wrong if the name is different. This method uses the Windows API to get the exact icon. Tested in Python 3.6 in Windows 10. from win32com.shell import shell, shellcon from PIL import Image, ImageTk import win32api import win32con import win32ui import win32gui def get_icon(PATH, size): SHGFI_ICON = 0x000000100 SHGFI_ICONLOCATION = 0x000001000 if size == "small": SHIL_SIZE= 0x00001 elif size == "large": SHIL_SIZE= 0x00002 else: raise TypeError("Invalid argument for 'size'. Must be equal to 'small' or 'large'") ret, info = shell.SHGetFileInfo(PATH, 0, SHGFI_ICONLOCATION | SHGFI_ICON | SHIL_SIZE) hIcon, iIcon, dwAttr, name, typeName = info ico_x = ...

Using inno to package your Python application

So you have just compiled your Windows Python application. You want to make an installer for your users to easily get the application onto their PCs. You have tried using cx_Freeze's bdist_msi option but it is not advanced enough. Perhaps you want to run a script on installation or on uninstall, or want an easy way of adding an icon to the user's desktop. Now the easiest tool to use is inno. In this post we're going to take a look at how to use inno's setup wizard to package your application, as well as editing the script file. So the first thing you are going to want to do is to jump onto  inno's download page  and download inno setup compiler. Once it's done, open the application and select 'Create a new script file using the Script Wizard'. You will then be greeted with the following window: Leave the box unchecked. We're going to use the Wizard to make the base of our install script. Fill out all the information in the entry bo...