在未调用Popup.open()(奇异)的方法中触发Popup.dismiss()

时间:2019-06-28 20:16:57

标签: python multithreading user-interface popup kivy

我正在尝试为那些不太精明的同事开发一个kivy应用程序,该应用程序围绕我为项目开发的一些计算包装了一个不错的GUI;目前,我有两种方法嵌入类中,一种(称为“虚拟”)可防止GUI冻结,而从此类多线程中,第二种方法(称为“ calculate”)可实际运行计算。在虚拟方法中,我希望打开一个弹出窗口,该弹出窗口显示一个“正在加载GIF”(表明程序正在运行,而不仅仅是冻结),我希望该弹出窗口在“计算”方法完成时关闭。如何将弹出窗口的自动关闭功能与完成calculate(self,* args)方法绑定在一起?

-GUI.py-

from kivy.app import App
from kivy.uix.gridlayout import GridLayout
from kivy.uix.popup import Popup
import threading
from calc import main


class Pop(Popup):
    pass


class MetaLevel(GridLayout):
    def dummy(self, *args):
        Pop().open()
        threading.Thread(target=self.calculate, args=(args,)).start()

    def calculate(self, *args):
        main()


class graphics(App):
    def build(self):
        return MetaLevel()


if __name__ == "__main__":
    graphics().run()

-calc.py-

def main():
    import numpy as np
    from pathos.multiprocessing import ProcessPool as Pool

    grid = np.array([(m, n)
                     for m in np.arange(1, 100, 1)
                     for n in np.arange(1, 100, 1)])

    def calc(grid):
        var1 = grid[0]
        var2 = grid[1]
        y = var1*var2
        return y

    res = Pool().map(calc, grid)
    print('done')
    # data output from res here

-graphics.kv-

<Button>:
    font_size: 12

<MetaLevel>:
    id: calculator
    rows: 5
    padding: 10
    spacing: 10

    BoxLayout:
        height: 10
        Label:
            spacing: 10
            text: 'test'

    BoxLayout:
        Button:
            id: run_button
            text: "Run"
            on_release: root.dummy()

--- 编辑1 ---

仍在努力解决问题;我遇到了Clock.create_trigger()函数,并尝试将其集成到Pop类中-问题是我无法弄清楚如何在calculate方法的main()之后调用trigger()。如果我们可以触发扳机,那可能是解决该问题的可行方法。

-GUI.py-


from kivy.app import App
from kivy.uix.gridlayout import GridLayout
from kivy.uix.popup import Popup
import threading
from kivy.clock import Clock
from calc import main


class Pop(Popup):
    def __init__(self, **kwargs):
        super(Pop, self).__init__(**kwargs)
        trigger = Clock.create_trigger(self.dismiss_popup)

    def dismiss_popup(self, dt):
        self.dismiss()


class MetaLevel(GridLayout):
    def dummy(self, *args):
        Pop().open()
        threading.Thread(target=self.calculate, args=(args,)).start()

    def calculate(self, *args):
        main()
        trigger() # after main finishes I want to toggle the trigger, but kivy/python doesn't like this



class graphics(App):
    def build(self):
        return MetaLevel()


if __name__ == "__main__":
    graphics().run()

1 个答案:

答案 0 :(得分:1)

尝试将您的弹出窗口分配给您的类中的一个变量,该变量继承自App(我将其称为“主应用程序类”)。您可以通过在Python中使用App.get_running_app().your_variable或在kv语言中仅使用app.your_variable从主应用程序类中引用变量和函数。

根据您的情况,删除GUI.py中的行 Pop().open()并将其替换为以下行:

App.get_running_app().pop.open()

然后在您的graphics类中,为弹出窗口创建一个变量。您可以在build函数中执行此操作,只需添加self.pop = Pop()

现在,在您的calc.py程序中,您需要添加from kivy.app import App,然后在main函数的末尾添加一行以消除弹出窗口:

App.get_running_app().pop.dismiss()