boxes.py 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  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):
  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. FUTURE: Support all line types.
  55. """
  56. self.size = size
  57. self.style = 0 # style
  58. self.color = color
  59. # default colors
  60. # self.c = merge(Style.BRIGHT + Fore.WHITE + Back.BLUE)
  61. # self.cp = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  62. # useful consts
  63. self.r = Style.RESET_ALL
  64. self.nl = "\n\r"
  65. # self.bsb = "\b \b"
  66. def top(self):
  67. """ Output the TOP line. """
  68. if self.color == "":
  69. c = ""
  70. r = ""
  71. else:
  72. c = self.color
  73. r = self.r
  74. s = self.box_styles[self.style]
  75. return self.nl + c + s["tl"] + s["top"] * self.size + s["tr"] + r + self.nl
  76. def middle(self):
  77. """ Output MIDDLE line.
  78. This solid line goes completely across.
  79. The edges connect to the top and bottom.
  80. """
  81. if self.color == "":
  82. c = ""
  83. r = ""
  84. else:
  85. c = self.color
  86. r = self.r
  87. s = self.box_styles[self.style]
  88. return c + s["ml"] + s["top"] * self.size + s["mr"] + r + self.nl
  89. def row(self, line):
  90. """ Output content line.
  91. NOTE: the line length must match the box size.
  92. Use "{0:30}".format(content) to match size of 30.
  93. """
  94. if self.color == "":
  95. c = ""
  96. r = ""
  97. else:
  98. c = self.color
  99. r = self.r
  100. s = self.box_styles[self.style]
  101. return c + "{0}{1}{2}".format(s["side"], r + line + c, s["side"]) + r + self.nl
  102. def bottom(self):
  103. """ Output BOTTOM line. """
  104. if self.color == "":
  105. c = ""
  106. r = ""
  107. else:
  108. c = self.color
  109. r = self.r
  110. s = self.box_styles[self.style]
  111. return c + s["bl"] + s["top"] * self.size + s["br"] + r + self.nl