ntpath.py上的特殊错误

时间:2013-12-09 19:26:23

标签: python windows-7 path subprocess

我正在尝试在Windows 7上的vlc播放器上编写类似于包装器的东西,以便它可以通过击键加载文件夹中的下一个和上一个文件。我在其中做的是,我将文件路径作为参数创建vlc类的实例并使用pyHook,扫描密钥,并在检测到特定键击时调用实例上的play_next和play_prev方法。这些方法通过终止最后一个进程并使用get_new_file方法找到的下一个文件生成新的vlc来工作。它在前几次工作,然后给出了特殊的错误。

None   
None   
Traceback (most recent call last):  
  File "C:\Python27\pyHook\HookManager.py", line 351, in KeyboardSwitch   
    return func(event)   
  File "filestuff.py", line 64, in kbwrap   
    kbeventhandler(event,instance)   
  File "filestuff.py", line 11, in kbeventhandler   
    instance.play_prev()   
  File "filestuff.py", line 34, in play_prev   
    f=self.get_new_file(-1)   
  File "filestuff.py", line 40, in get_new_file   
    dirname= os.path.dirname(self.fn)   
  File "C:\Python27\lib\ntpath.py", line 205, in dirname   
    return split(p)[0]   
  File "C:\Python27\lib\ntpath.py", line 178, in split   
    while head2 and head2[-1] in '/\\':   
TypeError: an integer is required

这是代码:

import os
import sys
import pythoncom, pyHook 
import win32api
import subprocess
import ctypes

def kbeventhandler(event,instance):

    if event.Key=='Home':
        instance.play_prev()
    if event.Key=='End':
        instance.play_next()
    return True

class vlc(object):
    def __init__(self,filepath,vlcp):
        self.fn=filepath
        self.vlcpath=vlcp
        self.process = subprocess.Popen([self.vlcpath, self.fn])
    def kill(self):
        PROCESS_TERMINATE = 1
        handle = ctypes.windll.kernel32.OpenProcess(PROCESS_TERMINATE, False, self.process.pid)
        ctypes.windll.kernel32.TerminateProcess(handle, -1)
        ctypes.windll.kernel32.CloseHandle(handle)
        print self.process.poll()
    def play_next(self):
        self.kill()
        f=self.get_new_file(1)
        self.process = subprocess.Popen([self.vlcpath, f])
        self.fn=f
    def play_prev(self):
        self.kill()
        f=self.get_new_file(-1)
        self.process = subprocess.Popen([self.vlcpath, f])
        self.fn=f

    def get_new_file(self,switch):

        dirname= os.path.dirname(self.fn)    
        supplist=['.mkv','.flv','.avi','.mpg','.wmv']
        files = [os.path.join(dirname,f) for f in os.listdir(dirname) if (os.path.isfile(os.path.join(dirname,f)) and os.path.splitext(f)[-1]in supplist)]
        files.sort()
        try: currentindex=files.index(self.fn)
        except: currentindex=0
        i=0
        if switch==1:
            if currentindex<(len(files)-1):i=currentindex+1

        else:
            if currentindex>0:i=currentindex-1

        return files[i]    

def main():
    vlcpath='vlc'
    if os.name=='nt': vlcpath='C:/Program Files (x86)/VideoLAN/VLC/vlc.exe'
    fn='H:\\Anime\\needless\\Needless_[E-D]\\[Exiled-Destiny]_Needless_Ep11v2_(04B16479).mkv'
    if len(sys.argv)>1:
        fn=sys.argv[1] #use argument if available or else use default file
    instance=vlc(fn,vlcpath)
    hm = pyHook.HookManager()
    def kbwrap(event):
        kbeventhandler(event,instance)
    hm.KeyDown = kbwrap
    hm.HookKeyboard()    
    pythoncom.PumpMessages()

if __name__ == '__main__':
    main() 

这里也是:http://pastebin.com/rh82XGzd

1 个答案:

答案 0 :(得分:1)

问题在于 在main中我设置hm.KeyDown = kbwrap然后从函数kbwrap调用实际事件处理程序kbeventhandler,但没有从kbwrap

返回任何值
def kbwrap(event):
    return kbeventhandler(event,flag)
hm.KeyDown = kbwrap

我还将vlc工作卸载到另一个线程,因为pyHook在子进程中运行得不好。 最终工作代码:

import os
import sys
import pythoncom, pyHook 
import win32api
import subprocess
import ctypes
import threading
from multiprocessing import *

class vlcThread(threading.Thread):
    def __init__(self,filepath,vlcp,fl):
        threading.Thread.__init__(self)
        self.fn,self.vlcpath,self.flag=filepath,vlcp,fl
        self.daemon=True

        self.start() # invoke the run method

    def run(self):
        vlcinstance=vlc(self.fn,self.vlcpath)
        while True:
            if(self.flag.value==1):
                vlcinstance.play_next()
                self.flag.value=0
            if(self.flag.value==-1):
                vlcinstance.play_prev()
                self.flag.value=0







def kbeventhandler(event,flag):

    if event.Key=='Home':
        flag.value =-1
        return False
    if event.Key=='End':
        flag.value =1
        return False
    return True

class vlc(object):
    def __init__(self,filepath,vlcp):
        self.fn=filepath
        self.vlcpath=vlcp
        self.process = subprocess.Popen([self.vlcpath,self.fn],close_fds=True)
    def kill(self):
        p, self.process = self.process, None
        if p is not None and p.poll() is None:
            p.kill() 
            p.wait()


    def play_next(self):
        self.kill()
        f=self.get_new_file(1)
        self.process = subprocess.Popen([self.vlcpath,f],close_fds=True)
        self.fn=f
    def play_prev(self):
        self.kill()
        f=self.get_new_file(-1)
        self.process = subprocess.Popen([self.vlcpath, f],close_fds=True)
        self.fn=f

    def get_new_file(self,switch):

        dirname= os.path.dirname(self.fn)    
        supplist=['.mkv','.flv','.avi','.mpg','.wmv','ogm','mp4']
        files = [os.path.join(dirname,f) for f in os.listdir(dirname) if (os.path.isfile(os.path.join(dirname,f)) and os.path.splitext(f)[-1]in supplist)]
        files.sort()
        try: currentindex=files.index(self.fn)
        except: currentindex=0
        i=0
        if switch==1:
            if currentindex<(len(files)-1):i=currentindex+1

        else:
            if currentindex>0:i=currentindex-1

        return files[i]    



def main():
    vlcpath='vlc'
    flag=Value('i')
    flag.value=0
    if os.name=='nt': vlcpath='C:/Program Files (x86)/VideoLAN/VLC/vlc.exe'
    fn='H:\\Anime\\needless\\Needless_[E-D]\\[Exiled-Destiny]_Needless_Ep11v2_(04B16479).mkv'
    if len(sys.argv)>1:
        fn=sys.argv[1] #use argument if available or else use default file

    t=vlcThread(fn,vlcpath,flag)
    hm = pyHook.HookManager()
    def kbwrap(event):
        return kbeventhandler(event,flag)
    hm.KeyDown = kbwrap
    hm.HookKeyboard()    
    pythoncom.PumpMessages()


if __name__ == '__main__':
    main()