11. 整合

优质
小牛编辑
133浏览
2023-12-01

英文原文

这是在 Kivy 1.0.8 版本以后添加的新功能。

在 Kivy 内使用 Twisted

特别注意

可以使用 kivy.support.install_twisted_reactor 这个函数来安装一个 twisted 反应器(reactor),这个反应器会在 Kivy 的事件循环内运行。

传递给此函数(kivy.support.install_twisted_reactor )的任何参数、关键字参数将在线程选择反应器交叉函数上传递(threadedselect reactors interleave function 这个我不懂是什么,强行翻译了,抱歉)。 这些参数通常是传递给twisted 的反应器启动函数(reactor.startRunning)的。

警告

Kivy 这里面的这个 reactore 和默认的 twisted reactor 反应器不一样,只有当你把 ‘installSignalHandlers’ 关键词参数设置为 1 来才去处理信号。这样做是为了保证 Kivy 能够按照常规情况来处理信号,而当你指定让 twisted reactor 来处理信号的时候再换用 twisted。(比如 SIGINT 智能信号等情境。)

Kivy 的样例代码中有一个例子,是一个简单的 twisted 服务端和客户端。服务端应用开启了一个 twisted 服务器,并对任何信息都进行日志记录。客户端的应用可以向服务端发送信息,然后接收服务端返回的信息并且输出显示。这些样例代码是基于 twisted 官方文档里面的简单的回声样例进行修改而实现的,原版代码可以在下面的地址中找到:

请按照如下步骤来尝试这个样例:先运行 echo_server_app.py, 然后运行 echo_client_app.py。在客户端的文本框中输入随便一些什么内容都可以,然后回车发送给服务端,服务端会把接收到的信息都原样返回,就像是回声一样。

服务端应用

  1. # install_twisted_rector must be called before importing and using the reactor
  2. # 一定要先要导入反应器,然后才能调用install_twisted_rector
  3. from kivy.support import install_twisted_reactor
  4. install_twisted_reactor()
  5. from twisted.internet import reactor
  6. from twisted.internet import protocol
  7. class EchoProtocol(protocol.Protocol):
  8. def dataReceived(self, data):
  9. response = self.factory.app.handle_message(data)
  10. if response:
  11. self.transport.write(response)
  12. class EchoFactory(protocol.Factory):
  13. protocol = EchoProtocol
  14. def __init__(self, app):
  15. self.app = app
  16. from kivy.app import App
  17. from kivy.uix.label import Label
  18. class TwistedServerApp(App):
  19. def build(self):
  20. self.label = Label(text="server started\n")
  21. reactor.listenTCP(8000, EchoFactory(self))
  22. return self.label
  23. def handle_message(self, msg):
  24. self.label.text = "received: %s\n" % msg
  25. if msg == "ping":
  26. msg = "pong"
  27. if msg == "plop":
  28. msg = "kivy rocks"
  29. self.label.text += "responded: %s\n" % msg
  30. return msg
  31. if __name__ == '__main__':
  32. TwistedServerApp().run()

客户端应用

  1. # install_twisted_rector must be called before importing the reactor
  2. # 一定要先要导入反应器,然后才能调用install_twisted_rector
  3. from kivy.support import install_twisted_reactor
  4. install_twisted_reactor()
  5. # A simple Client that send messages to the echo server
  6. # 这个简单的客户端是用来给回声服务器发送信息的
  7. from twisted.internet import reactor, protocol
  8. class EchoClient(protocol.Protocol):
  9. def connectionMade(self):
  10. self.factory.app.on_connection(self.transport)
  11. def dataReceived(self, data):
  12. self.factory.app.print_message(data)
  13. class EchoFactory(protocol.ClientFactory):
  14. protocol = EchoClient
  15. def __init__(self, app):
  16. self.app = app
  17. def clientConnectionLost(self, conn, reason):
  18. self.app.print_message("connection lost")
  19. def clientConnectionFailed(self, conn, reason):
  20. self.app.print_message("connection failed")
  21. from kivy.app import App
  22. from kivy.uix.label import Label
  23. from kivy.uix.textinput import TextInput
  24. from kivy.uix.boxlayout import BoxLayout
  25. # A simple kivy App, with a textbox to enter messages, and
  26. # a large label to display all the messages received from
  27. # the server
  28. # 这里的样例是一个很简单的 Kivy 应用,有一个字符输入框 textbox 来输入消息
  29. # 还有一个大的文本标签 label 来显示从服务器接收到的信息。
  30. class TwistedClientApp(App):
  31. connection = None
  32. def build(self):
  33. root = self.setup_gui()
  34. self.connect_to_server()
  35. return root
  36. def setup_gui(self):
  37. self.textbox = TextInput(size_hint_y=.1, multiline=False)
  38. self.textbox.bind(on_text_validate=self.send_message)
  39. self.label = Label(text='connecting...\n')
  40. self.layout = BoxLayout(orientation='vertical')
  41. self.layout.add_widget(self.label)
  42. self.layout.add_widget(self.textbox)
  43. return self.layout
  44. def connect_to_server(self):
  45. reactor.connectTCP('localhost', 8000, EchoFactory(self))
  46. def on_connection(self, connection):
  47. self.print_message("connected successfully!")
  48. self.connection = connection
  49. def send_message(self, *args):
  50. msg = self.textbox.text
  51. if msg and self.connection:
  52. self.connection.write(str(self.textbox.text))
  53. self.textbox.text = ""
  54. def print_message(self, msg):
  55. self.label.text += msg + "\n"
  56. if __name__ == '__main__':
  57. TwistedClientApp().run()