4

MPlayer をウィンドウ内に埋め込むための基本的な PyGTK アプリを作成しようとしています (それ以外の場合は、私が好きな WM のタイリングではうまく機能しないため)。

これまでのコードはこの投稿の最後に記載しますが、基本的に私のセットアップには、「-wid」コマンドライン オプションを使用して MPlayer を埋め込む DrawingArea を含む Window が含まれています。

私が抱えている問題は、サイズを変更すると、次のような視覚的アーティファクトが発生することです (赤いボックス内を参照)。

MPlayer アーティファクト

configure-event が発生したときに DrawingArea で queue_draw() を呼び出してみましたが、効果がないようです。誰にもアイデアはありますか?

私の完全なコードは次のとおりです: (コマンドラインの使用法は「$0 [ vid ]」です)

#!/usr/bin/env python2

import sys
import os
import subprocess
import time
import string

import gtk
import gobject
import pygtk

pygtk.require('2.0')

class MPlayer:
    def __init__(self, path, draw, show_output=True):
        self.path = path
        self.draw = draw
        self.fifo = "/tmp/%s.%d" % (os.path.basename(__file__), time.time())

        # Start mplayer in draw
        cmd = string.split("mplayer -slave -wid %d -input file=%s" % \
                (self.draw.window.xid, self.fifo))
        cmd.append(self.path)
        if show_output:
            process = subprocess.Popen(cmd)
        else:
            self.devnull = open(os.devnull)
            process = subprocess.Popen(cmd, stdout=self.devnull, \
                    stderr=self.devnull)

        self.pid = process.pid

    def __enter__(self):
        os.mkfifo(self.fifo)
        return self

    def __exit__(self, ext_type, exc_value, traceback):
        if hasattr(self, "devnull"):
            self.devnull.close()
        os.unlink(self.fifo)

    # Send cmd to mplayer via fifo
    def exe(self, cmd, *args):
        if not self.pid: return
        full_cmd = "%s %s\n" % (cmd, string.join([str(arg) for arg in args]))
        with open(self.fifo, "w+") as fifo:
            fifo.write(full_cmd)
            fifo.flush()

class MPlayerWrapper:
    def __init__(self):
        self.window = gtk.Window(gtk.WINDOW_TOPLEVEL)
        self.draw = gtk.DrawingArea()
        self.mplayer = None
        self.setup_widgets()

    def setup_widgets(self):
        self.window.connect("destroy", gtk.main_quit)
        self.window.connect("key_press_event", self.key_press_event)
        self.draw.modify_bg(gtk.STATE_NORMAL, gtk.gdk.color_parse("black"))
        self.draw.connect("configure_event", self.redraw)
        self.window.add(self.draw)
        self.window.show_all()

    def mplayer_exe(self, cmd, *args):
        if self.mplayer:
            self.mplayer.exe(cmd, *args)

    def key_press_event(self, widget, event, data=None):
        self.mplayer_exe("key_down_event", event.keyval)

    def redraw(self, draw, event, data=None):
        self.draw.queue_draw()

    def play(self, path):
        with MPlayer(path, self.draw, True) as self.mplayer:
            gobject.child_watch_add(self.mplayer.pid, gtk.main_quit)
            gtk.main()

if __name__ == "__main__":
    wrapper = MPlayerWrapper()
    wrapper.play(sys.argv[1])
4

1 に答える 1

4

解決しました-解決策は追加することでした

-vo gl

mplayer への呼び出しに。つまり交換した

cmd = string.split("mplayer -slave -wid %d -input file=%s" % \
        (self.draw.window.xid, self.fifo))

cmd = string.split("mplayer -slave -vo gl -wid %d -input file=%s" % \
        (self.draw.window.xid, self.fifo))

上記のコードで、正しくサイズ変更されるようになりました。

このコマンド ライン オプションのおかげで、そもそもこのラッパー スクリプトを作成しようと思った理由が完全になくなりました (つまり、mplayer を、ムービー自体が必要としないスペースを占有する黒い境界線を持つウィンドウに埋め込み、dwm で使用するため)。 )、しかし、この修正は、他の理由で mplayer を埋め込みたいと考えている他の人にとってはまだ役立つと思います。

于 2013-05-26T04:04:54.360 に答える