boxes.py 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  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. if base == "red":
  121. color = merge(Style.BRIGHT + Fore.YELLOW + Back.RED)
  122. elif base == "blue":
  123. color = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  124. elif base == "green":
  125. color = merge(Style.BRIGHT + Fore.YELLOW + Back.GREEN)
  126. elif base == "cyan":
  127. color = merge(Style.BRIGHT + Fore.YELLOW + Back.CYAN)
  128. elif base == "magenta":
  129. color = merge(Style.BRIGHT + Fore.YELLOW + Back.MAGENTA)
  130. else:
  131. # I don't know.
  132. color = Style.RESET_ALL
  133. if length == 0:
  134. length = len(message)
  135. box_size = length + pad * 2
  136. boxpad = " " * pad
  137. box = Boxes(box_size, color=color, start_nl=False)
  138. msg = "{0}{1}{2}{3}".format(color, boxpad, message, boxpad)
  139. # How much pad to add to the left side?
  140. left_pad = (width - box_size + 2) // 2
  141. pad = " " * left_pad
  142. return "{0}{1}{2}{3}{4}{5}".format(
  143. pad, box.top(), pad, box.row(msg), pad, box.bottom()
  144. )