2015-09-25 21:37:19 +02:00
|
|
|
// File: flabel.cpp
|
|
|
|
// Provides: class FLabel
|
2015-05-23 13:35:12 +02:00
|
|
|
|
|
|
|
#include "fapp.h"
|
|
|
|
#include "flabel.h"
|
|
|
|
#include "fstatusbar.h"
|
|
|
|
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// class FLabel
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
|
|
|
|
// constructors and destructor
|
|
|
|
//----------------------------------------------------------------------
|
2015-09-22 04:18:20 +02:00
|
|
|
FLabel::FLabel(FWidget* parent)
|
|
|
|
: FWidget(parent)
|
|
|
|
, multiline_text()
|
|
|
|
, multiline(false)
|
|
|
|
, text()
|
|
|
|
, emphasis(0)
|
|
|
|
, alignment(fc::alignLeft)
|
|
|
|
, emphasis_color(wc.label_emphasis_fg)
|
|
|
|
, ellipsis_color(wc.label_ellipsis_fg)
|
|
|
|
, reverse_mode(false)
|
|
|
|
, accel_widget(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
|
|
|
FLabel::FLabel (const FString& txt, FWidget* parent)
|
|
|
|
: FWidget(parent)
|
|
|
|
, multiline_text()
|
|
|
|
, multiline(false)
|
|
|
|
, text(txt)
|
|
|
|
, emphasis(0)
|
|
|
|
, alignment(fc::alignLeft)
|
|
|
|
, emphasis_color(wc.label_emphasis_fg)
|
|
|
|
, ellipsis_color(wc.label_ellipsis_fg)
|
|
|
|
, reverse_mode(false)
|
|
|
|
, accel_widget(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);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
FLabel::~FLabel() // destructor
|
|
|
|
{
|
2015-11-12 01:33:16 +01:00
|
|
|
delAccelerator();
|
2015-05-23 13:35:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// private methods of FLabel
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void FLabel::init()
|
|
|
|
{
|
|
|
|
if ( isEnabled() )
|
2015-09-22 04:18:20 +02:00
|
|
|
flags |= ACTIVE;
|
2015-05-23 13:35:12 +02:00
|
|
|
|
|
|
|
unsetFocusable();
|
|
|
|
|
|
|
|
foregroundColor = parentWidget()->getForegroundColor();
|
|
|
|
backgroundColor = parentWidget()->getBackgroundColor();
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
uChar FLabel::getHotkey()
|
|
|
|
{
|
|
|
|
uInt length;
|
|
|
|
|
|
|
|
if ( text.isEmpty() )
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
length = text.getLength();
|
|
|
|
|
|
|
|
for (uInt i=0; i < length; i++)
|
2015-09-30 22:39:02 +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;
|
|
|
|
}
|
|
|
|
|
2015-07-18 21:31:26 +02:00
|
|
|
//----------------------------------------------------------------------
|
2015-07-18 21:36:57 +02:00
|
|
|
int FLabel::getHotkeyPos (wchar_t*& src, wchar_t*& dest, uInt length)
|
2015-07-18 21:31:26 +02:00
|
|
|
{
|
|
|
|
// find hotkey position in string
|
|
|
|
// + generate a new string without the '&'-sign
|
|
|
|
int hotkeypos = -1;
|
|
|
|
wchar_t* txt = src;
|
|
|
|
|
|
|
|
for (uInt i=0; i < length; i++)
|
|
|
|
{
|
|
|
|
if ( (i < length) && (txt[i] == L'&') && (hotkeypos == -1) )
|
|
|
|
{
|
|
|
|
hotkeypos = int(i);
|
|
|
|
i++;
|
|
|
|
src++;
|
|
|
|
}
|
|
|
|
*dest++ = *src++;
|
|
|
|
}
|
|
|
|
return hotkeypos;
|
|
|
|
}
|
|
|
|
|
2015-05-23 13:35:12 +02:00
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void FLabel::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
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
2015-07-18 21:31:26 +02:00
|
|
|
int FLabel::getXOffset(int length)
|
2015-05-23 13:35:12 +02:00
|
|
|
{
|
2015-07-18 21:31:26 +02:00
|
|
|
switch ( alignment )
|
|
|
|
{
|
|
|
|
case fc::alignLeft:
|
|
|
|
return 0;
|
2015-05-23 13:35:12 +02:00
|
|
|
|
2015-07-18 21:31:26 +02:00
|
|
|
case fc::alignCenter:
|
|
|
|
if ( length < width )
|
|
|
|
return int((width - length) / 2);
|
|
|
|
else
|
|
|
|
return 0;
|
2015-05-23 13:35:12 +02:00
|
|
|
|
2015-07-18 21:31:26 +02:00
|
|
|
case fc::alignRight:
|
|
|
|
if ( length < width )
|
|
|
|
return width - length;
|
|
|
|
else
|
|
|
|
return 0;
|
2015-09-20 05:44:50 +02:00
|
|
|
|
|
|
|
default:
|
|
|
|
return 0;
|
2015-07-18 21:31:26 +02:00
|
|
|
}
|
|
|
|
}
|
2015-05-23 13:35:12 +02:00
|
|
|
|
2015-07-18 21:31:26 +02:00
|
|
|
//----------------------------------------------------------------------
|
2015-09-22 04:18:20 +02:00
|
|
|
void FLabel::printLine ( wchar_t*& line
|
|
|
|
, uInt length
|
|
|
|
, int hotkeypos
|
|
|
|
, int xoffset )
|
2015-07-18 21:31:26 +02:00
|
|
|
{
|
|
|
|
int to_char;
|
|
|
|
bool isActive, isNoUnderline;
|
2015-05-23 13:35:12 +02:00
|
|
|
|
|
|
|
isActive = ((flags & ACTIVE) != 0);
|
|
|
|
isNoUnderline = ((flags & NO_UNDERLINE) != 0);
|
|
|
|
|
|
|
|
for (int x=0; x < xoffset; x++)
|
|
|
|
print (' ');
|
|
|
|
|
|
|
|
if ( length <= uInt(width) )
|
|
|
|
to_char = int(length);
|
|
|
|
else
|
|
|
|
to_char = width - 2;
|
2015-05-26 23:15:49 +02:00
|
|
|
|
2015-05-23 13:35:12 +02:00
|
|
|
if ( hasReverseMode() )
|
|
|
|
setReverse(true);
|
|
|
|
|
|
|
|
for (int z=0; z < to_char; z++)
|
|
|
|
{
|
2015-07-18 21:31:26 +02:00
|
|
|
if ( ! iswprint(wint_t(line[z])) )
|
2015-10-17 21:05:49 +02:00
|
|
|
{
|
|
|
|
if ( ! isNewFont() && ( int(line[z]) < fc::NF_rev_left_arrow2
|
|
|
|
|| int(line[z]) > fc::NF_check_mark ) )
|
|
|
|
{
|
|
|
|
line[z] = L' ';
|
|
|
|
}
|
|
|
|
}
|
2015-05-23 13:35:12 +02:00
|
|
|
if ( (z == hotkeypos) && isActive )
|
|
|
|
{
|
|
|
|
setColor (wc.label_hotkey_fg, wc.label_hotkey_bg);
|
|
|
|
if ( ! isNoUnderline )
|
|
|
|
setUnderline();
|
2015-07-18 21:31:26 +02:00
|
|
|
print ( line[z] );
|
2015-05-23 13:35:12 +02:00
|
|
|
if ( ! isNoUnderline )
|
|
|
|
unsetUnderline();
|
|
|
|
if ( hasEmphasis() )
|
|
|
|
setColor (emphasis_color, backgroundColor);
|
|
|
|
else
|
|
|
|
setColor (foregroundColor, backgroundColor);
|
|
|
|
}
|
|
|
|
else
|
2015-07-18 21:31:26 +02:00
|
|
|
print ( line[z] );
|
2015-05-23 13:35:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( length > uInt(width) )
|
|
|
|
{
|
|
|
|
setColor (ellipsis_color, backgroundColor);
|
|
|
|
print ("..");
|
2015-07-18 21:31:26 +02:00
|
|
|
setColor (foregroundColor, backgroundColor);
|
2015-05-23 13:35:12 +02:00
|
|
|
}
|
2015-07-18 21:31:26 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
for (int x=xoffset+to_char; x < width; x++)
|
|
|
|
print (' ');
|
|
|
|
}
|
|
|
|
|
2015-05-23 13:35:12 +02:00
|
|
|
if ( hasReverseMode() )
|
|
|
|
setReverse(false);
|
2015-07-18 21:31:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void FLabel::draw()
|
|
|
|
{
|
|
|
|
wchar_t* src;
|
|
|
|
wchar_t* dest;
|
|
|
|
wchar_t* LabelText;
|
|
|
|
uInt length;
|
|
|
|
int hotkeypos, xoffset;
|
2015-05-26 23:15:49 +02:00
|
|
|
|
2015-07-18 21:31:26 +02:00
|
|
|
if ( text.isNull() || text.isEmpty() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
setUpdateVTerm(false);
|
|
|
|
|
2015-10-11 21:56:16 +02:00
|
|
|
if ( isMonochron() )
|
|
|
|
{
|
|
|
|
setReverse(true);
|
|
|
|
if ( hasEmphasis() )
|
|
|
|
setBold(true);
|
|
|
|
}
|
|
|
|
|
2015-07-18 21:31:26 +02:00
|
|
|
if ( hasEmphasis() )
|
|
|
|
setColor (emphasis_color, backgroundColor);
|
|
|
|
else
|
|
|
|
setColor (foregroundColor, backgroundColor);
|
|
|
|
|
|
|
|
hotkeypos = -1;
|
|
|
|
|
|
|
|
if ( multiline && height > 1 )
|
|
|
|
{
|
|
|
|
uInt y = 0;
|
|
|
|
uInt text_lines = uInt(multiline_text.size());
|
|
|
|
bool hotkey_printed = false;
|
|
|
|
|
|
|
|
while ( y < text_lines && y < uInt(height) )
|
|
|
|
{
|
|
|
|
length = multiline_text[y].getLength();
|
2015-10-17 19:40:43 +02:00
|
|
|
LabelText = new wchar_t[length+1]();
|
2015-09-30 22:39:02 +02:00
|
|
|
src = const_cast<wchar_t*>(multiline_text[y].wc_str());
|
|
|
|
dest = const_cast<wchar_t*>(LabelText);
|
2015-07-18 21:31:26 +02:00
|
|
|
|
|
|
|
if ( ! hotkey_printed )
|
|
|
|
hotkeypos = getHotkeyPos(src, dest, length);
|
|
|
|
else
|
2015-09-30 22:39:02 +02:00
|
|
|
wcsncpy(dest, src, length);
|
2015-07-18 21:31:26 +02:00
|
|
|
|
|
|
|
gotoxy (xpos+xmin-1, ypos+ymin-1+int(y));
|
|
|
|
|
|
|
|
if ( hotkeypos != -1 )
|
|
|
|
{
|
|
|
|
xoffset = getXOffset (int(length-1));
|
|
|
|
printLine (LabelText, length-1, hotkeypos, xoffset);
|
|
|
|
hotkey_printed = true;
|
|
|
|
hotkeypos = -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
xoffset = getXOffset (int(length));
|
|
|
|
printLine (LabelText, length, -1, xoffset);
|
|
|
|
}
|
|
|
|
y++;
|
|
|
|
delete[] LabelText;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
length = text.getLength();
|
2015-10-17 19:40:43 +02:00
|
|
|
LabelText = new wchar_t[length+1]();
|
2015-07-18 21:31:26 +02:00
|
|
|
src = const_cast<wchar_t*>(text.wc_str());
|
|
|
|
dest = const_cast<wchar_t*>(LabelText);
|
|
|
|
|
|
|
|
hotkeypos = getHotkeyPos (src, dest, length);
|
|
|
|
|
|
|
|
if ( hotkeypos != -1 )
|
|
|
|
length--;
|
|
|
|
|
|
|
|
gotoxy (xpos+xmin-1, ypos+ymin-1);
|
|
|
|
xoffset = getXOffset (int(length));
|
|
|
|
printLine (LabelText, length, hotkeypos, xoffset);
|
|
|
|
delete[] LabelText;
|
|
|
|
}
|
2015-10-11 21:56:16 +02:00
|
|
|
if ( isMonochron() )
|
|
|
|
{
|
|
|
|
setReverse(false);
|
|
|
|
if ( hasEmphasis() )
|
|
|
|
setBold(false);
|
|
|
|
}
|
2015-05-23 13:35:12 +02:00
|
|
|
setUpdateVTerm(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// public methods of FLabel
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void FLabel::hide()
|
|
|
|
{
|
2016-01-08 01:00:05 +01:00
|
|
|
short fg, bg;
|
2015-05-23 13:35:12 +02:00
|
|
|
char* blank;
|
|
|
|
|
|
|
|
FWidget::hide();
|
|
|
|
|
|
|
|
fg = parentWidget()->getForegroundColor();
|
|
|
|
bg = parentWidget()->getBackgroundColor();
|
|
|
|
setColor (fg, bg);
|
|
|
|
|
|
|
|
blank = new char[width+1];
|
|
|
|
memset(blank, ' ', uLong(width));
|
|
|
|
blank[width] = '\0';
|
|
|
|
print (blank);
|
|
|
|
delete[] blank;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
2015-09-20 05:44:50 +02:00
|
|
|
void FLabel::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 ( ! isEnabled() || ! accel_widget )
|
|
|
|
return;
|
|
|
|
|
|
|
|
if ( ! accel_widget->hasFocus() )
|
|
|
|
{
|
|
|
|
FWidget* focused_widget = getFocusWidget();
|
|
|
|
FFocusEvent out (FocusOut_Event);
|
|
|
|
FApplication::queueEvent(focused_widget, &out);
|
|
|
|
accel_widget->setFocus();
|
|
|
|
if ( focused_widget )
|
|
|
|
focused_widget->redraw();
|
|
|
|
accel_widget->redraw();
|
|
|
|
if ( statusBar() )
|
|
|
|
{
|
|
|
|
accel_widget->statusBar()->drawMessage();
|
|
|
|
updateTerminal();
|
|
|
|
flush_out();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
2015-09-20 05:44:50 +02:00
|
|
|
void FLabel::onAccel (FAccelEvent* ev)
|
2015-05-23 13:35:12 +02:00
|
|
|
{
|
|
|
|
if ( ! isEnabled() || ! accel_widget )
|
|
|
|
return;
|
|
|
|
|
|
|
|
if ( ! accel_widget->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);
|
|
|
|
accel_widget->setFocus();
|
|
|
|
if ( focused_widget )
|
|
|
|
focused_widget->redraw();
|
|
|
|
accel_widget->redraw();
|
|
|
|
if ( statusBar() )
|
|
|
|
{
|
|
|
|
accel_widget->statusBar()->drawMessage();
|
|
|
|
updateTerminal();
|
|
|
|
flush_out();
|
|
|
|
}
|
|
|
|
}
|
2015-09-20 05:44:50 +02:00
|
|
|
ev->accept();
|
2015-05-23 13:35:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void FLabel::cb_accel_widget_destroyed (FWidget*, void*)
|
|
|
|
{
|
|
|
|
accel_widget = 0;
|
2015-11-12 01:33:16 +01:00
|
|
|
delAccelerator();
|
2015-05-23 13:35:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void FLabel::setAccelWidget (FWidget* widget)
|
|
|
|
{
|
|
|
|
if ( widget )
|
|
|
|
accel_widget = widget;
|
|
|
|
|
|
|
|
accel_widget->addCallback
|
|
|
|
(
|
|
|
|
"destroy",
|
2015-09-24 19:01:27 +02:00
|
|
|
_METHOD_CALLBACK (this, &FLabel::cb_accel_widget_destroyed)
|
2015-05-23 13:35:12 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void FLabel::setAlignment (uInt align)
|
|
|
|
{
|
|
|
|
if ( align != fc::alignLeft
|
|
|
|
&& align != fc::alignCenter
|
|
|
|
&& align != fc::alignRight )
|
|
|
|
alignment = fc::alignLeft;
|
|
|
|
else
|
|
|
|
alignment = align;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
bool FLabel::setEmphasis (bool on)
|
|
|
|
{
|
|
|
|
if ( on )
|
2015-09-22 04:18:20 +02:00
|
|
|
emphasis |= EMPHASIS;
|
2015-05-23 13:35:12 +02:00
|
|
|
else
|
2015-09-22 04:18:20 +02:00
|
|
|
emphasis &= ~EMPHASIS;
|
2015-05-23 13:35:12 +02:00
|
|
|
return on;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
bool FLabel::setReverseMode (bool on)
|
|
|
|
{
|
|
|
|
if ( reverse_mode != on )
|
|
|
|
reverse_mode = on;
|
|
|
|
return on;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
bool FLabel::setEnable (bool on)
|
|
|
|
{
|
|
|
|
FWidget::setEnable(on);
|
|
|
|
|
|
|
|
if ( on )
|
|
|
|
{
|
2015-09-22 04:18:20 +02:00
|
|
|
flags |= ACTIVE;
|
2015-05-23 13:35:12 +02:00
|
|
|
setHotkeyAccelerator();
|
|
|
|
}
|
|
|
|
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
|
|
|
}
|
|
|
|
return on;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void FLabel::setNumber (long num)
|
|
|
|
{
|
|
|
|
setText(FString().setNumber(num));
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
void FLabel::setText (const FString& txt)
|
|
|
|
{
|
2015-09-22 04:18:20 +02:00
|
|
|
text = txt;
|
|
|
|
multiline_text = text.split("\r\n");
|
2015-07-18 21:31:26 +02:00
|
|
|
if ( int(multiline_text.size()) > 1 )
|
|
|
|
multiline = true;
|
|
|
|
else
|
|
|
|
multiline = false;
|
|
|
|
|
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();
|
|
|
|
}
|
|
|
|
}
|