452 lines
12 KiB
C++
452 lines
12 KiB
C++
/***********************************************************************
|
|
* fmessagebox.cpp - Widget FMessageBox (a text message window) *
|
|
* *
|
|
* This file is part of the Final Cut widget toolkit *
|
|
* *
|
|
* Copyright 2014-2019 Markus Gans *
|
|
* *
|
|
* The Final Cut is free software; you can redistribute it and/or *
|
|
* modify it under the terms of the GNU Lesser General Public License *
|
|
* as published by the Free Software Foundation; either version 3 of *
|
|
* the License, or (at your option) any later version. *
|
|
* *
|
|
* The Final Cut is distributed in the hope that it will be useful, *
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
* GNU Lesser General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU Lesser General Public *
|
|
* License along with this program. If not, see *
|
|
* <http://www.gnu.org/licenses/>. *
|
|
***********************************************************************/
|
|
|
|
#include <algorithm>
|
|
|
|
#include "final/fapplication.h"
|
|
#include "final/fbutton.h"
|
|
#include "final/fmessagebox.h"
|
|
|
|
namespace finalcut
|
|
{
|
|
|
|
static const char* const button_text[] =
|
|
{
|
|
0,
|
|
"&OK",
|
|
"&Cancel",
|
|
"&Yes",
|
|
"&No",
|
|
"&Abort",
|
|
"&Retry",
|
|
"&Ignore",
|
|
0
|
|
};
|
|
|
|
//----------------------------------------------------------------------
|
|
// class FMessageBox
|
|
//----------------------------------------------------------------------
|
|
|
|
// constructors and destructor
|
|
//----------------------------------------------------------------------
|
|
FMessageBox::FMessageBox (FWidget* parent)
|
|
: FDialog(parent)
|
|
{
|
|
setTitlebarText("Message for you");
|
|
init(FMessageBox::Ok, 0, 0);
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
FMessageBox::FMessageBox (const FMessageBox& mbox)
|
|
: FDialog(mbox.getParentWidget())
|
|
, headline_text(mbox.headline_text)
|
|
, text(mbox.text)
|
|
, text_components(mbox.text_components)
|
|
, max_line_width(mbox.max_line_width)
|
|
, emphasis_color(mbox.emphasis_color)
|
|
, num_buttons(mbox.num_buttons)
|
|
, text_num_lines(mbox.text_num_lines)
|
|
, center_text(mbox.center_text)
|
|
{
|
|
setTitlebarText (mbox.getTitlebarText());
|
|
init ( mbox.button_digit[0]
|
|
, mbox.button_digit[1]
|
|
, mbox.button_digit[2] );
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
FMessageBox::FMessageBox ( const FString& caption
|
|
, const FString& message
|
|
, int button0
|
|
, int button1
|
|
, int button2
|
|
, FWidget* parent )
|
|
: FDialog(parent)
|
|
, text(message)
|
|
{
|
|
setTitlebarText(caption);
|
|
init(button0, button1, button2);
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
FMessageBox::~FMessageBox() // destructor
|
|
{
|
|
deallocation();
|
|
}
|
|
|
|
|
|
// public methods of FMessageBox
|
|
//----------------------------------------------------------------------
|
|
FMessageBox& FMessageBox::operator = (const FMessageBox& mbox)
|
|
{
|
|
if ( &mbox == this )
|
|
{
|
|
return *this;
|
|
}
|
|
else
|
|
{
|
|
for (uInt n{0}; n < num_buttons; n++)
|
|
delete button[n];
|
|
|
|
if ( mbox.getParentWidget() )
|
|
mbox.getParentWidget()->addChild (this);
|
|
|
|
headline_text = mbox.headline_text;
|
|
text = mbox.text;
|
|
text_components = mbox.text_components;
|
|
max_line_width = mbox.max_line_width;
|
|
center_text = mbox.center_text;
|
|
emphasis_color = mbox.emphasis_color;
|
|
num_buttons = mbox.num_buttons;
|
|
text_num_lines = mbox.text_num_lines;
|
|
|
|
setTitlebarText (mbox.getTitlebarText());
|
|
init ( mbox.button_digit[0]
|
|
, mbox.button_digit[1]
|
|
, mbox.button_digit[2] );
|
|
|
|
return *this;
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void FMessageBox::setHeadline (const FString& headline)
|
|
{
|
|
headline_text.setString(headline);
|
|
setHeight(getHeight() + 2, true);
|
|
|
|
for (uInt n{0}; n < num_buttons; n++)
|
|
button[n]->setY (int(getHeight()) - 4, false);
|
|
|
|
std::size_t column_width = getColumnWidth(headline_text);
|
|
|
|
if ( column_width > max_line_width )
|
|
max_line_width = column_width;
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void FMessageBox::setText (const FString& txt)
|
|
{
|
|
text.setString(txt);
|
|
calculateDimensions();
|
|
button[0]->setY (int(getHeight()) - 4, false);
|
|
|
|
if ( button_digit[1] != 0 )
|
|
button[1]->setY (int(getHeight()) - 4, false);
|
|
|
|
if ( button_digit[2] != 0 )
|
|
button[2]->setY (int(getHeight()) - 4, false);
|
|
|
|
adjustButtons();
|
|
}
|
|
|
|
|
|
// protected methods of FMessageBox
|
|
//----------------------------------------------------------------------
|
|
void FMessageBox::adjustSize()
|
|
{
|
|
std::size_t max_width{};
|
|
std::size_t max_height{};
|
|
auto root_widget = getRootWidget();
|
|
|
|
if ( root_widget )
|
|
{
|
|
max_width = root_widget->getClientWidth();
|
|
max_height = root_widget->getClientHeight();
|
|
}
|
|
else
|
|
{
|
|
// fallback to xterm default size
|
|
max_width = 80;
|
|
max_height = 24;
|
|
}
|
|
|
|
int x = 1 + int((max_width - getWidth()) / 2);
|
|
int y = 1 + int((max_height - getHeight()) / 3);
|
|
setPos(FPoint(x, y), false);
|
|
FDialog::adjustSize();
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void FMessageBox::cb_processClick (FWidget*, FDataPtr data)
|
|
{
|
|
int reply = *(static_cast<int*>(data));
|
|
done (reply);
|
|
}
|
|
|
|
|
|
// private methods of FMessageBox
|
|
//----------------------------------------------------------------------
|
|
void FMessageBox::init (int button0, int button1, int button2)
|
|
{
|
|
calculateDimensions();
|
|
|
|
if ( (button2 && ! button1) || (button1 && ! button0) )
|
|
{
|
|
button0 = button1 = button2 = 0;
|
|
}
|
|
|
|
if ( button0 == 0 )
|
|
button0 = FMessageBox::Ok;
|
|
|
|
if ( button1 == 0 && button2 == 0 )
|
|
num_buttons = 1;
|
|
else if ( button2 == 0 )
|
|
num_buttons = 2;
|
|
else
|
|
num_buttons = 3;
|
|
|
|
button_digit[0] = button0;
|
|
button_digit[1] = button1;
|
|
button_digit[2] = button2;
|
|
|
|
allocation (button0, button1, button2);
|
|
resizeButtons();
|
|
adjustButtons();
|
|
initCallbacks();
|
|
setModal();
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
inline void FMessageBox::allocation (int button0, int button1, int button2)
|
|
{
|
|
try
|
|
{
|
|
button[0] = new FButton (this);
|
|
button[0]->setText(button_text[button0]);
|
|
button[0]->setPos(FPoint(3, int(getHeight()) - 4), false);
|
|
button[0]->setWidth(1, false);
|
|
button[0]->setHeight(1, false);
|
|
button[0]->setFocus();
|
|
|
|
if ( button1 > 0 )
|
|
{
|
|
button[1] = new FButton(this);
|
|
button[1]->setText(button_text[button1]);
|
|
button[1]->setPos(FPoint(17, int(getHeight()) - 4), false);
|
|
button[1]->setWidth(0, false);
|
|
button[1]->setHeight(1, false);
|
|
}
|
|
|
|
if ( button2 > 0 )
|
|
{
|
|
button[2] = new FButton(this);
|
|
button[2]->setText(button_text[button2]);
|
|
button[2]->setPos(FPoint(32, int(getHeight()) - 4), false);
|
|
button[2]->setWidth(0, false);
|
|
button[2]->setHeight(1, false);
|
|
}
|
|
}
|
|
catch (const std::bad_alloc& ex)
|
|
{
|
|
std::cerr << bad_alloc_str << ex.what() << std::endl;
|
|
return;
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
inline void FMessageBox::deallocation()
|
|
{
|
|
for (uInt n{0}; n < num_buttons; n++)
|
|
delete button[n];
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
inline void FMessageBox::initCallbacks()
|
|
{
|
|
if ( button_digit[0] != 0 )
|
|
{
|
|
button[0]->addCallback
|
|
(
|
|
"clicked",
|
|
F_METHOD_CALLBACK (this, &FMessageBox::cb_processClick),
|
|
static_cast<FDataPtr>(&button_digit[0])
|
|
);
|
|
}
|
|
|
|
if ( button_digit[1] != 0 )
|
|
{
|
|
button[1]->addCallback
|
|
(
|
|
"clicked",
|
|
F_METHOD_CALLBACK (this, &FMessageBox::cb_processClick),
|
|
static_cast<FDataPtr>(&button_digit[1])
|
|
);
|
|
}
|
|
|
|
if ( button_digit[2] != 0 )
|
|
{
|
|
button[2]->addCallback
|
|
(
|
|
"clicked",
|
|
F_METHOD_CALLBACK (this, &FMessageBox::cb_processClick),
|
|
static_cast<FDataPtr>(&button_digit[2])
|
|
);
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void FMessageBox::calculateDimensions()
|
|
{
|
|
FSize size{};
|
|
std::size_t headline_height{0};
|
|
text_components = text.split("\n");
|
|
max_line_width = 0;
|
|
text_num_lines = std::size_t(text_components.size());
|
|
|
|
if ( text_num_lines == 0 )
|
|
return;
|
|
|
|
if ( ! headline_text.isNull() )
|
|
headline_height = 2;
|
|
|
|
for (auto&& line : text_components)
|
|
{
|
|
std::size_t column_width = getColumnWidth(line);
|
|
|
|
if ( column_width > max_line_width )
|
|
max_line_width = column_width;
|
|
}
|
|
|
|
size.setHeight (text_num_lines + 8 + headline_height);
|
|
size.setWidth (max_line_width + 4);
|
|
|
|
if ( size.getWidth() < 20 )
|
|
size.setWidth(20);
|
|
|
|
setSize(size);
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void FMessageBox::draw()
|
|
{
|
|
FDialog::draw();
|
|
|
|
int y{0};
|
|
int head_offset{0};
|
|
int center_x{0};
|
|
// center the whole block
|
|
int msg_x = int((getWidth() - max_line_width) / 2);
|
|
|
|
if ( isMonochron() )
|
|
setReverse(true);
|
|
|
|
if ( ! headline_text.isEmpty() )
|
|
{
|
|
setColor(emphasis_color, getBackgroundColor());
|
|
std::size_t headline_width = getColumnWidth(headline_text);
|
|
|
|
if ( center_text ) // center one line
|
|
center_x = int((max_line_width - headline_width) / 2);
|
|
|
|
print() << FPoint(1 + msg_x + center_x, 4) << headline_text;
|
|
head_offset = 2;
|
|
}
|
|
|
|
setColor();
|
|
|
|
for (auto&& line : text_components)
|
|
{
|
|
std::size_t line_width = getColumnWidth(line);
|
|
|
|
if ( center_text ) // center one line
|
|
center_x = int((max_line_width - line_width) / 2);
|
|
|
|
print() << FPoint(1 + msg_x + center_x, 4 + head_offset + y)
|
|
<< line;
|
|
y++;
|
|
}
|
|
|
|
if ( isMonochron() )
|
|
setReverse(false);
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void FMessageBox::resizeButtons()
|
|
{
|
|
std::size_t len[3]{}, max_size{};
|
|
|
|
for (std::size_t n{0}; n < num_buttons; n++)
|
|
{
|
|
len[n] = button[n]->getText().getLength();
|
|
|
|
if ( button[n]->getText().includes('&') )
|
|
len[n]--;
|
|
}
|
|
|
|
if ( num_buttons == 1 )
|
|
max_size = len[0];
|
|
else
|
|
{
|
|
assert ( num_buttons > 1 );
|
|
max_size = std::max(len[0], len[1]);
|
|
|
|
if ( num_buttons == 3 )
|
|
max_size = std::max(max_size, len[2]);
|
|
}
|
|
|
|
if ( max_size < 7 )
|
|
max_size = 7;
|
|
|
|
for (std::size_t n{0}; n < num_buttons; n++)
|
|
button[n]->setWidth(max_size + 3, false);
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
void FMessageBox::adjustButtons()
|
|
{
|
|
static constexpr std::size_t gap = 4;
|
|
std::size_t btn_width{0};
|
|
|
|
for (std::size_t n{0}; n < num_buttons; n++)
|
|
{
|
|
if ( n == num_buttons - 1 )
|
|
btn_width += button[n]->getWidth();
|
|
else
|
|
btn_width += button[n]->getWidth() + gap;
|
|
}
|
|
|
|
if ( btn_width + 4 >= getWidth() )
|
|
{
|
|
std::size_t max_width;
|
|
auto root_widget = getRootWidget();
|
|
setWidth(btn_width + 5);
|
|
max_width = ( root_widget ) ? root_widget->getClientWidth() : 80;
|
|
setX (int((max_width - getWidth()) / 2));
|
|
}
|
|
|
|
int btn_x = int((getWidth() - btn_width) / 2);
|
|
|
|
for (std::size_t n{0}; n < num_buttons; n++)
|
|
{
|
|
if ( n == 0 )
|
|
button[n]->setX(btn_x);
|
|
else
|
|
{
|
|
int btn_size = int(button[n]->getWidth());
|
|
button[n]->setX(btn_x + int(n) * (btn_size + int(gap)));
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace finalcut
|