Better card action handling.
[naja.git] / naja / widgets / info_area.py
1 """
2 Widget for the game board information area.
3 """
4 import pygame
5 import pygame.locals as pgl
6
7 from naja.constants import INFO_SIZE, EIGHT_BIT_SCALE, MOVE, ACT, KEYS
8 from naja.events import finish_event
9 from naja.resources import resources
10 from naja.resources.mutators import EIGHT_BIT
11
12 from naja.widgets.base import Widget
13 from naja.widgets.tile import BIT_MAP
14 from naja.widgets.text import TextBoxWidget, TextWidget
15
16
17 HINTS = {
18         MOVE: "Move using the arrow keys.\nPress SPACE to stay in place",
19         ACT: "Choose an action using the Up/Down keys.\n"
20              "Press Return to execute the action.",
21         }
22
23 TITLES = {
24         MOVE: "Move the Robot",
25         ACT: "Choose an Action",
26         }
27
28
29 class InfoAreaWidget(Widget):
30     """
31     Widget for the game board information area.
32     """
33     def __init__(self, pos, state):
34         super(InfoAreaWidget, self).__init__(pos, INFO_SIZE)
35         self.state = state
36         self.set_position(state.player.position)
37
38     def prepare(self):
39         self.set_position(self.state.player.position)
40         self.surface = pygame.surface.Surface(INFO_SIZE)
41         self.surface.fill((0, 0, 0))
42         # Extract actions and such from the card
43         title = TextWidget((0, 0), TITLES[self.state.gameboard.player_mode],
44                            colour=(255, 255, 255))
45         title.render(self.surface)
46         y_offset = title.surface.get_rect().height + 8
47         for choice, action in enumerate(self.card.actions):
48             if action.required_bits:
49                 img_name = BIT_MAP[action.required_bits].replace(
50                     '.png', '_small.png')
51                 img = resources.get_image(img_name,
52                                           transforms=(EIGHT_BIT,))
53                 self.surface.blit(img, (0, y_offset))
54                 y_offset += 8
55             text = TextBoxWidget(
56                 (12, y_offset), action.TEXT,
57                 box_width=(INFO_SIZE[0] - 12) // EIGHT_BIT_SCALE,
58                 fontsize=28)
59             text.render(self.surface)
60             # self.chosen may be None, in which case we don't draw the border.
61             if choice == self.chosen:
62                 colour = (255, 255, 0, 128)
63                 bottom = y_offset + text.surface.get_rect().height
64                 right = text.surface.get_rect().width + 12
65                 pygame.draw.lines(self.surface, colour, True,
66                                   [(12, y_offset), (right, y_offset),
67                                    (right, bottom), (12, bottom)], 4)
68             y_offset += text.surface.get_rect().height + 16
69         # We cheat horribly for layout reasons
70         hint = TextBoxWidget((0, 0), HINTS[self.state.gameboard.player_mode],
71                              box_width=INFO_SIZE[0] // EIGHT_BIT_SCALE)
72         hint.prepare()
73         y_offset = INFO_SIZE[1] - hint.surface.get_rect().height
74         self.surface.blit(hint.surface, (0, y_offset))
75
76     def set_position(self, position):
77         self.card = self.state.board_locations[position]
78         if self.state.gameboard.player_mode == ACT:
79             if self.chosen is None:
80                 self.chosen = 0
81         else:
82             self.chosen = None
83
84     def draw(self, surface):
85         surface.blit(self.surface, self.pos)
86
87     def handle_event(self, ev):
88         if self.state.gameboard.player_mode == MOVE:
89             return super(InfoAreaWidget, self).handle_event(ev)
90         if ev.type == pgl.KEYDOWN:
91             if ev.key in KEYS.SELECT:
92                 player = self.state.player
93                 action = self.card.actions[self.chosen]
94                 if not action.check_available(player):
95                     print "BEEP!"
96                 else:
97                     action.perform_action(self.state.gameboard, self.card)
98                     self.state.gameboard.replace_card(player.position)
99                     self.state.gameboard.change_mode()
100                 return finish_event()
101             if ev.key in KEYS.UP:
102                 if self.chosen > 0:
103                     self.chosen -= 1
104                 return finish_event()
105             if ev.key in KEYS.DOWN:
106                 if self.chosen + 1 < len(self.card.actions):
107                     self.chosen += 1
108                 return finish_event()
109         return super(InfoAreaWidget, self).handle_event(ev)