mirror of
https://github.com/ultimatepp/ultimatepp.git
synced 2026-05-16 14:16:09 -06:00
301 lines
6.6 KiB
C++
301 lines
6.6 KiB
C++
#include "DockTabBar.h"
|
|
#include "DockCont.h"
|
|
|
|
#define ANIM_SPEED 10
|
|
#define ANIM_FRAMES 10
|
|
|
|
NAMESPACE_UPP
|
|
|
|
// DockTabBar
|
|
int AutoHideBar::autohide_timeout = 1000;
|
|
|
|
void DockTabBar::FrameLayout(Rect& r)
|
|
{
|
|
if (IsAutoHide()) return Hide();
|
|
Show();
|
|
TabBar::FrameLayout(r);
|
|
}
|
|
|
|
void DockTabBar::FrameAddSize(Size& sz)
|
|
{
|
|
if (!IsAutoHide())
|
|
TabBar::FrameAddSize(sz);
|
|
}
|
|
|
|
Size DockTabBar::GetStdSize(const Tab &t)
|
|
{
|
|
DockableCtrl *d;
|
|
const Value &q = t.key;
|
|
Value v;
|
|
if (IsTypeRaw<DockCont *>(q)) {
|
|
DockCont *c = ValueTo<DockCont *>(q);
|
|
d = &c->GetCurrent();
|
|
v = c->GetTitle();
|
|
}
|
|
else {
|
|
ASSERT(IsTypeRaw<DockableCtrl *>(q));
|
|
d = ValueTo<DockableCtrl *>(q);
|
|
v = d->GetTitle();
|
|
}
|
|
int isz = (IsVert() ? d->GetIcon().GetHeight() : d->GetIcon().GetWidth());
|
|
return showtext ? (TabBar::GetStdSize(v) + Size(isz+2, 0)) : Size(isz, isz);
|
|
}
|
|
|
|
void DockTabBar::RightDown(Point p, dword keyflags)
|
|
{
|
|
WhenContext(GetHighlight());
|
|
}
|
|
|
|
void DockTabBar::LeftDrag(Point p, dword keyflags)
|
|
{
|
|
if (keyflags & K_CTRL || keyflags & K_SHIFT)
|
|
return TabBar::LeftDrag(p, keyflags &= ~K_SHIFT);
|
|
if (GetHighlight() >= 0)
|
|
WhenDrag(GetHighlight());
|
|
}
|
|
|
|
DockTabBar::DockTabBar()
|
|
{
|
|
autohide = -1;
|
|
icons = true;
|
|
showtext = true;
|
|
AutoScrollHide().InactiveDisabled();
|
|
}
|
|
|
|
// AutoHide bar
|
|
void AutoHideBar::MouseEnter(Point p, dword keyflags)
|
|
{
|
|
if (ctrl && ctrl->IsOpen())
|
|
KillTimeCallback(TIMEID_HIDE_TIMEOUT);
|
|
DockTabBar::MouseEnter(p, keyflags);
|
|
}
|
|
|
|
void AutoHideBar::MouseLeave()
|
|
{
|
|
if (ctrl && ctrl->IsOpen())
|
|
KillSetTimeCallback(autohide_timeout, THISBACK1(HideAnimate, ctrl), TIMEID_HIDE_TIMEOUT);
|
|
DockTabBar::MouseLeave();
|
|
}
|
|
|
|
void AutoHideBar::AddCtrl(DockCont& c, const String& group)
|
|
{
|
|
TabBar::Add(RawToValue<DockCont *>(&c), Null, group);
|
|
if (GetCount() == autohide+1)
|
|
RefreshParentLayout();
|
|
else
|
|
Refresh();
|
|
}
|
|
|
|
void AutoHideBar::RemoveCtrl(DockCont& c, int ix)
|
|
{
|
|
if (c.IsOpen()) c.Close();
|
|
|
|
if (ctrl == &c) {
|
|
if (popup.IsOpen())
|
|
popup.Close();
|
|
ctrl->Remove();
|
|
//KillTimeCallback(TIMEID_ANIMATE);
|
|
KillTimeCallback(TIMEID_HIDE_TIMEOUT);
|
|
ctrl = NULL;
|
|
}
|
|
if (ix >= 0) DockTabBar::Close(ix);
|
|
if (GetCount() == autohide)
|
|
RefreshParentLayout();
|
|
}
|
|
|
|
void AutoHideBar::RemoveCtrl(int ix)
|
|
{
|
|
if (ix >= 0) {
|
|
DockCont *c = GetCtrl(ix);
|
|
ASSERT(c);
|
|
RemoveCtrl(*c, ix);
|
|
}
|
|
}
|
|
|
|
void AutoHideBar::TabHighlight()
|
|
{
|
|
Ctrl *c = NULL;
|
|
int ix = TabBar::GetHighlight();
|
|
if (ix >= 0)
|
|
c = GetCtrl(ix);
|
|
if (!c || ctrl == c)
|
|
return;
|
|
else if (ctrl) {
|
|
if (c) {
|
|
if (popup.IsPopUp())
|
|
popup.Close();
|
|
ctrl->Remove();
|
|
ctrl = NULL;
|
|
}
|
|
else
|
|
HideAnimate(ctrl);
|
|
}
|
|
if (c) {
|
|
ASSERT(ix >= 0 && ix < GetCount());
|
|
// Clear WhenHighlight ot prevent stack overflow. Perhaps a better solution should be found...
|
|
Callback cb = WhenHighlight;
|
|
WhenHighlight = Callback();
|
|
SetCursor(ix);
|
|
ShowAnimate(c);
|
|
WhenHighlight = cb;
|
|
}
|
|
}
|
|
|
|
void AutoHideBar::TabClose(Value v)
|
|
{
|
|
DockCont &dc = *(ValueTo<DockCont *>(v));
|
|
RemoveCtrl(dc, -1);
|
|
dc.StateNotDocked();
|
|
dc.SignalStateChange();
|
|
if (GetCount() == autohide)
|
|
RefreshParentLayout();
|
|
}
|
|
|
|
void AutoHideBar::ShowAnimate(Ctrl *c)
|
|
{
|
|
if (c == ctrl) return;
|
|
if (popup.IsPopUp()) popup.Close();
|
|
|
|
Rect target = Ctrl::GetScreenRect();
|
|
Size sz = c->GetStdSize();
|
|
switch (GetAlign()) {
|
|
case DockTabBar::LEFT:
|
|
sz.cy = target.Height();
|
|
target.left = target.right;
|
|
break;
|
|
case DockTabBar::TOP:
|
|
sz.cx = target.Width();
|
|
target.top = target.bottom;
|
|
break;
|
|
case DockTabBar::RIGHT:
|
|
sz.cy = target.Height();
|
|
target.right = target.left;
|
|
break;
|
|
case DockTabBar::BOTTOM:
|
|
sz.cx = target.Width();
|
|
target.bottom = target.top;
|
|
break;
|
|
};
|
|
if (IsVert())
|
|
sz.cx = min(sz.cx, GetParent()->GetSize().cx / 2);
|
|
else
|
|
sz.cy = min(sz.cy, GetParent()->GetSize().cy / 2);
|
|
|
|
Rect init = target;
|
|
AdjustSize(init, Size(5, 5));
|
|
AdjustSize(target, sz);
|
|
|
|
c->SetRect(sz);
|
|
c->SizePos();
|
|
popup << *(ctrl = c);
|
|
c->Show();
|
|
|
|
popup.SetRect(target);
|
|
popup.PopUp(GetParent(), false, true, false, false);
|
|
Animate(popup, target, GUIEFFECT_SLIDE);
|
|
}
|
|
|
|
void AutoHideBar::HideAnimate(Ctrl *c)
|
|
{
|
|
ASSERT(ctrl);
|
|
// If the popup has a child popup active then reset timer and keep the popup visible
|
|
Vector<Ctrl *> wins = Ctrl::GetTopCtrls();
|
|
for(int i = 0; i < wins.GetCount(); i++) {
|
|
if (wins[i]->IsPopUp() && wins[i]->GetOwner() == &popup) {
|
|
KillSetTimeCallback(autohide_timeout, THISBACK1(HideAnimate, ctrl), TIMEID_HIDE_TIMEOUT);
|
|
return;
|
|
}
|
|
}
|
|
DockTabBar::KillCursor();
|
|
#ifdef PLATFORM_WIN32
|
|
Rect r = popup.GetRect();
|
|
AdjustSize(r, -r.GetSize());
|
|
Animate(popup, r, GUIEFFECT_SLIDE);
|
|
#endif
|
|
popup.Close();
|
|
ctrl->Remove();
|
|
ctrl = NULL;
|
|
}
|
|
|
|
void AutoHideBar::AdjustSize(Rect& r, const Size& sz)
|
|
{
|
|
switch (DockTabBar::GetAlign()) {
|
|
case DockTabBar::LEFT:
|
|
r.right += sz.cx;
|
|
break;
|
|
case DockTabBar::TOP:
|
|
r.bottom += sz.cy;
|
|
break;
|
|
case DockTabBar::RIGHT:
|
|
r.left -= sz.cx;
|
|
break;
|
|
case DockTabBar::BOTTOM:
|
|
r.top -= sz.cy;
|
|
break;
|
|
};
|
|
}
|
|
|
|
int AutoHideBar::FindCtrl(const DockCont& c) const
|
|
{
|
|
if (&c == ctrl) return GetCursor();
|
|
for (int i = 0; i < GetCount(); i++) {
|
|
DockCont *v = ValueTo<DockCont *>(GetKey(i));
|
|
if (v == &c) return i;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
void AutoHideBar::TabDrag(int ix)
|
|
{
|
|
GetCtrl(ix)->MoveBegin();
|
|
}
|
|
|
|
void AutoHideBar::ComposeTab(Tab& tab, const Font &font, Color ink, int style)
|
|
{
|
|
DockableCtrl *d;
|
|
WString txt;
|
|
const Value &q = tab.value;
|
|
|
|
ink = (style == CTRL_DISABLED) ? SColorDisabled : ink;
|
|
|
|
if (IsTypeRaw<DockCont *>(q)) {
|
|
DockCont *c = ValueTo<DockCont *>(q);
|
|
d = &c->GetCurrent();
|
|
txt = c->GetTitle();
|
|
}
|
|
else {
|
|
ASSERT(IsTypeRaw<DockableCtrl *>(q));
|
|
d = ValueTo<DockableCtrl *>(q);
|
|
txt = d->GetTitle();
|
|
}
|
|
|
|
if(icons)
|
|
{
|
|
tab.AddImage((style == CTRL_DISABLED) ? DisabledImage(d->GetIcon()) : d->GetIcon());
|
|
}
|
|
if (showtext)
|
|
{
|
|
tab.AddText(txt, font, ink);
|
|
}
|
|
}
|
|
|
|
AutoHideBar::AutoHideBar()
|
|
{
|
|
ctrl = NULL;
|
|
AutoHideMin(0).InactiveDisabled().AllowNullCursor().MinTabCount(0);
|
|
popup.WhenEnter = THISBACK2(MouseEnter, Point(0, 0), 0);
|
|
popup.WhenLeave = THISBACK(MouseLeave);
|
|
DockTabBar::WhenHighlight = THISBACK(TabHighlight);
|
|
WhenClose = THISBACK(TabClose);
|
|
WhenDrag = THISBACK(TabDrag);
|
|
}
|
|
|
|
void AutoHideBar::HidePopup::ChildMouseEvent(Ctrl *child, int event, Point p, int zdelta, dword keyflags)
|
|
{
|
|
if (event == MOUSELEAVE)
|
|
WhenLeave();
|
|
else if (event != MOUSEMOVE)
|
|
WhenEnter();
|
|
}
|
|
|
|
END_UPP_NAMESPACE
|