boxes.py 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. from colorama import Fore, Back, Style
  2. # see http://xahlee.info/comp/unicode_drawing_shapes.html
  3. def merge(color_string):
  4. """ Given a string of colorama ANSI, merge them if you can. """
  5. return color_string.replace("m\x1b[", ";")
  6. class Boxes(object):
  7. """ Boxes makes nice ASCII graphic menus
  8. Example:
  9. c1 = merge(Style.BRIGHT + Fore.CYAN)
  10. c2 = merge(Style.NORMAL + Fore.CYAN)
  11. # Set the size of the box to 40 (Important!)
  12. box = Boxes(40, color=c1)
  13. self.queue_game.put(box.top())
  14. # format string "{0:^40}".format("Text Centered")
  15. # This centers the text into the size of the box.
  16. self.queue_game.put(box.row(c1 + "{0:^40}".format("Scripts")))
  17. # this draws the middle line. (The line that goes across, but is
  18. # connected on both sides to the one above it).
  19. self.queue_game.put(box.middle())
  20. def menu_item(ch, desc):
  21. # Again we want the final string to be 40!
  22. # c1 and c2 (below) take 0 space.
  23. # We have " " + ch + " " + "-" + " " .. which equals 5
  24. # So desc needs {:35} to pad out string out to 40 print chars.
  25. row = " {0}{1} {2}-{3} {4:35}".format(c1, ch, c2, c1, desc)
  26. # self.queue_game.put(
  27. # " " + c1 + ch + c2 + " - " + c1 + desc + self.nl
  28. # )
  29. self.queue_game.put(box.row(row))
  30. menu_item("1", "Ports (Trades between two sectors)")
  31. menu_item("2", "Explore (Strange new sectors)")
  32. menu_item("3", "Space... the final frontier...")
  33. menu_item("X", "eXit")
  34. self.queue_game.put(box.bottom())
  35. """
  36. box_styles = (
  37. # ┌──┐
  38. # │ │
  39. # ├──┤
  40. # └──┘
  41. {
  42. "tl": "\xda",
  43. "tr": "\xbf",
  44. "top": "\xc4",
  45. "side": "\xb3",
  46. "bl": "\xc0",
  47. "br": "\xd9",
  48. "ml": "\xc3",
  49. "mr": "\xb4",
  50. },
  51. # ╔══╗
  52. # ║ ║
  53. # ╠══╣
  54. # ╚══╝
  55. {
  56. "tl": "\xc9",
  57. "tr": "\xbb",
  58. "top": "\xcd",
  59. "side": "\xba",
  60. "bl": "\xc8",
  61. "br": "\xbc",
  62. "ml": "\xcc",
  63. "mr": "\xb9",
  64. },
  65. # ╒══╕
  66. # │ │
  67. # ╞══╡
  68. # ╘══╛
  69. {
  70. "tl": "\xd5",
  71. "tr": "\xb8",
  72. "top": "\xcd",
  73. "side": "\xb3",
  74. "bl": "\xd4",
  75. "br": "\xbe",
  76. "ml": "\xc6",
  77. "mr": "\xb5",
  78. },
  79. # ╓──╖
  80. # ║ ║
  81. # ╟──╢
  82. # ╙──╜
  83. {
  84. "tl": "\xd6",
  85. "tr": "\xb7",
  86. "top": "\xc4",
  87. "side": "\xba",
  88. "bl": "\xd3",
  89. "br": "\xbd",
  90. "ml": "\xc7",
  91. "mr": "\xb6",
  92. },
  93. )
  94. def __init__(self, size: int, color="", style=1, start_nl=True):
  95. """ Construct box
  96. size is width of text inside the box.
  97. (so actual width is size + 2).
  98. color = Style of color to use.
  99. If color is blank, we don't use the RESET_ALL.
  100. style=0 Is double lines.
  101. start_nl Should we start the top with a newline?
  102. FUTURE: Support all line types.
  103. """
  104. self.size = size
  105. self.style = 0 # style
  106. self.color = color
  107. self.start_nl = start_nl
  108. # default colors
  109. # self.c = merge(Style.BRIGHT + Fore.WHITE + Back.BLUE)
  110. # self.cp = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  111. # useful consts
  112. self.r = Style.RESET_ALL
  113. self.nl = "\n\r"
  114. # self.bsb = "\b \b"
  115. def top(self):
  116. """ Output the TOP line. """
  117. if self.color == "":
  118. c = ""
  119. r = ""
  120. else:
  121. c = self.color
  122. r = self.r
  123. s = self.box_styles[self.style]
  124. if self.start_nl:
  125. n = self.nl
  126. else:
  127. n = ""
  128. return n + c + s["tl"] + s["top"] * self.size + s["tr"] + r + self.nl
  129. def middle(self):
  130. """ Output MIDDLE line.
  131. This solid line goes completely across.
  132. The edges connect to the top and bottom.
  133. """
  134. if self.color == "":
  135. c = ""
  136. r = ""
  137. else:
  138. c = self.color
  139. r = self.r
  140. s = self.box_styles[self.style]
  141. return c + s["ml"] + s["top"] * self.size + s["mr"] + r + self.nl
  142. def row(self, line):
  143. """ Output content line.
  144. NOTE: the line length must match the box size.
  145. Use "{0:30}".format(content) to match size of 30.
  146. """
  147. if self.color == "":
  148. c = ""
  149. r = ""
  150. else:
  151. c = self.color
  152. r = self.r
  153. s = self.box_styles[self.style]
  154. return c + "{0}{1}{2}".format(s["side"], r + line + c, s["side"]) + r + self.nl
  155. def bottom(self):
  156. """ Output BOTTOM line. """
  157. if self.color == "":
  158. c = ""
  159. r = ""
  160. else:
  161. c = self.color
  162. r = self.r
  163. s = self.box_styles[self.style]
  164. return c + s["bl"] + s["top"] * self.size + s["br"] + r + self.nl
  165. @staticmethod
  166. def alert(message, length=0, pad=2, width=78, base="red", style=1):
  167. """
  168. Display alert message
  169. Example:
  170. self.queue_game.put(Boxes.alert("Proxy done.", base="green"))
  171. length: the printable character length.
  172. If the string contains color codes, you'll need
  173. to use this so it will be centered properly.
  174. If not given, len(message) is used.
  175. pad: The spaces at the front and back of the message.
  176. width: Defaults to 78 chars wide.
  177. base: The background color to use for the alert.
  178. """
  179. if base == "red":
  180. color = merge(Style.BRIGHT + Fore.YELLOW + Back.RED)
  181. elif base == "blue":
  182. color = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  183. elif base == "green":
  184. color = merge(Style.BRIGHT + Fore.YELLOW + Back.GREEN)
  185. elif base == "cyan":
  186. color = merge(Style.BRIGHT + Fore.YELLOW + Back.CYAN)
  187. elif base == "magenta":
  188. color = merge(Style.BRIGHT + Fore.YELLOW + Back.MAGENTA)
  189. else:
  190. # I don't know.
  191. color = Style.RESET_ALL
  192. if length == 0:
  193. length = len(message)
  194. box_size = length + pad * 2
  195. boxpad = " " * pad
  196. box = Boxes(box_size, color=color, start_nl=False, style=style)
  197. msg = "{0}{1}{2}{3}".format(color, boxpad, message, boxpad)
  198. # How much pad to add to the left side?
  199. left_pad = (width - box_size + 2) // 2
  200. pad = " " * left_pad
  201. return "{0}{1}{2}{3}{4}{5}".format(
  202. pad, box.top(), pad, box.row(msg), pad, box.bottom()
  203. )