// File: ftogglebutton.cpp // Provides: class FToggleButton #include "fapp.h" #include "fbuttongroup.h" #include "fstatusbar.h" #include "ftogglebutton.h" //---------------------------------------------------------------------- // class FToggleButton //---------------------------------------------------------------------- // constructor and destructor //---------------------------------------------------------------------- FToggleButton::FToggleButton(FWidget* parent) : FWidget(parent) , focus_inside_group(true) , text() , checked(false) , label_offset_pos(0) , button_width(0) , button_group(0) { init(); if ( parent && strcmp ( parent->getClassName() , const_cast("FButtonGroup") ) == 0 ) { setGroup( static_cast(parent) ); group()->insert(this); // insert into button group } } //---------------------------------------------------------------------- FToggleButton::FToggleButton ( const FString& txt, FWidget* parent ) : FWidget(parent) , focus_inside_group(true) , text() , checked(false) , label_offset_pos(0) , button_width(0) , button_group(0) { init(); setText(txt); if ( parent && strcmp ( parent->getClassName() , const_cast("FButtonGroup") ) == 0 ) { setGroup( static_cast(parent) ); group()->insert( this ); // insert into button group } } //---------------------------------------------------------------------- FToggleButton::~FToggleButton() // destructor { delAccelerator(this); if ( group() ) group()->remove(this); if ( hasFocus() ) hideCursor(); } // private methods of FToggleButton //---------------------------------------------------------------------- void FToggleButton::init() { setGeometry (1, 1, 4, 1, false); // initialize geometry values if ( hasFocus() ) flags = FOCUS; if ( isEnabled() ) { flags |= ACTIVE; if ( hasFocus() ) { foregroundColor = wc.toggle_button_active_focus_fg; backgroundColor = wc.toggle_button_active_focus_bg; } else { foregroundColor = wc.toggle_button_active_fg; backgroundColor = wc.toggle_button_active_bg; } } else // inactive { foregroundColor = wc.label_inactive_fg; backgroundColor = wc.label_inactive_bg; } } //---------------------------------------------------------------------- void FToggleButton::setGroup (FButtonGroup* btngroup) { button_group = btngroup; } // protected methods of FToggleButton //---------------------------------------------------------------------- uChar FToggleButton::getHotkey() { uInt length; if ( text.isEmpty() ) return 0; length = text.getLength(); for (uInt i=0; i < length; i++) if ( (i+1 < length) && (text[i] == '&') ) return uChar(text[++i]); return 0; } //---------------------------------------------------------------------- void FToggleButton::setHotkeyAccelerator() { int hotkey = getHotkey(); if ( hotkey ) { if ( isalpha(hotkey) || isdigit(hotkey) ) { addAccelerator (tolower(hotkey), this); addAccelerator (toupper(hotkey), this); // Meta + hotkey addAccelerator (0x20000e0 + tolower(hotkey), this); } else addAccelerator (getHotkey(), this); } else delAccelerator(this); } //---------------------------------------------------------------------- void FToggleButton::draw() { bool isFocus = ((flags & FOCUS) != 0); if ( isFocus && statusBar() ) { FString msg = getStatusbarMessage(); FString curMsg = statusBar()->getMessage(); if ( curMsg != msg ) { statusBar()->setMessage(msg); statusBar()->drawMessage(); } } // set the cursor to the button if ( isRadioButton() || isCheckboxButton() ) { setCursorPos (xpos+xmin, ypos+ymin-1); if ( isCursorInside() && hasFocus() ) { if ( isHiddenCursor() ) showCursor(); } else if ( ! isHiddenCursor() ) hideCursor(); } } //---------------------------------------------------------------------- void FToggleButton::drawLabel() { uInt length, i; int hotkeypos; FString txt; wchar_t* LabelText; register wchar_t* src; register wchar_t* dest; bool isActive, isNoUnderline; if ( ! isVisible() ) return; if ( text.isNull() || text.isEmpty() ) return; length = text.getLength(); hotkeypos = -1; LabelText = new wchar_t[length+1]; txt = text; src = const_cast(txt.wc_str()); dest = const_cast(LabelText); isActive = ((flags & ACTIVE) != 0); isNoUnderline = ((flags & NO_UNDERLINE) != 0); // find hotkey position in string // + generate a new string without the '&'-sign for (i=0; i < length; i++) { if ( (i < length) && (txt[i] == '&') && (hotkeypos == -1) ) { hotkeypos = int(i); i++; src++; } *dest++ = *src++; } if ( hotkeypos != -1 ) length--; gotoxy (xpos+xmin-1+label_offset_pos, ypos+ymin-1); if ( isEnabled() ) setColor(wc.label_fg, wc.label_bg); else setColor(wc.label_inactive_fg, wc.label_inactive_bg); for (int z=0; z < int(length); z++) { if ( (z == hotkeypos) && isActive ) { setColor (wc.label_hotkey_fg, wc.label_hotkey_bg); if ( ! isNoUnderline ) setUnderline(); print ( LabelText[z] ); if ( ! isNoUnderline ) unsetUnderline(); setColor (wc.label_fg, wc.label_bg); } else print ( LabelText[z] ); } delete[] LabelText; } //---------------------------------------------------------------------- void FToggleButton::processClick() { emitCallback("clicked"); } //---------------------------------------------------------------------- void FToggleButton::processToggle() { emitCallback("toggled"); } //---------------------------------------------------------------------- FButtonGroup* FToggleButton::group() const { return button_group; } //---------------------------------------------------------------------- bool FToggleButton::isRadioButton() const { return ( strcmp ( getClassName() , const_cast("FRadioButton") ) == 0 ); } //---------------------------------------------------------------------- bool FToggleButton::isCheckboxButton() const { return ( strcmp ( getClassName() , const_cast("FCheckBox") ) == 0 ); } //---------------------------------------------------------------------- void FToggleButton::onKeyPress (FKeyEvent* ev) { int key; if ( ! isEnabled() ) return; key = ev->key(); switch ( key ) { case fc::Fkey_return: case fc::Fkey_enter: case fc::Fkey_space: if ( isRadioButton() ) { if ( ! checked ) { checked = true; processToggle(); } } else { checked = not checked; processToggle(); } processClick(); ev->accept(); break; case fc::Fkey_down: case fc::Fkey_right: focus_inside_group = true; focusNextChild(); ev->accept(); break; case fc::Fkey_up: case fc::Fkey_left: focus_inside_group = true; focusPrevChild(); ev->accept(); break; default: break; } if ( ev->isAccepted() ) { draw(); updateTerminal(); } } // public methods of FToggleButton //---------------------------------------------------------------------- void FToggleButton::hide() { int fg, bg, size; char* blank; FWidget::hide(); fg = parentWidget()->getForegroundColor(); bg = parentWidget()->getBackgroundColor(); setColor (fg, bg); size = width; blank = new char[size+1]; memset(blank, ' ', uLong(size)); blank[size] = '\0'; gotoxy (xpos+xmin-1, ypos+ymin-1); print (blank); delete[] blank; } //---------------------------------------------------------------------- void FToggleButton::setGeometry (int x, int y, int w, int h, bool adjust) { int min_width = button_width + int(text.getLength()); if ( w < min_width ) w = min_width; FWidget::setGeometry(x, y, w, h, adjust); } //---------------------------------------------------------------------- bool FToggleButton::setNoUnderline(bool on) { if ( on ) flags |= NO_UNDERLINE; else flags &= ~NO_UNDERLINE; return on; } //---------------------------------------------------------------------- bool FToggleButton::setEnable(bool on) { FWidget::setEnable(on); if ( on ) { flags |= ACTIVE; setHotkeyAccelerator(); if ( hasFocus() ) { foregroundColor = wc.toggle_button_active_focus_fg; backgroundColor = wc.toggle_button_active_focus_bg; } else { foregroundColor = wc.toggle_button_active_fg; backgroundColor = wc.toggle_button_active_bg; } } else { flags &= ~ACTIVE; delAccelerator (this); foregroundColor = wc.toggle_button_inactive_fg; backgroundColor = wc.toggle_button_inactive_bg; } return on; } //---------------------------------------------------------------------- bool FToggleButton::setFocus(bool on) { FWidget::setFocus(on); if ( on ) { flags |= FOCUS; if ( isEnabled() ) { if ( isRadioButton() ) focus_inside_group = false; foregroundColor = wc.toggle_button_active_focus_fg; backgroundColor = wc.toggle_button_active_focus_bg; if ( isCursorInside() && (isRadioButton() || isCheckboxButton()) ) showCursor(); if ( statusBar() ) { FString msg = getStatusbarMessage(); FString curMsg = statusBar()->getMessage(); if ( curMsg != msg ) statusBar()->setMessage(msg); } } } else { flags &= ~FOCUS; if ( isEnabled() ) { foregroundColor = wc.toggle_button_active_fg; backgroundColor = wc.toggle_button_active_bg; hideCursor(); if ( statusBar() ) statusBar()->clearMessage(); } } return on; } //---------------------------------------------------------------------- void FToggleButton::onMouseDown (FMouseEvent* ev) { if ( ev->getButton() != LeftButton ) return; if ( ! hasFocus() ) { FWidget* focused_widget = getFocusWidget(); FFocusEvent out (FocusOut_Event); FApplication::queueEvent(focused_widget, &out); setFocus(); if ( focused_widget ) focused_widget->redraw(); redraw(); if ( statusBar() ) { statusBar()->drawMessage(); updateTerminal(); flush_out(); } } } //---------------------------------------------------------------------- void FToggleButton::onMouseUp (FMouseEvent* ev) { if ( ev->getButton() != LeftButton ) return; if ( getGeometryGlobal().contains(ev->getGlobalPos()) ) { if ( isRadioButton() ) { if ( ! checked ) { checked = true; processToggle(); } } else { checked = not checked; processToggle(); } redraw(); processClick(); } } //---------------------------------------------------------------------- void FToggleButton::onAccel (FAccelEvent* ev) { if ( isEnabled() ) { if ( ! hasFocus() ) { FWidget* focused_widget = static_cast(ev->focusedWidget()); FFocusEvent out (FocusOut_Event); FApplication::queueEvent(focused_widget, &out); setFocus(); if ( focused_widget ) focused_widget->redraw(); } if ( isRadioButton() ) { if ( ! checked ) { checked = true; processToggle(); } } else { checked = not checked; processToggle(); } redraw(); if ( statusBar() ) { statusBar()->drawMessage(); updateTerminal(); flush_out(); } processClick(); ev->accept(); } } //---------------------------------------------------------------------- void FToggleButton::onHide (FHideEvent*) { if ( hasFocus() ) hideCursor(); } //---------------------------------------------------------------------- void FToggleButton::onFocusIn (FFocusEvent*) { if ( isCursorInside() && (isRadioButton() || isCheckboxButton()) ) showCursor(); if ( statusBar() ) statusBar()->drawMessage(); } //---------------------------------------------------------------------- void FToggleButton::onFocusOut (FFocusEvent* out_ev) { if ( statusBar() ) { statusBar()->clearMessage(); statusBar()->drawMessage(); } hideCursor(); if ( group() ) { if ( ! focus_inside_group && isRadioButton() ) { focus_inside_group = true; out_ev->ignore(); if ( out_ev->getFocusType() == FocusNextWidget ) group()->focusNextChild(); if ( out_ev->getFocusType() == FocusPreviousWidget ) group()->focusPrevChild(); redraw(); } else if ( this == group()->getLastButton() && out_ev->getFocusType() == FocusNextWidget ) { out_ev->ignore(); group()->focusNextChild(); redraw(); } else if ( this == group()->getFirstButton() && out_ev->getFocusType() == FocusPreviousWidget ) { out_ev->ignore(); group()->focusPrevChild(); redraw(); } } } //---------------------------------------------------------------------- bool FToggleButton::setChecked(bool on) { if ( checked != on ) { checked = on; processToggle(); } return checked; } //---------------------------------------------------------------------- void FToggleButton::setText (FString txt) { text = txt; setWidth(button_width + int(text.getLength())); if ( isEnabled() ) { delAccelerator (this); setHotkeyAccelerator(); } }