ultimatepp/uppsrc/CtrlLib/EditField.cpp
mdelfede 263ff5f895 changed svn layout
git-svn-id: svn://ultimatepp.org/upp/trunk@281 f0d560ea-af0d-0410-9eb7-867de7ffcac7
2008-06-07 22:31:27 +00:00

1130 lines
20 KiB
C++

#include "CtrlLib.h"
NAMESPACE_UPP
CH_VALUE(ViewEdge, CtrlsImg::VE());
Value EditFieldEdge()
{
return EditField::StyleDefault().edge[0];
}
CtrlFrame& EditFieldFrame()
{
static LookFrame m(EditFieldEdge);
return m;
}
CtrlFrame& ViewFrame()
{
static LookFrame m(ViewEdge);
return m;
}
bool IsWCh(int c)
{
return IsLeNum(c) || c == '_';
}
bool TextArrayOps::GetWordSelection(int c, int& l, int& h)
{
if(IsWCh(GetChar(c))) {
l = h = c;
while(l > 0 && IsWCh(GetChar(l - 1)))
l--;
while(h < GetLength() && IsWCh(GetChar(h)))
h++;
if(h != c)
while(h < GetLength() && GetChar(h) == ' ')
h++;
return true;
}
return false;
}
int TextArrayOps::GetNextWord(int cursor)
{
bool a = IsWCh(GetChar(cursor));
int n = 0;
int c = cursor;
while(c <= GetLength() && IsWCh(GetChar(c)) == a) {
if(++n > 10000) return cursor;
c++;
}
return c;
}
int TextArrayOps::GetPrevWord(int cursor)
{
int n = 0;
int c = cursor;
if(c == 0) return 0;
bool a = IsWCh(GetChar(c - 1));
while(c > 0 && IsWCh(GetChar(c - 1)) == a) {
if(++n > 10000) return cursor;
c--;
}
return c;
}
void LookFrame::FrameLayout(Rect& r)
{
Rect m = ChMargins(Get());
r.left += m.left;
r.right -= m.right;
r.top += m.top;
r.bottom -= m.bottom;
}
void LookFrame::FramePaint(Draw& w, const Rect& r)
{
ChPaintEdge(w, r, Get());
}
void LookFrame::FrameAddSize(Size& sz)
{
Rect m = ChMargins(Get());
sz.cx += m.left + m.right;
sz.cy += m.top + m.bottom;
}
void ActiveEdgeFrame::FrameLayout(Rect& r)
{
Rect m = ChMargins(edge[0]);
r.left += m.left;
r.right -= m.right;
r.top += m.top;
r.bottom -= m.bottom;
}
void ActiveEdgeFrame::FramePaint(Draw& w, const Rect& r)
{
int i = 0;
if(ctrl) {
i = !ctrl->IsEnabled() ? CTRL_DISABLED
: ctrl->HasFocus() ? CTRL_PRESSED
: mousein ? CTRL_HOT
: CTRL_NORMAL;
}
ChPaintEdge(w, r, edge[i]);
}
void ActiveEdgeFrame::FrameAddSize(Size& sz)
{
Rect m = ChMargins(edge[0]);
sz.cx += m.left + m.right;
sz.cy += m.top + m.bottom;
}
void ActiveEdgeFrame::Set(const Ctrl *ctrl_, const Value *edge_, bool active)
{
ctrl = active ? ctrl_ : NULL;
edge = edge_;
}
CH_STYLE(EditField, Style, StyleDefault)
{
paper = SColorPaper();
disabled = SColorFace();
focus = paper;
invalid = Blend(paper, Color(255, 0, 0), 32);
text = SColorText();
textdisabled = SColorDisabled();
selected = SColorHighlight();
selectedtext = SColorHighlightText();
for(int i = 0; i < 4; i++)
edge[i] = CtrlsImg::EFE();
activeedge = false;
}
bool EditField::FrameIsEdge()
{
return &GetFrame() == &edge;
}
void EditField::SyncEdge()
{
if(FrameIsEdge() && style->activeedge)
RefreshFrame();
}
void EditField::MouseEnter(Point p, dword keyflags)
{
edge.Mouse(true);
SyncEdge();
}
void EditField::MouseLeave()
{
edge.Mouse(false);
SyncEdge();
}
EditField& EditField::SetStyle(const Style& s)
{
style = &s;
edge.Set(this, style->edge, style->activeedge);
RefreshLayout();
RefreshFrame();
return *this;
}
void EditField::CancelMode()
{
keep_selection = false;
selclick = false;
dropcaret.Clear();
}
int EditField::GetTextCx(const wchar *txt, int n, bool password, Font fnt)
{
FontInfo fi = fnt.Info();
if(password)
return n * fi['*'];
const wchar *s = txt;
int x = 0;
while(n--)
x += fi[*s++];
return x;
}
int EditField::GetCaret(int cursor)
{
return GetTextCx(text, cursor, password, font);
}
int EditField::GetViewHeight(Font font)
{
Size sz = Size(font.Info().GetHeight() * 4, font.Info().GetHeight() + 4);
return sz.cy;
}
int EditField::GetStdHeight(Font font)
{
Size sz = Size(font.Info().GetHeight() * 4, font.Info().GetHeight() + 4);
EditFieldFrame().FrameAddSize(sz);
return sz.cy;
}
Size EditField::GetMinSize() const
{
return AddFrameSize(10, font.Info().GetHeight() + 4);
}
int EditField::GetCursor(int posx)
{
FontInfo fi = font.Info();
const wchar *s = text;
posx -= 2;
int x = 0;
int x0 = 0;
int i = 0;
for(i = 0; i < text.GetLength(); i++) {
x0 = x;
x += fi[*s];
if(posx < (x + x0) / 2)
return i;
s++;
}
return posx > (x + x0) / 2 ? i : max(i - 1, 0);
}
Image EditField::CursorImage(Point, dword)
{
return Image::IBeam();
}
int EditField::GetTy()
{
return (GetSize().cy + 1 - font.Info().GetHeight()) / 2;
}
void EditField::Paints(Draw& w, int& x, int fcy, const wchar *&txt,
Color ink, Color paper, int n, bool password, Font fnt)
{
if(n < 0) return;
int cx = GetTextCx(txt, n, password, font);
w.DrawRect(x, 0, cx, fcy, paper);
if(password) {
String h;
h.Cat('*', n);
w.DrawText(x, 0, ~h, fnt, ink, n);
}
else
w.DrawText(x, 0, txt, fnt, ink, n);
txt += n;
x += cx;
}
void EditField::Paint(Draw& w)
{
Size sz = GetSize();
const EditField::Style *st = style ? style : &StyleDefault();
bool enabled = IsShowEnabled();
Color paper = enabled && !IsReadOnly() ? (HasFocus() ? st->focus : st->paper) : st->disabled;
if(nobg)
paper = Null;
Color ink = enabled ? st->text : st->textdisabled;
if(convert && enabled && convert->Scan(text).IsError())
paper = st->invalid;
int fcy = font.Info().GetHeight();
int yy = GetTy();
w.DrawRect(0, 0, 2, sz.cy, paper);
w.DrawRect(0, 0, sz.cx, yy, paper);
w.DrawRect(0, yy + fcy, sz.cx, sz.cy - yy - fcy, paper);
w.DrawRect(sz.cx - 2, 0, 2, sz.cy, paper);
w.Clipoff(2, yy, sz.cx - 4, fcy);
int x = -sc;
bool ar = alignright && !HasFocus();
if(IsNull(text) && !IsNull(nulltext)) {
const wchar *txt = nulltext;
Paints(w, x, fcy, txt, nullink, paper, nulltext.GetLength(), false, nullfont);
}
else {
const wchar *txt = text;
if(ar) {
x = sz.cx - 4 - GetTextCx(text, text.GetLength(), password, font);
w.DrawRect(0, 0, x, fcy, paper);
}
int l, h;
if(GetSelection(l, h)) {
Paints(w, x, fcy, txt, ink, paper, l, password, font);
Paints(w, x, fcy, txt, enabled ? st->selectedtext : paper,
enabled ? st->selected : ink, h - l, password, font);
Paints(w, x, fcy, txt, ink, paper, text.GetLength() - h, password, font);
}
else
Paints(w, x, fcy, txt, ink, paper, text.GetLength(), password, font);
}
if(!ar)
w.DrawRect(x, 0, 9999, fcy, paper);
w.End();
DrawTiles(w, dropcaret, CtrlImg::checkers());
}
bool EditField::GetSelection(int& l, int& h) const
{
if(anchor < 0 || anchor == cursor) {
l = h = cursor;
return false;
}
if(anchor < cursor) {
l = anchor;
h = cursor;
}
else {
l = cursor;
h = anchor;
}
return true;
}
bool EditField::IsSelection() const
{
return anchor >= 0 && anchor != cursor;
}
void EditField::SyncCaret()
{
FontInfo fi = font.Info();
SetCaret(GetCaret(cursor) - sc + 2 - fi.GetRightSpace('o') + fi.GetLeftSpace('o'), GetTy(),
1, min(GetSize().cy - 2 * GetTy(), fi.GetHeight()));
}
void EditField::Finish(bool refresh)
{
if(anchor > text.GetLength()) anchor = text.GetLength();
if(cursor > text.GetLength()) cursor = text.GetLength();
if(cursor < 0) cursor = 0;
Size sz = GetSize();
if(autosize) {
Rect r = GetRect();
int mw = min(r.Width(), Draw::GetStdFontSize().cx);
sz.cx = GetCaret(text.GetLength()) + 4;
sz = AddFrameSize(sz);
if(GetParent())
sz.cx = min(sz.cx, GetParent()->GetSize().cx - r.left);
sz.cx = minmax(sz.cx, mw, autosize);
if(sz.cx != r.Width())
LeftPos(r.left, sz.cx);
sz = GetSize();
}
sz.cx -= 2;
if(sz.cx <= 0) return;
int x = GetCaret(cursor);
int wx = x + font.Info().GetRightSpace('o');
if(wx > sz.cx + sc - 1) {
sc = wx - sz.cx + 1;
Refresh();
}
if(x < sc) {
sc = x;
Refresh();
}
if(refresh)
Refresh();
SyncCaret();
}
void EditField::Layout()
{
Ctrl::Layout();
sc = 0;
Finish();
}
void EditField::GotFocus()
{
if(autoformat && IsEditable() && !IsNull(text) && inactive_convert) {
Value v = convert->Scan(text);
if(!v.IsError()) {
WString s = convert->Format(v);
if(s != text) text = s;
}
}
if(!keep_selection) {
anchor = 0;
cursor = text.GetLength();
}
Finish();
SyncEdge();
}
void EditField::LostFocus()
{
if(autoformat && IsEditable() && !IsNull(text)) {
Value v = convert->Scan(text);
if(!v.IsError()) {
const Convert * cv = inactive_convert ? inactive_convert : convert;
WString s = cv->Format(v);
if(s != text) text = s;
}
}
if(!keep_selection) {
anchor = -1;
cursor = sc = 0;
}
Refresh();
SyncEdge();
}
void EditField::LeftDown(Point p, dword flags)
{
int c = GetCursor(p.x + sc);
if(!HasFocus()) {
SetFocus();
if(clickselect) {
SetSelection();
Finish();
return;
}
sc = 0;
Move(c);
}
int l, h;
selclick = false;
if(GetSelection(l, h) && c >= l && c < h) {
selclick = true;
return;
}
SetCapture();
Move(c, flags & K_SHIFT);
Finish();
}
void EditField::MiddleDown(Point p, dword flags)
{
if(IsReadOnly())
return;
if(AcceptText(Selection())) {
WString w = GetWString(Selection());
selclick = false;
LeftDown(p, flags);
Insert(w);
Action();
Finish();
}
}
void EditField::LeftUp(Point p, dword flags)
{
int c = GetCursor(p.x + sc);
int l, h;
if(GetSelection(l, h) && c >= l && c < h && !HasCapture() && selclick)
Move(c, false);
Finish();
selclick = false;
}
void EditField::LeftDouble(Point p, dword flags)
{
int l, h;
if(GetWordSelection(cursor, l, h))
SetSelection(l, h);
}
void EditField::LeftTriple(Point p, dword keyflags)
{
anchor = 0;
cursor = text.GetLength();
Finish();
}
void EditField::MouseMove(Point p, dword flags)
{
if(!HasCapture()) return;
Move(GetCursor(p.x + sc), true);
Finish();
}
void EditField::SaveUndo()
{
undotext = text;
undoanchor = anchor;
undocursor = cursor;
}
void EditField::Move(int newpos, bool select)
{
bool refresh = anchor >= 0;
if(select) {
if(anchor < 0) anchor = cursor;
refresh = true;
}
else
anchor = -1;
cursor = newpos;
Finish(refresh);
if(select)
SetSelectionSource(ClipFmtsText());
}
void EditField::SetSelection(int l, int h)
{
if(l < h) {
anchor = max(l, 0);
cursor = min(h, text.GetLength());
}
else {
cursor = l;
anchor = -1;
}
Finish();
}
void EditField::CancelSelection()
{
int l, h;
if(GetSelection(l, h)) {
cursor = l;
anchor = -1;
sc = 0;
Finish();
}
}
bool EditField::RemoveSelection()
{
int l, h;
if(!GetSelection(l, h)) {
anchor = -1;
return false;
}
SaveUndo();
Remove(l, h - l);
cursor = l;
anchor = -1;
sc = 0;
return true;
}
void EditField::Copy()
{
int l, h;
if(password) return;
if(!GetSelection(l, h)) {
l = 0;
h = text.GetLength();
}
WriteClipboardUnicodeText(text.Mid(l, h - l));
}
int EditField::Insert(int pos, const WString& itext)
{
if(IsReadOnly()) return 0;
WString ins;
const wchar *s = itext;
for(;;) {
wchar chr = *s++;
if(chr == '\t')
ins.Cat(WString(' ', 4));
else
if(chr >= ' ') {
chr = (*filter)(chr);
if(chr) {
chr = convert->Filter(chr);
if(chr && (charset == CHARSET_UNICODE || FromUnicode(chr, charset, 0)))
ins.Cat(chr);
}
}
else
break;
}
if(ins.GetCount() + text.GetCount() > maxlen) {
BeepExclamation();
return 0;
}
text.Insert(pos, ins);
Update();
return ins.GetLength();
}
void EditField::Remove(int pos, int n)
{
if(IsReadOnly()) return;
text.Remove(pos, n);
Update();
}
void EditField::Insert(int chr)
{
if(IsReadOnly()) return;
if(initcaps && cursor == 0 && text.GetCount() == 0)
chr = ToUpper(chr);
cursor += Insert(cursor, WString(chr, 1));
Finish();
}
void EditField::Insert(const WString& s)
{
if(!RemoveSelection()) SaveUndo();
cursor += Insert(cursor, s);
Finish();
}
void EditField::DragAndDrop(Point p, PasteClip& d)
{
if(IsReadOnly()) return;
int c = GetCursor(p.x + sc);
if(AcceptText(d)) {
SaveUndo();
int sell, selh;
if(GetSelection(sell, selh)) {
if(c >= sell && c < selh) {
RemoveSelection();
if(IsDragAndDropSource())
d.SetAction(DND_COPY);
c = sell;
}
else
if(d.GetAction() == DND_MOVE && IsDragAndDropSource()) {
if(c > sell)
c -= selh - sell;
RemoveSelection();
d.SetAction(DND_COPY);
}
}
int count = Insert(c, GetWString(d));
SetFocus();
SetSelection(c, c + count);
Action();
return;
}
if(!d.IsAccepted()) return;
Rect dc(0, 0, 0, 0);
if(c >= 0) {
FontInfo fi = font.Info();
int x = GetCaret(c);
dc = RectC(x - sc + 2 - fi.GetRightSpace('o'), GetTy(),
1, min(GetSize().cy - 2 * GetTy(), fi.GetHeight()));
}
if(dc != dropcaret) {
Refresh(dropcaret);
dropcaret = dc;
Refresh(dropcaret);
}
}
void EditField::DragRepeat(Point p)
{
if(IsReadOnly())
return;
Size sz = GetSize();
int sd = min(sz.cx / 6, 16);
int d = 0;
if(p.x < sd)
d = -3;
if(p.x > sz.cx - sd)
d = 3;
int a = minmax((int)sc + minmax(d, -16, 16), 0, max(0, GetCaret(GetLength()) - sz.cx + 2));
if(a != sc) {
sc = a;
Refresh();
SyncCaret();
}
}
void EditField::DragLeave()
{
Refresh(dropcaret);
dropcaret.Clear();
}
void EditField::LeftDrag(Point p, dword flags)
{
int c = GetCursor(p.x + sc);
Size ssz = StdSampleSize();
int sell, selh;
if(!HasCapture() && GetSelection(sell, selh) && c >= sell && c <= selh) {
WString sel = text.Mid(sell, selh - sell);
ImageDraw iw(ssz);
iw.DrawText(0, 0, sel);
iw.Alpha().DrawRect(ssz, Black);
iw.Alpha().DrawText(0, 0, sel, StdFont(), White);
VectorMap<String, ClipData> data;
Append(data, sel);
if(DoDragAndDrop(data, iw) == DND_MOVE) {
CancelSelection();
SaveUndo();
Remove(sell, selh - sell);
sc = 0;
Finish();
Action();
}
}
}
String EditField::GetSelectionData(const String& fmt) const
{
int sell, selh;
if(GetSelection(sell, selh))
return GetTextClip(text.Mid(sell, selh - sell), fmt);
return String();
}
void EditField::Undo()
{
Swap(undotext, text);
Swap(undoanchor, anchor);
Swap(undocursor, cursor);
anchor = -1;
UpdateAction();
Finish();
}
void EditField::Cut()
{
Copy();
RemoveSelection();
Action();
Finish();
}
void EditField::Paste()
{
Insert(ReadClipboardUnicodeText());
Action();
Finish();
}
void EditField::Erase()
{
if(!IsSelection())
SelectAll();
RemoveSelection();
Finish();
}
void EditField::SelectAll()
{
SetSelection();
Finish();
}
void EditField::MenuBar(Bar& menu) {
menu.Add(t_("Undo"), THISBACK(Undo))
.Key(K_ALT_BACKSPACE)
.Key(K_CTRL_Z);
menu.Separator();
menu.Add(IsEditable() && IsSelection(),
t_("Cut"), CtrlImg::cut(), THISBACK(Cut))
.Key(K_SHIFT_DELETE)
.Key(K_CTRL_X);
menu.Add(IsSelection(),
t_("Copy"), CtrlImg::copy(), THISBACK(Copy))
.Key(K_CTRL_INSERT)
.Key(K_CTRL_C);
menu.Add(IsEditable()
#ifdef PLATFORM_WIN32
&& ::IsClipboardFormatAvailable(CF_TEXT)
#endif
,
t_("Paste"), CtrlImg::paste(), THISBACK(Paste))
.Key(K_SHIFT_INSERT)
.Key(K_CTRL_V);
menu.Add(IsEditable(),
t_("Erase"), CtrlImg::remove(), THISBACK(Erase))
.Key(K_DELETE);
menu.Separator();
menu.Add(GetLength(),
t_("Select all"), THISBACK(SelectAll))
.Key(K_CTRL_A);
}
void EditField::RightDown(Point p, dword keyflags)
{
keep_selection = true;
MenuBar::Execute(THISBACK(MenuBar));
SetFocus();
keep_selection = false;
}
bool EditField::Key(dword key, int rep)
{
int q;
bool h;
String s;
bool select = key & K_SHIFT;
switch(key & ~K_SHIFT) {
case K_LEFT:
Move(cursor - 1, select);
return true;
case K_CTRL_LEFT:
Move(GetPrevWord(cursor), select);
return true;
case K_CTRL_RIGHT:
Move(GetNextWord(cursor), select);
return true;
case K_RIGHT:
Move(cursor + 1, select);
return true;
case K_HOME:
Move(0, select);
return true;
case K_END:
Move(text.GetLength(), select);
return true;
case K_CTRL_C:
case K_CTRL_INSERT:
Copy();
return true;
case K_CTRL_A:
SetSelection();
return true;
}
if(!IsEditable())
return false;
switch(key) {
case K_CTRL_X:
case K_SHIFT_DELETE:
Cut();
return true;
case K_CTRL_V:
case K_SHIFT_INSERT:
Paste();
return true;
case K_CTRL_Z:
case K_ALT_BACKSPACE:
Undo();
return true;
case K_BACKSPACE:
case K_SHIFT|K_BACKSPACE:
if(RemoveSelection()) {
Action();
break;
}
if(cursor == 0 || IsReadOnly()) return true;
SaveUndo();
cursor--;
Remove(cursor, 1);
Action();
break;
case K_CTRL_BACKSPACE:
if(RemoveSelection()) {
Action();
break;
}
if(cursor == 0 || IsReadOnly()) return true;
SaveUndo();
q = cursor;
h = IsWCh(text[--cursor]);
while(cursor > 0 && IsWCh(text[cursor - 1]) == h)
cursor--;
Remove(cursor, q - cursor);
Action();
break;
case K_DELETE:
if(RemoveSelection()) {
Action();
break;
}
if(cursor >= text.GetLength()) return true;
SaveUndo();
Remove(cursor, 1);
Action();
break;
case K_CTRL_DELETE:
if(RemoveSelection()) {
Action();
break;
}
if(cursor >= text.GetLength()) return true;
q = cursor;
h = IsWCh(text[q]);
while(IsWCh(text[q]) == h && q < text.GetLength()) q++;
SaveUndo();
Remove(cursor, q - cursor);
Action();
break;
default:
if(key >= ' ' && key < 65536 || key == K_SHIFT_SPACE) {
if(!RemoveSelection()) SaveUndo();
while(rep--)
Insert(key == K_SHIFT_SPACE ? ' ' : key);
Action();
return true;
}
else
return false;
}
Finish();
return true;
}
void EditField::SetText(const WString& txt)
{
if(text == txt) {
Update();
return;
}
text = txt;
sc = 0;
if(HasFocus()) {
cursor = txt.GetLength();
anchor = 0;
}
else {
cursor = 0;
anchor = -1;
}
Update();
Finish();
}
void EditField::SetData(const Value& data)
{
const Convert * cv = convert;
if(!HasFocus() && inactive_convert)
cv = inactive_convert;
SetText((WString) cv->Format(data));
}
Value EditField::GetData() const
{
return convert->Scan(text);
}
void EditField::Clear()
{
SetText(WString());
sc = cursor = 0;
}
void EditField::Reset()
{
Clear();
ClearModify();
sc = 0;
cursor = 0;
anchor = -1;
password = false;
autoformat = true;
clickselect = false;
filter = CharFilterUnicode;
convert = &NoConvert();
inactive_convert = NULL;
initcaps = false;
maxlen = INT_MAX;
autosize = false;
keep_selection = false;
nobg = false;
charset = CHARSET_UNICODE;
alignright = false;
SetStyle(StyleDefault());
SetFrame(edge);
}
EditField& EditField::SetFont(Font _font)
{
font = _font;
Refresh();
return *this;
}
EditField& EditField::NullText(const char *text, Font fnt, Color ink)
{
nulltext = text;
nullink = ink;
nullfont = fnt;
Refresh();
return *this;
}
EditField& EditField::NullText(const char *text, Color ink)
{
return NullText(text, GetFont(), ink);
}
EditField::EditField()
{
Unicode();
Reset();
}
EditField::~EditField() {}
void EditIntSpin::Inc()
{
if(IsReadOnly()) return;
int i = GetData();
if(IsNull(i)) {
if(IsNull(GetMin()) || GetMin() == INT_MIN) return;
SetData(GetMin());
}
else
if(!IsNull(GetMax()) && i < GetMax())
SetData(min(i + inc, GetMax()));
else
return;
SetFocus();
SetSelection();
UpdateAction();
}
void EditIntSpin::Dec()
{
if(IsReadOnly()) return;
int i = GetData();
if(IsNull(i)) {
if(IsNull(GetMax()) || GetMax() == INT_MAX) return;
SetData(GetMax());
}
else
if(!IsNull(GetMin()) && i > GetMin())
SetData(max(i - inc, GetMin()));
else
return;
SetFocus();
SetSelection();
UpdateAction();
}
void EditIntSpin::Init()
{
sb.inc.WhenAction = sb.inc.WhenRepeat = callback(this, &EditIntSpin::Inc);
sb.dec.WhenAction = sb.dec.WhenRepeat = callback(this, &EditIntSpin::Dec);
AddFrame(sb);
inc = 1;
}
EditIntSpin::EditIntSpin()
{
Init();
}
EditIntSpin::EditIntSpin(int min, int max)
{
MinMax(min, max);
Init();
}
EditIntSpin::~EditIntSpin() {}
bool EditIntSpin::Key(dword key, int repcnt)
{
if(key == K_UP) { Inc(); return true; }
if(key == K_DOWN) { Dec(); return true; }
return EditInt::Key(key, repcnt);
}
void EditIntSpin::MouseWheel(Point, int zdelta, dword)
{
if(zdelta < 0)
Dec();
else
Inc();
}
EditDoubleSpin::EditDoubleSpin(double inc) : inc(inc) { Init(); }
EditDoubleSpin::EditDoubleSpin(double min, double max, double inc) : EditDouble(min, max), inc(inc) { Init(); }
EditDoubleSpin::~EditDoubleSpin() {}
void EditDoubleSpin::Init()
{
AddFrame(spin);
spin.inc.WhenRepeat = spin.inc.WhenAction = THISBACK(Inc);
spin.dec.WhenRepeat = spin.dec.WhenAction = THISBACK(Dec);
}
void EditDoubleSpin::Inc()
{
if(IsReadOnly())
{
BeepExclamation();
return;
}
double d = GetData();
if(!IsNull(d))
{
d = (floor(d / inc + 1e-3) + 1) * inc;
if(IsNull(GetMax()) || d <= GetMax())
{
SetData(d);
Action();
}
}
else if(minval != DOUBLE_NULL_LIM)
SetData(minval);
SetFocus();
}
void EditDoubleSpin::Dec()
{
if(IsReadOnly())
{
BeepExclamation();
return;
}
double d = GetData();
if(!IsNull(d))
{
d = (ceil(d / inc - 1e-3) - 1) * inc;
if(IsNull(GetMin()) || d >= GetMin())
{
SetData(d);
Action();
}
}
else if(maxval != -DOUBLE_NULL_LIM)
SetData(maxval);
SetFocus();
}
bool EditDoubleSpin::Key(dword key, int repcnt)
{
if(key == K_UP) { Inc(); return true; }
if(key == K_DOWN) { Dec(); return true; }
return EditDouble::Key(key, repcnt);
}
void EditDoubleSpin::MouseWheel(Point, int zdelta, dword)
{
if(zdelta < 0)
Dec();
else
Inc();
}
END_UPP_NAMESPACE