ultimatepp/uppsrc/CtrlLib/MultiButton.cpp

866 lines
16 KiB
C++

#include "CtrlLib.h"
namespace Upp {
CH_STYLE(MultiButton, Style, StyleDefault)
{
for(int i = 0; i < 4; i++) {
simple[i] = left[i] = right[i] = lmiddle[i] = rmiddle[i]
= Button::StyleEdge().look[i];
monocolor[i] = Button::StyleEdge().monocolor[i];
fmonocolor[i] = i == CTRL_DISABLED ? SColorDisabled() : SColorText();
look[i] = trivial[i] = ChLookWith(simple[i], CtrlsImg::DA(), monocolor[i]);
edge[i] = EditFieldEdge();
}
activeedge = false;
trivialborder = DPI(1);
border = DPI(1);
pressoffset = Button::StyleEdge().pressoffset;
sep1 = SColorHighlight();
sep2 = SColorLight();
sepm = DPI(2);
stdwidth = FrameButtonWidth();
trivialsep = false;
margin = Rect(DPI(2), 2, DPI(2), 2);
usetrivial = false;
overpaint = loff = roff = 0;
error = Blend(LtRed(), Red());
paper = SColorPaper();
coloredge = Null;
clipedge = false;
}
CH_STYLE(MultiButton, Style, StyleFrame)
{
Assign(MultiButton::StyleDefault());
}
MultiButton::SubButton::SubButton()
{
cx = Null;
left = false;
monoimg = false;
enabled = true;
main = false;
visible = true;
}
void MultiButton::SubButton::Refresh()
{
if(owner) {
owner->Refresh();
if(owner->Frame() && owner->GetParent())
owner->GetParent()->RefreshLayout();
}
}
MultiButton::SubButton& MultiButton::SubButton::SetImage(const Image& m)
{
img = m;
monoimg = false;
Refresh();
return *this;
}
MultiButton::SubButton& MultiButton::SubButton::SetMonoImage(const Image& m)
{
img = m;
monoimg = true;
Refresh();
return *this;
}
MultiButton::SubButton& MultiButton::SubButton::SetStdImage()
{
img = Null;
monoimg = true;
Refresh();
return *this;
}
MultiButton::SubButton& MultiButton::SubButton::SetLabel(const char *text)
{
label = text;
Refresh();
return *this;
}
MultiButton::SubButton& MultiButton::SubButton::Left(bool b)
{
left = b;
Refresh();
return *this;
}
MultiButton::SubButton& MultiButton::SubButton::Width(int w)
{
cx = w;
Refresh();
return *this;
}
MultiButton::SubButton& MultiButton::SubButton::Enable(bool b)
{
enabled = b;
Refresh();
return *this;
}
MultiButton::SubButton& MultiButton::SubButton::Show(bool b)
{
visible = b;
Refresh();
return *this;
}
void MultiButton::MultiButtons()
{
if(droppush) {
SubButton& b = buttons.Add();
b.owner = this;
b.WhenPush = [this] { DropPush(); };
b.main = true;
droppush = false;
}
}
MultiButton::SubButton& MultiButton::AddButton()
{
MultiButtons();
SubButton& b = buttons.Add();
b.owner = this;
return b;
}
MultiButton::SubButton& MultiButton::InsertButton(int i)
{
MultiButtons();
SubButton& b = buttons.Insert(i);
b.owner = this;
return b;
}
void MultiButton::RemoveButton(int i)
{
MultiButtons();
buttons.Remove(i);
}
void MultiButton::Reset()
{
MultiButtons();
buttons.Clear();
}
int MultiButton::GetButtonCount() const
{
if(droppush)
return 1;
return buttons.GetCount();
}
MultiButton::SubButton& MultiButton::Button(int i) const
{
if(droppush) {
static SubButton b;
b.main = true;
return b;
}
return const_cast<MultiButton *>(this)->buttons[i];
}
const MultiButton::SubButton& MultiButton::GetButton(int i) const
{
return Button(i);
}
MultiButton::SubButton& MultiButton::GetButton(int i)
{
MultiButtons();
return Button(i);
}
MultiButton::SubButton& MultiButton::MainButton()
{
for(int i = 0; i < GetButtonCount(); i++) {
SubButton& b = GetButton(i);
if(b.main)
return b;
}
NEVER();
return GetButton(0);
}
MultiButton::SubButton& MultiButton::SubButton::Main(bool b)
{
if(b && owner)
for(int i = 0; i < owner->GetButtonCount(); i++)
owner->GetButton(i).main = false;
main = b;
return *this;
}
Size MultiButton::GetMinSize() const
{
return Size(4, StdFont().Info().GetHeight() + 6);
}
void MultiButton::GotFocus()
{
Refresh();
}
void MultiButton::LostFocus()
{
Refresh();
}
static CtrlFrame& sNullFrame()
{
static NullFrameClass f;
return f;
}
int MultiButton::OverPaint() const
{
return style->overpaint;
}
bool MultiButton::Frame()
{
return false;
}
MultiButton& MultiButton::NoBackground(bool b)
{
SetFrame(EditFieldFrame());
nobg = b;
Transparent();
Refresh();
return *this;
}
bool MultiButton::ComplexFrame()
{
return Frame() ? GetParent() && &GetParent()->GetFrame() != &sNullFrame()
: &GetFrame() != &sNullFrame();
}
bool MultiButton::Metrics(int& border, int& lx, int &rx, const Rect& r)
{
border = IsTrivial() ? style->trivialborder : style->border;
lx = r.left;
rx = r.right;
if(ComplexFrame()) {
border = 0;
return false;
}
if(!IsNull(style->edge[0])) {
lx += border;
rx -= border;
return true;
}
border = 0;
return false;
}
bool MultiButton::Metrics(int& border, int& lx, int &rx)
{
return Metrics(border, lx, rx, GetSize());
}
int MultiButton::FindButton(int px)
{
if(IsReadOnly())
return Null;
if(IsTrivial() && !Frame() && GetButtonCount())
return GetButton(0).enabled ? 0 : Null;
int border, lx, rx;
Metrics(border, lx, rx);
for(int i = 0; i < GetButtonCount(); i++) {
SubButton& b = Button(i);
int x = 0, cx = 0;
if(GetPos(b, lx, rx, x, cx, px))
return b.enabled ? i : Null;
}
if(HasMain())
return MAIN;
if(display)
for(int i = 0; i < GetButtonCount(); i++)
if(Button(i).main)
return i;
return Null;
}
bool MultiButton::GetPos(SubButton& b, int& lx, int& rx, int& x, int& cx, int px)
{
Size tsz = GetTextSize(b.label, StdFont());
if(b.visible) {
cx = Nvl(b.cx, style->stdwidth + tsz.cx);
if(IsNull(b.cx) && tsz.cx > 0 && !IsNull(b.img))
cx += DPI(LB_IMAGE + LB_MARGIN);
}
else
cx = 0;
if(b.left) {
x = lx;
lx += cx;
if(px >= 0 && px < lx)
return true;
}
else {
rx -= cx;
x = rx;
if(px >= 0 && px >= rx)
return true;
}
return false;
}
void MultiButton::GetPos(int ii, int& x, int& cx)
{
int border, lx, rx;
Metrics(border, lx, rx);
x = cx = 0;
for(int i = 0; i <= ii; i++) {
SubButton& b = Button(i);
GetPos(b, lx, rx, x, cx);
}
}
void MultiButton::GetLR(int& lx, int& rx)
{
int border;
Metrics(border, lx, rx);
int x = 0;
int cx = 0;
for(int i = 0; i < GetButtonCount(); i++) {
SubButton& b = Button(i);
GetPos(b, lx, rx, x, cx);
}
}
int MultiButton::ChState(int i)
{
bool frm = Frame();
Ctrl *p = GetParent();
if(i == MAIN && frm && style->activeedge) {
int q = 0;
if(p)
q = !p->IsEnabled() || !IsEnabled() || p->IsReadOnly() ? CTRL_DISABLED
: p->HasFocus() || push ? CTRL_PRESSED
: p->HasMouse() || hl >= 0 ? CTRL_HOT
: CTRL_NORMAL;
return q;
}
if(!IsShowEnabled() || IsReadOnly() || frm && p && p->IsReadOnly() || i >= 0 && !Button(i).enabled)
return CTRL_DISABLED;
if(IsTrivial() && !frm)
i = 0;
return hl == i ? push ? CTRL_PRESSED
: CTRL_HOT
: CTRL_NORMAL;
}
Rect MultiButton::GetMargin()
{
Rect m = style->margin;
int fcy = GetStdFontCy();
int cy = GetSize().cy;
if(m.top + m.bottom + fcy > cy)
m.top = m.bottom = max((cy - fcy) / 2, 0);
return m;
}
void MultiButton::Lay(Rect& r, bool minsize)
{
int border, lx, rx;
bool frm = Metrics(border, lx, rx);
bool left = false;
bool right = false;
for(int i = 0; i < GetButtonCount(); i++) {
SubButton& b = Button(i);
int cx = 0; int x = 0;
GetPos(b, lx, rx, x, cx);
(b.left ? left : right) = true;
}
if(ComplexFrame()) {
r.right = r.left + rx;
r.left += lx;
}
else
if(frm) {
Rect m = minsize ? style->margin : GetMargin();
r = Rect(r.left + max(lx, m.left), r.top + m.top, min(rx, r.right - m.right), r.bottom - m.bottom);
}
else {
Rect m = style->margin;
r = Rect(r.left + max(lx, m.left), r.top + m.top, min(rx, r.right - m.right), r.bottom - m.bottom);
if(!IsTrivial() || style->trivialsep) {
if(left)
r.left++;
if(right)
r.right--;
}
}
if(!IsNull(valuecy)) {
r.top += (r.GetHeight() - valuecy) / 2;
r.bottom = r.top + valuecy;
}
}
void MultiButton::Paint(Draw& w)
{
Paint0(w, false);
}
Rect MultiButton::Paint0(Draw& w, bool getcr)
{
Size sz = GetSize();
int border, lx, rx;
bool frm = Metrics(border, lx, rx);
int mst = ChState(MAIN);
Color fpaper = Null;
Color text = SColorLabel;
bool hotpressed = false;
if(!nobg && !ComplexFrame() && frm) {
if(mst == CTRL_HOT && !IsTrivial()) {
fpaper = Blend(SColorHighlight, SColorPaper, 235);
hotpressed = true;
}
else
if(mst == CTRL_PRESSED && !IsTrivial()) {
fpaper = Blend(SColorHighlight, SColorFace, 235);
hotpressed = true;
}
else
if(HasFocus()) {
fpaper = SColorHighlight();
text = SColorHighlightText();
hotpressed = true;
}
else
fpaper = IsEnabled() && IsEditable() ? style->paper : SColorFace();
}
if(frm && !nobg && !getcr) {
if(style->clipedge) {
int l, r;
GetLR(l, r);
w.Clip(l, 0, r - l, sz.cy);
}
ChPaint(w, sz, style->edge[style->activeedge ? mst : 0]);
Color p = paper;
if(frm && style->activeedge && HasFocus())
p = SColorHighlight();
if(hotpressed && HasMain())
p = Nvl(fpaper, paper);
if(IsEnabled() && IsEditable())
p = Nvl(p, style->paper);
if(!IsNull(p) && !IsNull(style->coloredge))
ChPaint(w, sz, style->coloredge, p);
if(style->clipedge)
w.End();
}
bool left = false;
bool right = false;
for(int i = 0; i < GetButtonCount(); i++) {
SubButton& b = Button(i);
int st = ChState(i);
int x = 0, cx = 0;
GetPos(b, lx, rx, x, cx);
if(getcr)
continue;
bool dopaint = true;
Value v = b.left ? left ? style->lmiddle[st] : style->left[st]
: right ? style->rmiddle[st] : style->right[st];
if(!nobg) {
if(ComplexFrame())
ChPaint(w, x, border, cx, sz.cy - 2 * border, style->simple[st]);
else
if(frm) {
if(IsTrivial() && style->usetrivial)
dopaint = false;
ChPaint(w, x, border, cx, sz.cy - 2 * border,
dopaint ? v : style->trivial[st]);
}
else {
w.Clip(x, 0, cx, sz.cy);
ChPaint(w, sz, style->look[Frame() ? mst : st]);
if(IsNull(v) || !Frame()) {
if((!IsTrivial() || style->trivialsep) && IsEnabled() && IsEditable()) {
if(b.left) {
if(left)
ChPaint(w, x, style->sepm, 1, sz.cy - 2 * style->sepm, style->sep1);
ChPaint(w, x + cx - 1, style->sepm, 1, sz.cy - 2 * style->sepm, style->sep2);
}
else {
ChPaint(w, x, style->sepm, 1, sz.cy - 2 * style->sepm, style->sep1);
if(right)
ChPaint(w, x + cx - 1, style->sepm, 1, sz.cy - 2 * style->sepm, style->sep2);
}
}
}
else
ChPaint(w, x, 0, cx, sz.cy, v);
w.End();
}
}
if(dopaint) {
Size tsz = GetTextSize(b.label, StdFont());
Image m = tsz.cx > 0 ? b.img : (Image)Nvl(b.img, CtrlsImg::DA());
Size isz = m.GetSize();
Point p = (st == CTRL_PRESSED) * style->pressoffset;
p.x += x + (cx - isz.cx - tsz.cx - (tsz.cx > 0 && isz.cx > 0 ? DPI(LB_IMAGE) : 0)) / 2;
p.y += (sz.cy - isz.cy) / 2;
if(b.left) {
if(!left) p.x += style->loff;
}
else
if(!right) p.x += style->roff;
Color ink = frm ? style->fmonocolor[st] : style->monocolor[st];
if(b.monoimg || IsNull(b.img))
w.DrawImage(p.x, p.y, m, ink);
else
w.DrawImage(p.x, p.y, m);
if(tsz.cx > 0) {
if(isz.cx > 0)
p.x += isz.cx + DPI(LB_IMAGE);
w.DrawText(p.x, (sz.cy - tsz.cy) / 2, b.label, StdFont(), ink);
}
}
(b.left ? left : right) = true;
}
Rect r, cr;
cr = GetSize();
cr.left = lx;
cr.right = rx;
Color paper = Null;
if(!nobg) {
if(ComplexFrame()) {
r = cr;
paper = HasFocus() ? SColorHighlight() : style->paper;
if(HasFocus())
text = SColorHighlightText();
w.DrawRect(r, paper);
}
else
if(frm) {
Rect m = GetMargin();
r = Rect(max(lx, m.left), m.top, min(rx, sz.cx - m.right), sz.cy - m.bottom);
w.DrawRect(r, fpaper);
cr = r;
}
else {
w.Clip(lx, 0, rx - lx, sz.cy);
ChPaint(w, sz, style->look[mst]);
Rect m = style->margin;
r = Rect(max(lx, m.left), m.top, min(rx, sz.cx - m.right), sz.cy - m.bottom);
if(!IsTrivial() || style->trivialsep) {
if(left) {
r.left++;
if(IsEnabled() && IsEditable())
ChPaint(w, lx, style->sepm, 1, sz.cy - 2 * style->sepm, style->sep1);
}
if(right) {
if(IsEnabled() && IsEditable())
ChPaint(w, rx - 1, style->sepm, 1, sz.cy - 2 * style->sepm, style->sep2);
r.right--;
}
}
w.End();
cr = r;
}
}
cr.left += DPI(1);
Rect clr = cr;
if(!IsNull(valuecy) && cr.GetHeight() > valuecy) {
cr.top += (cr.GetHeight() - valuecy) / 2;
cr.bottom = cr.top + valuecy;
}
if(getcr)
return cr;
Value v = convert->Format(value);
bool f = HasFocus() && !push && frm;
if(cr.left < cr.right && display) {
w.Clip(clr);
if(!IsNull(error)) {
v = error;
text = style->error;
}
display->Paint(w, cr, v, IsShowEnabled() ? text : SColorDisabled, paper, f * Display::FOCUS);
w.End();
}
if(!frm && HasFocus())
DrawFocus(w, r);
return cr;
}
void MultiButton::SyncInfo()
{
if((HasMouse() || info.HasMouse()) && display &&
(GetMouseFlags() & (K_MOUSELEFT|K_MOUSERIGHT|K_MOUSEMIDDLE)) == 0) {
Point p = GetMouseViewPos();
NilDraw nw;
Rect r = Paint0(nw, true);
if(r.Contains(p)) {
Value v = convert->Format(value);
int cm = DPI(2);
r.left -= cm;
r.right += cm;
info.Set(this, r, v, display, SColorText, style->paper, 0, DPI(2));
return;
}
}
info.Cancel();
}
void MultiButton::MouseMove(Point p, dword flags)
{
int h = FindButton(p.x);
String tip = GetTextAttr(ATTR_TIP);
Ctrl::Tip(h >= 0 && h < GetButtonCount() ? Nvl(Button(h).tip, tip) : tip);
if(hl != h) {
hl = h;
Refresh();
}
bool b = flags & K_MOUSELEFT;
if(b != push) {
push = b;
Refresh();
}
SyncInfo();
}
void MultiButton::DoPush(int i)
{
if(i == 0 && droppush)
DropPush();
else
Button(i).WhenPush();
}
void MultiButton::LeftDown(Point p, dword flags)
{
push = true;
Refresh();
if(hl == NONE)
pushrect.Clear();
else {
if(hl == MAIN)
pushrect = GetScreenRect();
else {
int x, cx;
int border = IsTrivial() ? style->trivialborder : style->border;
GetPos(hl, x, cx);
pushrect = RectC(x, border, cx, GetSize().cy - 2 * border)
.Offseted(GetScreenView().TopLeft());
}
Sync();
if(hl >= 0)
DoPush(hl);
else
MainPush();
}
SyncInfo();
}
void MultiButton::LeftUp(Point p, dword flags)
{
push = false;
Refresh();
Sync();
if(hl != NONE) {
if(hl >= 0)
Button(hl).WhenClick();
else
MainClick();
}
SyncInfo();
}
void MultiButton::MouseLeave()
{
if(!info.IsOpen()) {
hl = NONE;
Refresh();
SyncInfo();
}
}
void MultiButton::CancelMode()
{
hl = NONE;
push = false;
Refresh();
info.Cancel();
}
bool MultiButton::IsTrivial() const
{
return GetButtonCount() == 1 && IsNull(Button(0).img) && !HasMain();
}
MultiButton& MultiButton::SetDisplay(const Display& d)
{
if(display != &d) {
display = &d;
Refresh();
}
return *this;
}
MultiButton& MultiButton::NoDisplay()
{
if(display) {
display = NULL;
Refresh();
}
return *this;
}
MultiButton& MultiButton::SetConvert(const Convert& c)
{
if(convert != &c) {
convert = &c;
Refresh();
}
return *this;
}
MultiButton& MultiButton::Set(const Value& v, bool update)
{
if(value != v) {
value = v;
if(update)
UpdateRefresh();
else
Refresh();
}
return *this;
}
MultiButton& MultiButton::SetValueCy(int cy)
{
if(cy != valuecy) {
valuecy = cy;
Refresh();
if(Frame() && GetParent())
GetParent()->RefreshLayout();
}
return *this;
}
void MultiButton::SetData(const Value& v)
{
if(value != v) {
value = v;
UpdateRefresh();
}
SyncInfo();
}
Value MultiButton::GetData() const
{
return value;
}
void MultiButton::PseudoPush(int bi)
{
hl = bi;
push = true;
DoPush(bi);
Sync();
Sleep(50);
Button(bi).WhenClick();
hl = NONE;
push = false;
Sync();
}
void MultiButton::PseudoPush()
{
hl = MAIN;
MainPush();
Sync();
Sleep(50);
MainClick();
hl = NONE;
push = false;
Sync();
}
bool MultiButton::HasMain() const
{
return WhenPush || WhenClick;
}
void MultiButton::MainPush()
{
WhenPush();
}
void MultiButton::MainClick()
{
WhenClick();
}
void MultiButton::DropPush() {}
MultiButton::MultiButton()
{
Transparent();
SetStyle(StyleDefault());
display = &StdDisplay();
convert = &NoConvert();
valuecy = Null;
push = false;
SetFrame(sNullFrame());
nobg = false;
hl = NONE;
droppush = false;
}
//
void MultiButtonFrame::FrameAdd(Ctrl& parent)
{
parent.Add(*this);
NoWantFocus();
}
void MultiButtonFrame::FrameRemove()
{
Remove();
WantFocus();
}
void MultiButtonFrame::FrameLayout(Rect& r)
{
SetFrameRect(r);
Lay(r);
}
void MultiButtonFrame::FrameAddSize(Size& sz)
{
Rect r(0, 0, 100, 100);
Lay(r, true);
sz.cx += r.left + 100 - r.right;
sz.cy += r.top + 100 - r.bottom;
}
bool MultiButtonFrame::Frame()
{
return true;
}
void MultiButtonFrame::AddTo(Ctrl& w)
{
w.SetFrame(sNullFrame());
w.AddFrame(*this);
}
}