| # Module 'Buttons' |
| |
| |
| # Import module 'rect' renamed as '_rect' to avoid exporting it on |
| # 'from Buttons import *' |
| # |
| import rect |
| _rect = rect |
| del rect |
| |
| |
| # Field indices in mouse event detail |
| # |
| _HV = 0 |
| _CLICKS = 1 |
| _BUTTON = 2 |
| _MASK = 3 |
| |
| |
| # LabelAppearance provides defaults for all appearance methods. |
| # selected state not visible |
| # disabled --> crossed out |
| # hilited --> inverted |
| # |
| class LabelAppearance: |
| # |
| # Initialization |
| # |
| def init_appearance(self): |
| self.bounds = _rect.empty |
| self.enabled = 1 |
| self.hilited = 0 |
| self.selected = 0 |
| self.text = '' |
| # |
| # Size enquiry |
| # |
| def getminsize(self, (m, (width, height))): |
| width = max(width, m.textwidth(self.text) + 6) |
| height = max(height, m.lineheight() + 6) |
| return width, height |
| # |
| def getbounds(self): |
| return self.bounds |
| # |
| # Changing the parameters |
| # |
| def settext(self, text): |
| self.text = text |
| if self.bounds <> _rect.empty: |
| self.recalctextpos() |
| self.redraw() |
| # |
| def setbounds(self, bounds): |
| self.bounds = bounds |
| if self.bounds <> _rect.empty: |
| self.recalc() |
| # |
| def realize(self): |
| pass |
| # |
| # Changing the state bits |
| # |
| def enable(self, flag): |
| if flag <> self.enabled: |
| self.enabled = flag |
| if self.bounds <> _rect.empty: |
| self.flipenable(self.parent.begindrawing()) |
| # |
| def hilite(self, flag): |
| if flag <> self.hilited: |
| self.hilited = flag |
| if self.bounds <> _rect.empty: |
| self.fliphilite(self.parent.begindrawing()) |
| # |
| def select(self, flag): |
| if flag <> self.selected: |
| self.selected = flag |
| if self.bounds <> _rect.empty: |
| self.redraw() |
| # |
| # Recalculate the box bounds and text position. |
| # This can be overridden by buttons that draw different boxes |
| # or want their text in a different position. |
| # |
| def recalc(self): |
| if self.bounds <> _rect.empty: |
| self.recalcbounds() |
| self.recalctextpos() |
| # |
| def recalcbounds(self): |
| self.hilitebounds = _rect.inset(self.bounds, (3, 3)) |
| self.crossbounds = self.bounds |
| # |
| def recalctextpos(self): |
| (left, top), (right, bottom) = self.bounds |
| m = self.parent.beginmeasuring() |
| h = (left + right - m.textwidth(self.text)) / 2 |
| v = (top + bottom - m.lineheight()) / 2 |
| self.textpos = h, v |
| # |
| # Generic drawing interface. |
| # Do not override redraw() or draw() methods; override drawit() c.s. |
| # |
| def redraw(self): |
| if self.bounds <> _rect.empty: |
| d = self.parent.begindrawing() |
| d.erase(self.bounds) |
| self.draw(d, self.bounds) |
| # |
| def draw(self, (d, area)): |
| area = _rect.intersect(area, self.bounds) |
| if area == _rect.empty: |
| return |
| d.cliprect(area) |
| self.drawit(d) |
| d.noclip() |
| # |
| # The drawit() method is fairly generic but may be overridden. |
| # |
| def drawit(self, d): |
| self.drawpict(d) |
| if self.text: |
| d.text(self.textpos, self.text) |
| if not self.enabled: |
| self.flipenable(d) |
| if self.hilited: |
| self.fliphilite(d) |
| # |
| # Default drawing detail functions. |
| # Overriding these is normally sufficient to get different |
| # appearances. |
| # |
| def drawpict(self, d): |
| pass |
| # |
| def flipenable(self, d): |
| _xorcross(d, self.crossbounds) |
| # |
| def fliphilite(self, d): |
| d.invert(self.hilitebounds) |
| |
| |
| # A Strut is a label with no width of its own. |
| |
| class StrutAppearance(LabelAppearance): |
| # |
| def getminsize(self, (m, (width, height))): |
| height = max(height, m.lineheight() + 6) |
| return width, height |
| # |
| |
| |
| # ButtonAppearance displays a centered string in a box. |
| # selected --> bold border |
| # disabled --> crossed out |
| # hilited --> inverted |
| # |
| class ButtonAppearance(LabelAppearance): |
| # |
| def drawpict(self, d): |
| d.box(_rect.inset(self.bounds, (1, 1))) |
| if self.selected: |
| # Make a thicker box |
| d.box(self.bounds) |
| d.box(_rect.inset(self.bounds, (2, 2))) |
| d.box(_rect.inset(self.bounds, (3, 3))) |
| # |
| |
| |
| # CheckAppearance displays a small square box and a left-justified string. |
| # selected --> a cross appears in the box |
| # disabled --> whole button crossed out |
| # hilited --> box is inverted |
| # |
| class CheckAppearance(LabelAppearance): |
| # |
| def getminsize(self, (m, (width, height))): |
| minwidth = m.textwidth(self.text) + 6 |
| minheight = m.lineheight() + 6 |
| width = max(width, minwidth + minheight + m.textwidth(' ')) |
| height = max(height, minheight) |
| return width, height |
| # |
| def drawpict(self, d): |
| d.box(self.boxbounds) |
| if self.selected: _xorcross(d, self.boxbounds) |
| # |
| def recalcbounds(self): |
| LabelAppearance.recalcbounds(self) |
| (left, top), (right, bottom) = self.bounds |
| self.size = bottom - top - 4 |
| self.boxbounds = (left+2, top+2), (left+2+self.size, bottom-2) |
| self.hilitebounds = self.boxbounds |
| # |
| def recalctextpos(self): |
| m = self.parent.beginmeasuring() |
| (left, top), (right, bottom) = self.boxbounds |
| h = right + m.textwidth(' ') |
| v = top + (self.size - m.lineheight()) / 2 |
| self.textpos = h, v |
| # |
| |
| |
| # RadioAppearance displays a round indicator and a left-justified string. |
| # selected --> a dot appears in the indicator |
| # disabled --> whole button crossed out |
| # hilited --> indicator is inverted |
| # |
| class RadioAppearance(CheckAppearance): |
| # |
| def drawpict(self, d): |
| (left, top), (right, bottom) = self.boxbounds |
| radius = self.size / 2 |
| center = left + radius, top + radius |
| d.circle(center, radius) |
| if self.selected: |
| d.fillcircle(center, radius*3/5) |
| # |
| |
| |
| # NoReactivity ignores mouse events. |
| # |
| class NoReactivity: |
| def init_reactivity(self): pass |
| |
| |
| # BaseReactivity defines hooks and asks for mouse events, |
| # but provides only dummy mouse event handlers. |
| # The trigger methods call the corresponding hooks set by the user. |
| # Hooks (and triggers) mean the following: |
| # down_hook called on some mouse-down events |
| # move_hook called on some mouse-move events |
| # up_hook called on mouse-up events |
| # on_hook called for buttons with on/off state, when it goes on |
| # hook called when a button 'fires' or a radiobutton goes on |
| # There are usually extra conditions, e.g., hooks are only called |
| # when the button is enabled, or active, or selected (on). |
| # |
| class BaseReactivity: |
| # |
| def init_reactivity(self): |
| self.down_hook = self.move_hook = self.up_hook = \ |
| self.on_hook = self.off_hook = \ |
| self.hook = self.active = 0 |
| self.parent.need_mouse(self) |
| # |
| def mousetest(self, hv): |
| return _rect.pointinrect(hv, self.bounds) |
| # |
| def mouse_down(self, detail): |
| pass |
| # |
| def mouse_move(self, detail): |
| pass |
| # |
| def mouse_up(self, detail): |
| pass |
| # |
| def down_trigger(self): |
| if self.down_hook: self.down_hook(self) |
| # |
| def move_trigger(self): |
| if self.move_hook: self.move_hook(self) |
| # |
| def up_trigger(self): |
| if self.up_hook: self.up_hook(self) |
| # |
| def on_trigger(self): |
| if self.on_hook: self.on_hook(self) |
| # |
| def off_trigger(self): |
| if self.off_hook: self.off_hook(self) |
| # |
| def trigger(self): |
| if self.hook: self.hook(self) |
| |
| |
| # ToggleReactivity acts like a simple pushbutton. |
| # It toggles its hilite state on mouse down events. |
| # |
| class ToggleReactivity(BaseReactivity): |
| # |
| def mouse_down(self, detail): |
| if self.enabled and self.mousetest(detail[_HV]): |
| self.active = 1 |
| self.hilite(not self.hilited) |
| self.down_trigger() |
| # |
| def mouse_move(self, detail): |
| if self.active: |
| self.move_trigger() |
| # |
| def mouse_up(self, detail): |
| if self.active: |
| self.up_trigger() |
| self.active = 0 |
| # |
| def down_trigger(self): |
| if self.hilited: |
| self.on_trigger() |
| else: |
| self.off_trigger() |
| self.trigger() |
| # |
| |
| |
| # TriggerReactivity acts like a fancy pushbutton. |
| # It hilites itself while the mouse is down within its bounds. |
| # |
| class TriggerReactivity(BaseReactivity): |
| # |
| def mouse_down(self, detail): |
| if self.enabled and self.mousetest(detail[_HV]): |
| self.active = 1 |
| self.hilite(1) |
| self.down_trigger() |
| # |
| def mouse_move(self, detail): |
| if self.active: |
| self.hilite(self.mousetest(detail[_HV])) |
| if self.hilited: |
| self.move_trigger() |
| # |
| def mouse_up(self, detail): |
| if self.active: |
| self.hilite(self.mousetest(detail[_HV])) |
| if self.hilited: |
| self.up_trigger() |
| self.trigger() |
| self.active = 0 |
| self.hilite(0) |
| # |
| |
| |
| # CheckReactivity handles mouse events like TriggerReactivity, |
| # It overrides the up_trigger method to flip its selected state. |
| # |
| class CheckReactivity(TriggerReactivity): |
| # |
| def up_trigger(self): |
| self.select(not self.selected) |
| if self.selected: |
| self.on_trigger() |
| else: |
| self.off_trigger() |
| self.trigger() |
| |
| |
| # RadioReactivity turns itself on and the other buttons in its group |
| # off when its up_trigger method is called. |
| # |
| class RadioReactivity(TriggerReactivity): |
| # |
| def init_reactivity(self): |
| TriggerReactivity.init_reactivity(self) |
| self.group = [] |
| # |
| def up_trigger(self): |
| for b in self.group: |
| if b <> self: |
| if b.selected: |
| b.select(0) |
| b.off_trigger() |
| self.select(1) |
| self.on_trigger() |
| self.trigger() |
| |
| |
| # Auxiliary class for 'define' method. |
| # Call the initializers in the right order. |
| # |
| class Define: |
| # |
| def define(self, parent): |
| self.parent = parent |
| parent.addchild(self) |
| self.init_appearance() |
| self.init_reactivity() |
| return self |
| # |
| def destroy(self): |
| self.parent = 0 |
| # |
| def definetext(self, (parent, text)): |
| self = self.define(parent) |
| self.settext(text) |
| return self |
| |
| |
| # Subroutine to cross out a rectangle. |
| # |
| def _xorcross(d, bounds): |
| ((left, top), (right, bottom)) = bounds |
| # This is s bit funny to make it look better |
| left = left + 2 |
| right = right - 2 |
| top = top + 2 |
| bottom = bottom - 3 |
| d.xorline(((left, top), (right, bottom))) |
| d.xorline((left, bottom), (right, top)) |
| |
| |
| # Ready-made button classes. |
| # |
| class Label(NoReactivity, LabelAppearance, Define): pass |
| class Strut(NoReactivity, StrutAppearance, Define): pass |
| class PushButton(TriggerReactivity, ButtonAppearance, Define): pass |
| class CheckButton(CheckReactivity, CheckAppearance, Define): pass |
| class RadioButton(RadioReactivity, RadioAppearance, Define): pass |
| class ToggleButton(ToggleReactivity, ButtonAppearance, Define): pass |