2

我正在设计一个包含多个组件和两个 wx.Frame 对象 F1 和 F2 的 GUI。F1 是主框架,F2 是辅助框架。我想有一个机制,所以用户可以将这两个框架附加到一个框架中,如果需要也可以将它们再次分离到两个框架中。

假设 F1 和 F2 分别包含面板 P1 和 P2。分离时,用户应该能够独立移动和调整每一帧的大小,关闭 F1 将关闭整个 GUI。连接后,F1 将垂直包含 P1 和 P2,F2 似乎消失并成为 F1 的一部分。在 P1 和 P2 之间传递了很多连线、事件和消息,它们应该在附加和分离模式下都可以工作。

我已经在一些现代 GUI 中看到了这种效果,但我无法在网上找到合适的技术来实现这一点。这样做的正确方法是什么?

谢谢

4

3 回答 3

2

我想出了一个使用 pubsub 模块的解决方案。以下是我写的一个小例子来展示它是如何完成的:

import wx
import gettext
from wx.lib.pubsub import pub

class SubFramePanel(wx.Panel):
    def __init__(self, parent):
        wx.Panel.__init__(self, parent, wx.ID_ANY)
        self.attachDetachButton = wx.Button(self, wx.ID_ANY, _("Attach"))
        self.sayHelloButton = wx.Button(self, wx.ID_ANY, _("Say Hello"))        
        subPanelSizer = wx.BoxSizer(wx.HORIZONTAL)
        subPanelSizer.Add(self.attachDetachButton, 0, wx.ALIGN_CENTER_HORIZONTAL | wx.ALIGN_CENTER_VERTICAL, 0)
        subPanelSizer.Add(self.sayHelloButton, 0, wx.ALIGN_RIGHT | wx.ALIGN_CENTER_VERTICAL, 0)
        self.SetSizer(subPanelSizer)
        self.attachDetachButton.Bind(wx.EVT_BUTTON, self.OnAttachDetachButton)
        self.sayHelloButton.Bind(wx.EVT_BUTTON, self.OnSayHelloButton)

    def OnAttachDetachButton(self, event):
        if self.attachDetachButton.GetLabel() == "Attach":
            self.attachDetachButton.SetLabel("Detach")
            pub.sendMessage("show.mainframe.OnAttach", data=self)
        else:
            self.attachDetachButton.SetLabel("Attach")
            pub.sendMessage("show.mainframe.OnDetach", data=self)
        event.Skip()

    def OnSayHelloButton(self, event):
        pub.sendMessage("show.mainframe.addText", data="Say Hello\n")
        event.Skip()


class SubFrame(wx.Frame):
    def __init__(self, *args, **kwds):
        kwds["style"] = wx.DEFAULT_FRAME_STYLE
        if kwds.has_key("panel"):
            self.panel = kwds["panel"]
            del kwds["panel"]
        else:
            self.panel = None

        wx.Frame.__init__(self, *args, **kwds)

        if self.panel is None:
            self.panel = SubFramePanel(self)
        else:
            self.panel.Reparent(self)

        self.SetTitle(_("Sub Frame"))
        self.SetSize((291, 93))

        subFrameSizer = wx.BoxSizer(wx.VERTICAL)

        subFrameSizer.Add(self.panel, 1, wx.EXPAND | wx.LEFT, 5)
        self.SetSizer(subFrameSizer)
        self.Layout()
        pub.subscribe(self.OnClose, "show.subframe.OnClose")

    def OnClose(self, data=None):
        self.Close()


# end of class SubFrame

class MainFrame(wx.Frame):
    def __init__(self, *args, **kwds):
        kwds["style"] = wx.DEFAULT_FRAME_STYLE
        wx.Frame.__init__(self, *args, **kwds)
        self.text_ctrl_1 = wx.TextCtrl(self, wx.ID_ANY, "", style=wx.TE_MULTILINE)

        pub.subscribe(self.OnAddText, "show.mainframe.addText")
        pub.subscribe(self.OnAttach, "show.mainframe.OnAttach")
        pub.subscribe(self.OnDetach, "show.mainframe.OnDetach")

        self.SetTitle(_("Main Frame"))
        self.SetSize((492, 271))
        self.mainFrameSizer = wx.BoxSizer(wx.VERTICAL)
        self.mainFrameSizer.Add(self.text_ctrl_1, 1, wx.ALL | wx.EXPAND, 5)
        self.SetSizer(self.mainFrameSizer)
        self.Layout()

    def OnAddText(self, data):
        self.text_ctrl_1.WriteText(data)

    def OnAttach(self, data):
        self.mainFrameSizer.Add(data, 0, wx.ALL | wx.EXPAND, 5)
        data.Reparent(self)
        self.Layout()
        pub.sendMessage("show.subframe.OnClose")

    def OnDetach(self, data):
        subFrame = SubFrame(self, wx.ID_ANY, "", panel=data)
        self.mainFrameSizer.Remove(data)
        self.Layout()
        subFrame.Show()

class MyApp(wx.App):
    def OnInit(self):
        mainFrame = MainFrame(None, wx.ID_ANY, "")
        self.SetTopWindow(mainFrame)
        mainFrame.Show()
        subFrame = SubFrame(mainFrame, wx.ID_ANY, "")
        subFrame.Show()
        return 1

if __name__ == "__main__":
    gettext.install("app")
    app = MyApp(0)
    app.MainLoop()
于 2013-05-16T18:36:24.937 回答
0

我不确定您是否可以在 wx 上即时将 awxPanel从 a移动到另一个。wxFrame

主要原因是面板依赖于其父级,您无法即时更改它。

现在,如果您真的想这样做,则必须在另一个框架中创建面板的副本并删除以前的面板和框架(或只是隐藏它们)。

没有内置副本,但您可以找到一种方法来获取原始面板的内容并将其复制到另一个面板上。

于 2013-05-14T07:18:32.250 回答
0

wxPython 中有一个名为 AUI 的库。它提供了将面板与框架分离的机制。以下链接有一个示例以及一些其他信息:

http://wiki.wxpython.org/AuiNotebook%20(AGW)

于 2014-05-30T18:19:02.740 回答