+ def draw(self, surface):
+ surface.blit(self.surface, self.pos)
+
+
+class TextBoxWidget(TextWidget):
+ def __init__(self, *args, **kwargs):
+ self.padding = kwargs.pop('padding', 5)
+ self.border = kwargs.pop('border', 2)
+ self.bg_colour = convert_colour(kwargs.pop('bg_colour',
+ (255, 255, 255, 192)))
+ self.border_colour = convert_colour(kwargs.pop('border_colour',
+ (0, 0, 0)))
+ self.box_width = kwargs.pop('box_width', 0)
+
+ super(TextBoxWidget, self).__init__(*args, **kwargs)
+
+ def lines(self, image_map):
+ if self.box_width != 0:
+ return self._wrapped_lines(image_map)
+ else:
+ return self.text.splitlines()
+
+ def _wrapped_lines(self, image_map):
+ def words_fit(words):
+ words_line = ' '.join(words)
+ width = self.font.size(words_line)[0]
+ if width < self.box_width:
+ return True
+ elif len(words) == 1:
+ Exception("Word %r too long for box." % (words[0],))
+ return False
+
+ line_count = 0
+ for line in self.text.splitlines():
+ current_words = []
+ for word in line.split():
+ suffix = ''
+ if word[-1] in '.,':
+ suffix = word[-1]
+ word = word[:-1]
+ markup_data = MARKUP_MAP.get(word, None)
+ if markup_data is not None:
+ word = ' ' * markup_data[0]
+ word += suffix
+ current_words.append(word)
+ if words_fit(current_words):
+ if markup_data is not None:
+ size = self.font.size(
+ ' '.join(current_words[:-1]) + ' ')
+ pos = (size[0] * EIGHT_BIT_SCALE,
+ size[1] * line_count * EIGHT_BIT_SCALE)
+ pos = (pos[0] + self.padding, pos[1] + self.padding)
+ image_map[pos] = resources.get_image(
+ markup_data[1], transforms=(EIGHT_BIT,))
+ continue
+ else:
+ line_count += 1
+ yield ' '.join(current_words[:-1])
+ current_words = current_words[-1:]
+ if current_words and words_fit(current_words):
+ yield ' '.join(current_words)
+
+ def prepare(self):
+ self.font = resources.get_font(self.fontname, self.fontsize)
+ image_map = {}
+ rendered_lines = []
+ width, height = self.padding * 2, self.padding * 2
+ for line in self.lines(image_map):
+ line_surface = self.render_line(line)
+ line_rect = line_surface.get_rect()
+ rendered_lines.append(line_surface)
+ width = max(width, line_rect.width + self.padding * 2)
+ height += line_rect.height
+
+ self.surface = pygame.surface.Surface((width, height),
+ pygame.locals.SRCALPHA)
+ self.surface.fill(self.bg_colour)
+ self.size = self.surface.get_rect().size
+
+ x, y = self.padding, self.padding
+ for line_surface in rendered_lines:
+ self.surface.blit(line_surface, (x, y))
+ y += line_surface.get_rect().height
+ for pos, img in image_map.items():
+ self.surface.blit(img, pos)
+