flexible.py 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. from twisted.internet import reactor
  2. from twisted.internet import task
  3. from twisted.internet import defer
  4. from colorama import Fore, Back, Style
  5. from twisted.python import log
  6. import pendulum
  7. def merge(color_string):
  8. """ Given a string of colorama ANSI, merge them if you can. """
  9. return color_string.replace("m\x1b[", ";")
  10. class PlayerInput(object):
  11. def __init__(self, game):
  12. # I think game gives us access to everything we need
  13. self.game = game
  14. self.observer = self.game.observer
  15. self.save = None
  16. self.deferred = None
  17. self.queue_game = game.queue_game
  18. # default colors, and useful consts
  19. self.c = merge(Style.BRIGHT + Fore.WHITE + Back.BLUE)
  20. self.r = Style.RESET_ALL
  21. self.nl = "\n\r"
  22. self.bsb = "\b \b"
  23. self.keepalive = None
  24. def color(self, c):
  25. self.c = c
  26. def alive(self):
  27. log.msg("PlayerInput.alive()")
  28. self.game.queue_player.put(" ")
  29. def prompt(self, prompt, limit, default=""):
  30. log.msg("PlayerInput({0}, {1}, {2}".format(prompt, limit, default))
  31. self.prompt = prompt
  32. self.limit = limit
  33. self.default = default
  34. self.input = ""
  35. assert self.save is None
  36. assert self.keepalive is None
  37. # Note: This clears out the server "keep alive"
  38. self.save = self.observer.save()
  39. self.observer.connect("player", self.get_input)
  40. self.keepalive = task.LoopingCall(self.alive)
  41. self.keepalive.start(30)
  42. # We need to "hide" the game output.
  43. # Otherwise it WITH mess up the user input display.
  44. self.to_player = self.game.to_player
  45. self.game.to_player = False
  46. # Display prompt
  47. self.queue_game.put(self.r + self.nl + self.c + prompt)
  48. # Set "Background of prompt"
  49. self.queue_game.put(" " * limit + "\b" * limit)
  50. assert self.deferred is None
  51. d = defer.Deferred()
  52. self.deferred = d
  53. log.msg("Return deferred ...", self.deferred)
  54. return d
  55. def get_input(self, chunk):
  56. """ Data from player (in bytes) """
  57. chunk = chunk.decode("utf-8", "ignore")
  58. for ch in chunk:
  59. if ch == "\b":
  60. if len(self.input) > 0:
  61. self.queue_game.put(self.bsb)
  62. self.input = self.input[0:-1]
  63. else:
  64. self.queue_game.put("\a")
  65. if ch == "\r":
  66. self.queue_game.put(self.r + self.nl)
  67. log.msg("Restore observer dispatch", self.save)
  68. assert not self.save is None
  69. self.observer.load(self.save)
  70. self.save = None
  71. log.msg("Disable keepalive")
  72. self.keepalive.stop()
  73. self.keepalive = None
  74. line = self.input
  75. self.input = ""
  76. assert not self.deferred is None
  77. # Ok, use deferred.callback, or reactor.callLater?
  78. # self.deferred.callback(line)
  79. reactor.callLater(0, self.deferred.callback, line)
  80. self.deferred = None
  81. if ch.isprintable():
  82. if len(self.input) + 1 <= self.limit:
  83. self.input += ch
  84. self.queue_game.put(ch)
  85. else:
  86. self.queue_game.put("\a")
  87. def output(self, line):
  88. """ A default display of what they just input. """
  89. log.msg("PlayerInput.output({0})".format(line))
  90. self.game.queue_game.put(self.r + self.nl + "[{0}]".format(line) + self.nl)
  91. return line
  92. class ProxyMenu(object):
  93. def __init__(self, game):
  94. self.nl = "\n\r"
  95. self.c = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  96. self.r = Style.RESET_ALL
  97. self.c1 = merge(Style.BRIGHT + Fore.BLUE)
  98. self.c2 = merge(Style.NORMAL + Fore.BLUE)
  99. self.game = game
  100. self.queue_game = game.queue_game
  101. self.observer = game.observer
  102. # Yes, at this point we would activate
  103. self.prompt = game.buffer
  104. self.save = self.observer.save()
  105. self.observer.connect("player", self.player)
  106. # If we want it, it's here.
  107. self.defer = None
  108. self.keepalive = task.LoopingCall(self.awake)
  109. self.keepalive.start(30)
  110. self.menu()
  111. def __del__(self):
  112. log.msg("ProxyMenu {0} RIP".format(self))
  113. def whenDone(self):
  114. self.defer = defer.Deferred()
  115. # Call this to chain something after we exit.
  116. return self.defer
  117. def menu(self):
  118. self.queue_game.put(
  119. self.nl + self.c + "TradeWars Proxy active." + self.r + self.nl
  120. )
  121. def menu_item(ch, desc):
  122. self.queue_game.put(
  123. " " + self.c1 + ch + self.c2 + " - " + self.c1 + desc + self.nl
  124. )
  125. self.queue_game.put(
  126. " " + self.c1 + "D" + self.c2 + " - " + self.c1 + "Diagnostics" + self.nl
  127. )
  128. menu_item("Q", "Quest")
  129. menu_item("T", "Display current Time")
  130. self.queue_game.put(
  131. " "
  132. + self.c1
  133. + "P"
  134. + self.c2
  135. + " - "
  136. + self.c1
  137. + "Port CIM Report"
  138. + self.nl
  139. )
  140. self.queue_game.put(
  141. " " + self.c1 + "S" + self.c2 + " - " + self.c1 + "Scripts" + self.nl
  142. )
  143. self.queue_game.put(
  144. " " + self.c1 + "X" + self.c2 + " - " + self.c1 + "eXit" + self.nl
  145. )
  146. self.queue_game.put(" " + self.c + "-=>" + self.r + " ")
  147. def awake(self):
  148. log.msg("ProxyMenu.awake()")
  149. self.game.queue_player.put(" ")
  150. def player(self, chunk):
  151. """ Data from player (in bytes). """
  152. chunk = chunk.decode("utf-8", "ignore")
  153. key = chunk.upper()
  154. log.msg("ProxyMenu.player({0})".format(key))
  155. # Stop the keepalive if we are activating something else
  156. # or leaving...
  157. self.keepalive.stop()
  158. if key == "T":
  159. self.queue_game.put(self.c + key + self.r + self.nl)
  160. # perform T option
  161. now = pendulum.now()
  162. self.queue_game.put(
  163. self.nl + self.c1 + "Current time " + now.to_datetime_string() + self.nl
  164. )
  165. elif key == "Q":
  166. self.queue_game.put(self.c + key + self.r + self.nl)
  167. # Ok, keepalive is stop(), and we are leaving this.
  168. # So, when the PlayerInput is done, have it call welcome_back,
  169. # which reinstates keepalive, and displays the menu.
  170. ask = PlayerInput(self.game)
  171. d = ask.prompt("What is your quest? ", 20)
  172. # Display the user's input
  173. d.addCallback(ask.output)
  174. # To "return" to the ProxyMenu, call self.welcome_back
  175. d.addCallback(self.welcome_back)
  176. return
  177. elif key == "X":
  178. self.queue_game.put(self.c + key + self.r + self.nl)
  179. self.observer.load(self.save)
  180. self.save = None
  181. # It isn't running (NOW), so don't try to stop it.
  182. # self.keepalive.stop()
  183. self.keepalive = None
  184. self.queue_game.put(self.prompt)
  185. self.prompt = None
  186. # Possibly: Send '\r' to re-display the prompt
  187. # instead of displaying the original one.
  188. # Were we asked to do something when we were done here?
  189. if self.defer:
  190. reactor.CallLater(0, self.defer.callback)
  191. # self.defer.callback()
  192. self.defer = None
  193. return
  194. self.keepalive.start(30, True)
  195. self.menu()
  196. def welcome_back(self, *_):
  197. log.msg("welcome_back")
  198. self.keepalive.start(30, True)
  199. self.menu()