boxes.py 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. from colorama import Fore, Back, Style
  2. def merge(color_string):
  3. """ Given a string of colorama ANSI, merge them if you can. """
  4. return color_string.replace("m\x1b[", ";")
  5. class Boxes(object):
  6. """ Boxes makes nice ASCII graphic menus
  7. Example:
  8. c1 = merge(Style.BRIGHT + Fore.CYAN)
  9. c2 = merge(Style.NORMAL + Fore.CYAN)
  10. # Set the size of the box to 40 (Important!)
  11. box = Boxes(40, color=c1)
  12. self.queue_game.put(box.top())
  13. # format string "{0:^40}".format("Text Centered")
  14. # This centers the text into the size of the box.
  15. self.queue_game.put(box.row(c1 + "{0:^40}".format("Scripts")))
  16. # this draws the middle line. (The line that goes across, but is
  17. # connected on both sides to the one above it).
  18. self.queue_game.put(box.middle())
  19. def menu_item(ch, desc):
  20. # Again we want the final string to be 40!
  21. # c1 and c2 (below) take 0 space.
  22. # We have " " + ch + " " + "-" + " " .. which equals 5
  23. # So desc needs {:35} to pad out string out to 40 print chars.
  24. row = " {0}{1} {2}-{3} {4:35}".format(c1, ch, c2, c1, desc)
  25. # self.queue_game.put(
  26. # " " + c1 + ch + c2 + " - " + c1 + desc + self.nl
  27. # )
  28. self.queue_game.put(box.row(row))
  29. menu_item("1", "Ports (Trades between two sectors)")
  30. menu_item("2", "Explore (Strange new sectors)")
  31. menu_item("3", "Space... the final frontier...")
  32. menu_item("X", "eXit")
  33. self.queue_game.put(box.bottom())
  34. """
  35. box_styles = (
  36. {
  37. "tl": "\xc9",
  38. "tr": "\xbb",
  39. "top": "\xcd",
  40. "side": "\xba",
  41. "bl": "\xc8",
  42. "br": "\xbc",
  43. "ml": "\xcc",
  44. "mr": "\xb9",
  45. },
  46. )
  47. def __init__(self, size: int, color="", style=0, start_nl=True):
  48. """ Construct box
  49. size is width of text inside the box.
  50. (so actual width is size + 2).
  51. color = Style of color to use.
  52. If color is blank, we don't use the RESET_ALL.
  53. style=0 Is double lines.
  54. start_nl Should we start the top with a newline?
  55. FUTURE: Support all line types.
  56. """
  57. self.size = size
  58. self.style = 0 # style
  59. self.color = color
  60. self.start_nl = start_nl
  61. # default colors
  62. # self.c = merge(Style.BRIGHT + Fore.WHITE + Back.BLUE)
  63. # self.cp = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  64. # useful consts
  65. self.r = Style.RESET_ALL
  66. self.nl = "\n\r"
  67. # self.bsb = "\b \b"
  68. def top(self):
  69. """ Output the TOP line. """
  70. if self.color == "":
  71. c = ""
  72. r = ""
  73. else:
  74. c = self.color
  75. r = self.r
  76. s = self.box_styles[self.style]
  77. if self.start_nl:
  78. n = self.nl
  79. else:
  80. n = ""
  81. return n + c + s["tl"] + s["top"] * self.size + s["tr"] + r + self.nl
  82. def middle(self):
  83. """ Output MIDDLE line.
  84. This solid line goes completely across.
  85. The edges connect to the top and bottom.
  86. """
  87. if self.color == "":
  88. c = ""
  89. r = ""
  90. else:
  91. c = self.color
  92. r = self.r
  93. s = self.box_styles[self.style]
  94. return c + s["ml"] + s["top"] * self.size + s["mr"] + r + self.nl
  95. def row(self, line):
  96. """ Output content line.
  97. NOTE: the line length must match the box size.
  98. Use "{0:30}".format(content) to match size of 30.
  99. """
  100. if self.color == "":
  101. c = ""
  102. r = ""
  103. else:
  104. c = self.color
  105. r = self.r
  106. s = self.box_styles[self.style]
  107. return c + "{0}{1}{2}".format(s["side"], r + line + c, s["side"]) + r + self.nl
  108. def bottom(self):
  109. """ Output BOTTOM line. """
  110. if self.color == "":
  111. c = ""
  112. r = ""
  113. else:
  114. c = self.color
  115. r = self.r
  116. s = self.box_styles[self.style]
  117. return c + s["bl"] + s["top"] * self.size + s["br"] + r + self.nl
  118. @staticmethod
  119. def alert(message, length=0, pad=2, width=78, base="red"):
  120. """
  121. Display alert message
  122. Example:
  123. self.queue_game.put(Boxes.alert("Proxy done.", base="green"))
  124. length: the printable character length.
  125. If the string contains color codes, you'll need
  126. to use this so it will be centered properly.
  127. If not given, len(message) is used.
  128. pad: The spaces at the front and back of the message.
  129. width: Defaults to 78 chars wide.
  130. base: The background color to use for the alert.
  131. """
  132. if base == "red":
  133. color = merge(Style.BRIGHT + Fore.YELLOW + Back.RED)
  134. elif base == "blue":
  135. color = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  136. elif base == "green":
  137. color = merge(Style.BRIGHT + Fore.YELLOW + Back.GREEN)
  138. elif base == "cyan":
  139. color = merge(Style.BRIGHT + Fore.YELLOW + Back.CYAN)
  140. elif base == "magenta":
  141. color = merge(Style.BRIGHT + Fore.YELLOW + Back.MAGENTA)
  142. else:
  143. # I don't know.
  144. color = Style.RESET_ALL
  145. if length == 0:
  146. length = len(message)
  147. box_size = length + pad * 2
  148. boxpad = " " * pad
  149. box = Boxes(box_size, color=color, start_nl=False)
  150. msg = "{0}{1}{2}{3}".format(color, boxpad, message, boxpad)
  151. # How much pad to add to the left side?
  152. left_pad = (width - box_size + 2) // 2
  153. pad = " " * left_pad
  154. return "{0}{1}{2}{3}{4}{5}".format(
  155. pad, box.top(), pad, box.row(msg), pad, box.bottom()
  156. )