finalcut/src/ftogglebutton.cpp

632 lines
14 KiB
C++
Raw Normal View History

// File: ftogglebutton.cpp
// Provides: class FToggleButton
2015-05-23 13:35:12 +02:00
#include "fapp.h"
#include "fbuttongroup.h"
#include "fstatusbar.h"
#include "ftogglebutton.h"
//----------------------------------------------------------------------
// class FToggleButton
//----------------------------------------------------------------------
// constructor and destructor
//----------------------------------------------------------------------
2015-09-22 04:18:20 +02:00
FToggleButton::FToggleButton(FWidget* parent)
: FWidget(parent)
, focus_inside_group(true)
, text()
, checked(false)
, label_offset_pos(0)
, button_width(0)
, button_group(0)
2015-05-23 13:35:12 +02:00
{
2015-09-22 04:18:20 +02:00
init();
2015-05-23 13:35:12 +02:00
2015-09-22 04:18:20 +02:00
if ( parent && strcmp ( parent->getClassName()
, const_cast<char*>("FButtonGroup") ) == 0 )
2015-05-23 13:35:12 +02:00
{
setGroup( static_cast<FButtonGroup*>(parent) );
group()->insert(this); // insert into button group
}
}
//----------------------------------------------------------------------
2015-12-21 18:48:38 +01:00
FToggleButton::FToggleButton (const FString& txt, FWidget* parent)
2015-09-22 04:18:20 +02:00
: FWidget(parent)
, focus_inside_group(true)
, text()
, checked(false)
, label_offset_pos(0)
, button_width(0)
, button_group(0)
2015-05-23 13:35:12 +02:00
{
2015-09-22 04:18:20 +02:00
init();
2015-05-23 13:35:12 +02:00
setText(txt);
2015-09-22 04:18:20 +02:00
if ( parent && strcmp ( parent->getClassName()
, const_cast<char*>("FButtonGroup") ) == 0 )
2015-05-23 13:35:12 +02:00
{
setGroup( static_cast<FButtonGroup*>(parent) );
group()->insert( this ); // insert into button group
}
}
//----------------------------------------------------------------------
FToggleButton::~FToggleButton() // destructor
{
2015-11-12 01:33:16 +01:00
delAccelerator();
2015-05-23 13:35:12 +02:00
if ( group() )
group()->remove(this);
if ( hasFocus() )
hideCursor();
}
// private methods of FToggleButton
//----------------------------------------------------------------------
void FToggleButton::init()
{
2015-07-04 22:51:47 +02:00
setGeometry (1, 1, 4, 1, false); // initialize geometry values
2015-05-23 13:35:12 +02:00
if ( hasFocus() )
2015-09-22 04:18:20 +02:00
flags = FOCUS;
2015-05-23 13:35:12 +02:00
if ( isEnabled() )
{
2015-09-22 04:18:20 +02:00
flags |= ACTIVE;
2015-05-23 13:35:12 +02:00
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++)
2015-10-01 03:48:58 +02:00
{
try
{
if ( (i+1 < length) && (text[i] == '&') )
return uChar(text[++i]);
}
catch (const std::out_of_range&)
{
return 0;;
}
}
2015-05-23 13:35:12 +02:00
return 0;
}
//----------------------------------------------------------------------
void FToggleButton::setHotkeyAccelerator()
{
int hotkey = getHotkey();
if ( hotkey )
{
if ( isalpha(hotkey) || isdigit(hotkey) )
{
2015-11-12 01:33:16 +01:00
addAccelerator (tolower(hotkey));
addAccelerator (toupper(hotkey));
2015-05-23 13:35:12 +02:00
// Meta + hotkey
2015-11-12 01:33:16 +01:00
addAccelerator (fc::Fmkey_meta + tolower(hotkey));
2015-05-23 13:35:12 +02:00
}
else
2015-11-12 01:33:16 +01:00
addAccelerator (getHotkey());
2015-05-23 13:35:12 +02:00
}
else
2015-11-12 01:33:16 +01:00
delAccelerator();
2015-05-23 13:35:12 +02:00
}
//----------------------------------------------------------------------
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
2015-07-06 10:50:46 +02:00
if ( isRadioButton() || isCheckboxButton() )
{
2015-07-06 10:50:46 +02:00
setCursorPos (xpos+xmin, ypos+ymin-1);
2015-07-06 10:50:46 +02:00
if ( isCursorInside() && hasFocus() )
{
if ( isHiddenCursor() )
showCursor();
}
else if ( ! isHiddenCursor() )
hideCursor();
2015-05-23 13:35:12 +02:00
}
}
//----------------------------------------------------------------------
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;
2015-10-17 19:40:43 +02:00
LabelText = new wchar_t[length+1]();
2015-09-22 04:18:20 +02:00
txt = text;
2015-05-23 13:35:12 +02:00
src = const_cast<wchar_t*>(txt.wc_str());
dest = const_cast<wchar_t*>(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--;
2015-07-04 22:51:47 +02:00
gotoxy (xpos+xmin-1+label_offset_pos, ypos+ymin-1);
2015-05-23 13:35:12 +02:00
2015-10-11 21:56:16 +02:00
if ( isMonochron() )
setReverse(true);
2015-05-23 13:35:12 +02:00
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
2015-10-11 21:56:16 +02:00
print (LabelText[z]);
2015-05-23 13:35:12 +02:00
}
2015-10-11 21:56:16 +02:00
if ( isMonochron() )
setReverse(false);
2015-05-23 13:35:12 +02:00
delete[] LabelText;
}
//----------------------------------------------------------------------
void FToggleButton::processClick()
{
emitCallback("clicked");
}
//----------------------------------------------------------------------
void FToggleButton::processToggle()
{
emitCallback("toggled");
}
//----------------------------------------------------------------------
FButtonGroup* FToggleButton::group() const
{
return button_group;
}
//----------------------------------------------------------------------
bool FToggleButton::isRadioButton() const
{
2015-09-22 04:18:20 +02:00
return ( strcmp ( getClassName()
, const_cast<char*>("FRadioButton") ) == 0 );
2015-05-23 13:35:12 +02:00
}
2015-07-06 10:50:46 +02:00
//----------------------------------------------------------------------
bool FToggleButton::isCheckboxButton() const
{
2015-09-22 04:18:20 +02:00
return ( strcmp ( getClassName()
, const_cast<char*>("FCheckBox") ) == 0 );
2015-07-06 10:50:46 +02:00
}
2015-05-23 13:35:12 +02:00
//----------------------------------------------------------------------
2015-09-20 05:44:50 +02:00
void FToggleButton::onKeyPress (FKeyEvent* ev)
2015-05-23 13:35:12 +02:00
{
int key;
if ( ! isEnabled() )
return;
2015-09-20 05:44:50 +02:00
key = ev->key();
2015-05-23 13:35:12 +02:00
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();
2015-09-20 05:44:50 +02:00
ev->accept();
2015-05-23 13:35:12 +02:00
break;
case fc::Fkey_down:
case fc::Fkey_right:
focus_inside_group = true;
focusNextChild();
2015-09-20 05:44:50 +02:00
ev->accept();
2015-05-23 13:35:12 +02:00
break;
case fc::Fkey_up:
case fc::Fkey_left:
focus_inside_group = true;
focusPrevChild();
2015-09-20 05:44:50 +02:00
ev->accept();
break;
default:
2015-05-23 13:35:12 +02:00
break;
}
2015-09-20 05:44:50 +02:00
if ( ev->isAccepted() )
2015-05-23 13:35:12 +02:00
{
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)
{
2015-07-04 22:51:47 +02:00
int min_width = button_width + int(text.getLength());
2015-05-23 13:35:12 +02:00
if ( w < min_width )
w = min_width;
FWidget::setGeometry(x, y, w, h, adjust);
}
//----------------------------------------------------------------------
2015-12-21 18:48:38 +01:00
bool FToggleButton::setNoUnderline (bool on)
2015-05-23 13:35:12 +02:00
{
if ( on )
2015-09-22 04:18:20 +02:00
flags |= NO_UNDERLINE;
2015-05-23 13:35:12 +02:00
else
2015-09-22 04:18:20 +02:00
flags &= ~NO_UNDERLINE;
2015-05-23 13:35:12 +02:00
return on;
}
//----------------------------------------------------------------------
2015-12-21 18:48:38 +01:00
bool FToggleButton::setEnable (bool on)
2015-05-23 13:35:12 +02:00
{
FWidget::setEnable(on);
if ( on )
{
2015-09-22 04:18:20 +02:00
flags |= ACTIVE;
2015-05-23 13:35:12 +02:00
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
{
2015-09-22 04:18:20 +02:00
flags &= ~ACTIVE;
2015-11-12 01:33:16 +01:00
delAccelerator();
2015-05-23 13:35:12 +02:00
foregroundColor = wc.toggle_button_inactive_fg;
backgroundColor = wc.toggle_button_inactive_bg;
}
return on;
}
//----------------------------------------------------------------------
2015-12-21 18:48:38 +01:00
bool FToggleButton::setFocus (bool on)
2015-05-23 13:35:12 +02:00
{
FWidget::setFocus(on);
if ( on )
{
2015-09-22 04:18:20 +02:00
flags |= FOCUS;
2015-05-23 13:35:12 +02:00
if ( isEnabled() )
{
if ( isRadioButton() )
focus_inside_group = false;
foregroundColor = wc.toggle_button_active_focus_fg;
backgroundColor = wc.toggle_button_active_focus_bg;
2015-10-11 21:56:16 +02:00
if ( isCursorInside() && (isRadioButton() || isCheckboxButton()) )
2015-05-23 13:35:12 +02:00
showCursor();
if ( statusBar() )
{
FString msg = getStatusbarMessage();
FString curMsg = statusBar()->getMessage();
if ( curMsg != msg )
statusBar()->setMessage(msg);
}
}
}
else
{
2015-09-22 04:18:20 +02:00
flags &= ~FOCUS;
2015-05-23 13:35:12 +02:00
if ( isEnabled() )
{
foregroundColor = wc.toggle_button_active_fg;
backgroundColor = wc.toggle_button_active_bg;
hideCursor();
if ( statusBar() )
statusBar()->clearMessage();
}
}
return on;
}
//----------------------------------------------------------------------
2015-09-20 05:44:50 +02:00
void FToggleButton::onMouseDown (FMouseEvent* ev)
2015-05-23 13:35:12 +02:00
{
2015-09-20 05:44:50 +02:00
if ( ev->getButton() != LeftButton )
2015-05-23 13:35:12 +02:00
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();
}
}
}
//----------------------------------------------------------------------
2015-09-20 05:44:50 +02:00
void FToggleButton::onMouseUp (FMouseEvent* ev)
2015-05-23 13:35:12 +02:00
{
2015-09-20 05:44:50 +02:00
if ( ev->getButton() != LeftButton )
2015-05-23 13:35:12 +02:00
return;
2015-09-20 05:44:50 +02:00
if ( getGeometryGlobal().contains(ev->getGlobalPos()) )
2015-05-23 13:35:12 +02:00
{
if ( isRadioButton() )
{
if ( ! checked )
{
checked = true;
processToggle();
}
}
else
{
checked = not checked;
processToggle();
}
2015-09-22 04:18:20 +02:00
redraw();
2015-05-23 13:35:12 +02:00
processClick();
}
}
//----------------------------------------------------------------------
2015-09-20 05:44:50 +02:00
void FToggleButton::onAccel (FAccelEvent* ev)
2015-05-23 13:35:12 +02:00
{
if ( isEnabled() )
{
if ( ! hasFocus() )
{
2015-09-20 05:44:50 +02:00
FWidget* focused_widget = static_cast<FWidget*>(ev->focusedWidget());
2015-05-23 13:35:12 +02:00
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();
}
2015-09-22 04:18:20 +02:00
redraw();
2015-05-23 13:35:12 +02:00
if ( statusBar() )
{
statusBar()->drawMessage();
updateTerminal();
flush_out();
}
processClick();
2015-09-20 05:44:50 +02:00
ev->accept();
2015-05-23 13:35:12 +02:00
}
}
//----------------------------------------------------------------------
void FToggleButton::onHide (FHideEvent*)
{
if ( hasFocus() )
hideCursor();
}
//----------------------------------------------------------------------
void FToggleButton::onFocusIn (FFocusEvent*)
{
if ( isCursorInside() && (isRadioButton() || isCheckboxButton()) )
2015-05-23 13:35:12 +02:00
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();
}
}
}
//----------------------------------------------------------------------
2015-12-21 18:48:38 +01:00
bool FToggleButton::setChecked (bool on)
2015-05-23 13:35:12 +02:00
{
if ( checked != on )
{
checked = on;
processToggle();
}
return checked;
}
//----------------------------------------------------------------------
void FToggleButton::setText (FString txt)
{
2015-09-22 04:18:20 +02:00
text = txt;
2015-07-04 22:51:47 +02:00
setWidth(button_width + int(text.getLength()));
2015-05-23 13:35:12 +02:00
if ( isEnabled() )
{
2015-11-12 01:33:16 +01:00
delAccelerator();
2015-05-23 13:35:12 +02:00
setHotkeyAccelerator();
}
}