ultimatepp/uppsrc/IconDes/List.cpp
cxl 00a11f2c47 IconDes: UHD issues, GLCtrl developing
git-svn-id: svn://ultimatepp.org/upp/trunk@12249 f0d560ea-af0d-0410-9eb7-867de7ffcac7
2018-09-04 21:24:11 +00:00

552 lines
12 KiB
C++

#include "IconDes.h"
namespace Upp {
static String sFormatImageName(const String& name, const Image& img, bool exp)
{
Size sz = img.GetSize();
String r;
r << name << " (" << sz.cx << " x " << sz.cy
<< decode(img.GetResolution(), IMAGE_RESOLUTION_UHD, " UHD", IMAGE_RESOLUTION_NONE, " n.r.", "")
<< ')';
if(exp)
r << " ex.";
return r;
}
void IconDes::SyncList()
{
if(syncinglist)
return;
syncinglist++;
int sc = ilist.GetScroll();
int q = ilist.GetKey();
ilist.Clear();
String s = ToUpper((String)~search);
for(int i = 0; i < slot.GetCount(); i++) {
Slot& c = slot[i];
if(ToUpper(c.name).Find(s) >= 0)
ilist.Add(i, sFormatImageName(c.name, c.image, c.exp), c.image);
}
ilist.ScrollTo(sc);
ilist.FindSetCursor(q);
syncinglist--;
}
void IconDes::Search()
{
SyncList();
}
void IconDes::GoTo(int q)
{
ilist.FindSetCursor(q);
if(ilist.IsCursor())
return;
search <<= Null;
SyncList();
ilist.FindSetCursor(q);
}
static int sCharFilterCid(int c)
{
return IsAlNum(c) || c == '_' ? c : 0;
}
void IconDes::PlaceDlg(TopWindow& dlg)
{
Rect r = ilist.GetScreenRect();
Size sz = dlg.GetSize();
dlg.NoCenter().SetRect(max(0, r.left + (r.Width() - sz.cx) / 2), r.bottom + 32, sz.cx, sz.cy);
}
void IconDes::PrepareImageDlg(WithImageLayout<TopWindow>& dlg)
{
CtrlLayoutOKCancel(dlg, "New image");
dlg.cx <<= 16;
dlg.cy <<= 16;
if(IsCurrent()) {
Size sz = GetImageSize();
dlg.cx <<= sz.cx;
dlg.cy <<= sz.cy;
int resolution = IMAGE_RESOLUTION_STANDARD;
if(IsCurrent())
resolution = Current().image.GetResolution();
dlg.resolution.Set(0, IMAGE_RESOLUTION_STANDARD);
dlg.resolution.Set(1, IMAGE_RESOLUTION_UHD);
dlg.resolution.Set(2, IMAGE_RESOLUTION_NONE);
dlg.resolution <<= resolution;
}
dlg.name.SetFilter(sCharFilterCid);
}
void IconDes::PrepareImageSizeDlg(WithImageSizeLayout<TopWindow>& dlg)
{
CtrlLayoutOKCancel(dlg, "New image");
dlg.cx <<= 16;
dlg.cy <<= 16;
if(IsCurrent()) {
Size sz = GetImageSize();
dlg.cx <<= sz.cx;
dlg.cy <<= sz.cy;
}
}
bool CheckName(WithImageLayout<TopWindow>& dlg)
{
String n = ~dlg.name;
CParser p(n);
if(p.IsId()) return true;
Exclamation("Invalid name!");
return false;
}
void IconDes::InsertRemoved(int q)
{
if(q >= 0 && q < removed.GetCount()) {
int ii = ilist.IsCursor() ? (int)ilist.GetKey() : 0;
slot.Insert(ii) = removed[q];
removed.Remove(q);
SyncList();
GoTo(ii);
}
}
void SetRes(Image& m, int resolution)
{
ImageBuffer ib(m);
ib.SetResolution(findarg(resolution, IMAGE_RESOLUTION_STANDARD, IMAGE_RESOLUTION_UHD,
IMAGE_RESOLUTION_NONE)
>= 0 ? resolution : IMAGE_RESOLUTION_STANDARD);
m = ib;
}
void IconDes::ImageInsert(int ii, const String& name, const Image& m, bool exp)
{
Slot& c = slot.Insert(ii);
c.name = name;
c.image = m;
c.exp = exp;
SyncList();
GoTo(ii);
}
void IconDes::ImageInsert(const String& name, const Image& m, bool exp)
{
int ii = ilist.IsCursor() ? (int)ilist.GetKey() : 0;
if(ii == slot.GetCount() - 1)
ii = slot.GetCount();
ImageInsert(ii, name, m, exp);
}
void IconDes::InsertImage()
{
WithImageLayout<TopWindow> dlg;
PrepareImageDlg(dlg);
dlg.resolution <<= IMAGE_RESOLUTION_STANDARD;
do {
if(dlg.Run() != IDOK)
return;
}
while(!CheckName(dlg));
Image m = CreateImage(Size(~dlg.cx, ~dlg.cy), Null);
SetRes(m, ~dlg.resolution);
ImageInsert(~dlg.name, m, dlg.exp);
}
void IconDes::Slice()
{
if(!IsCurrent())
return;
Image src = Current().image;
Size isz = src.GetSize();
int cc = min(isz.cx, isz.cy);
if(!cc)
return;
WithImageLayout<TopWindow> dlg;
PrepareImageDlg(dlg);
dlg.name <<= Current().name;
dlg.cx <<= cc;
dlg.cy <<= cc;
dlg.Title("Slice image");
dlg.resolution <<= IMAGE_RESOLUTION_STANDARD;
do {
if(dlg.Run() != IDOK)
return;
}
while(!CheckName(dlg));
String s = ~dlg.name;
int n = 0;
int ii = ilist.GetKey();
for(int y = 0; y < isz.cy; y += (int)~dlg.cy)
for(int x = 0; x < isz.cx; x += (int)~dlg.cx) {
Image m = Crop(src, x, y, ~dlg.cx, ~dlg.cy);
SetRes(m, ~dlg.resolution);
ImageInsert(++ii, s + AsString(n++), m, ~dlg.exp);
}
}
void IconDes::Duplicate()
{
if(!IsCurrent())
return;
Slot& c = Current();
ImageInsert(c.name, c.image);
EditImage();
}
void IconDes::InsertPaste()
{
Image m = ReadClipboardImage();
if(IsNull(m)) {
Exclamation("Clipboard does not contain an image.");
return;
}
SetRes(m, IMAGE_RESOLUTION_STANDARD);
ImageInsert("", m);
EditImage();
}
struct FileImage : ImageMaker {
String filename;
Size size;
virtual String Key() const { return filename + '/' + AsString(size); }
virtual Image Make() const {
if(GetFileLength(filename) > 1024 * 1024 * 20)
return Null;
Image m = StreamRaster::LoadFileAny(filename);
Size sz = m.GetSize();
if(sz.cx > size.cx || sz.cy > size.cy) {
if(sz.cx * size.cy > sz.cy * size.cx)
sz = GetRatioSize(sz, size.cx, 0);
else
sz = GetRatioSize(sz, 0, size.cy);
return Rescale(m, sz);
}
return m;
}
};
struct ImgPreview : Display {
virtual void Paint(Draw& w, const Rect& r, const Value& q, Color ink, Color paper, dword style) const {
if(!IsNull(q)) {
FileImage im;
im.size = r.GetSize();
im.filename = q;
w.DrawRect(r, SColorPaper);
Image m = MakeImage(im);
Point p = r.CenterPos(m.GetSize());
w.DrawImage(p.x, p.y, m);
}
}
};
#ifdef _MULTITHREADED
static void sLoadImage(const String& path, Image& result)
{
if(findarg(ToLower(GetFileExt(path)), ".png", ".gif", ".jpeg", ".jpg") < 0)
return;
FileIn in(path);
if(!in)
return;
One<StreamRaster> r = StreamRaster::OpenAny(in);
if(!r)
return;
Size sz = r->GetSize();
if(sz.cx > 80 || sz.cy > 80)
return;
result = r->GetImage();
}
#endif
FileSel& IconDes::ImgFile()
{
static FileSel sel;
ONCELOCK {
sel.Type("Image files", "*.png *.bmp *.jpg *.jpeg *.gif");
sel.AllFilesType();
sel.Multi();
#ifdef _MULTITHREADED
sel.WhenIconLazy = sLoadImage;
#endif
sel.Preview(Single<ImgPreview>());
}
return sel;
}
int CharFilterImageId(int c)
{
return IsAlNum(c) ? c : '_';
}
void IconDes::InsertFile()
{
if(!ImgFile().ExecuteOpen()) return;
for(int i = 0; i < ImgFile().GetCount(); i++) {
String fn = ImgFile()[i];
Image m = StreamRaster::LoadFileAny(fn);
if(IsNull(m))
Exclamation(DeQtf(fn) + " not an image.");
String id = Filter(GetFileTitle(fn), CharFilterImageId);
if(!IsAlpha(*id) && *id != '_')
id = '_' + id;
SetRes(m, IMAGE_RESOLUTION_STANDARD);
ImageInsert(id, m);
}
}
void IconDes::ExportPngs()
{
String dir = SelectDirectory();
if(!dir.IsEmpty())
for(int i = 0; i < GetCount(); i++)
PNGEncoder().SaveFile(AppendFileName(dir, GetName(i) + ".png"), GetImage(i));
}
void IconDes::InsertIml()
{
Array<ImlImage> m;
int f;
if(LoadIml(SelectLoadFile("Iml files\t*.iml"), m, f))
for(int i = 0; i < m.GetCount(); i++) {
ImageInsert(m[i].name, m[i].image, m[i].exp);
GoTo((int)ilist.GetKey() + 1);
}
}
void IconDes::ListCursor()
{
SyncImage();
}
void IconDes::EditImageSize()
{
Slot& c = Current();
WithImageSizeLayout<TopWindow> dlg;
PrepareImageSizeDlg(dlg);
dlg.Breaker(dlg.cx);
dlg.Breaker(dlg.cy);
Image img = c.image;
dlg.cx <<= img.GetWidth();
dlg.cy <<= img.GetHeight();
for(;;) {
switch(dlg.Run()) {
case IDCANCEL:
c.image = img;
Reset();
return;
case IDOK:
Reset();
SyncList();
return;
}
c.image = CreateImage(Size(minmax((int)~dlg.cx, 1, 8192), minmax((int)~dlg.cy, 1, 8192)), Null);
UPP::Copy(c.image, Point(0, 0), img, img.GetSize());
Reset();
}
}
void IconDes::EditImage()
{
if(!IsCurrent())
return;
if(single_mode) {
EditImageSize();
return;
}
Slot& c = Current();
WithImageLayout<TopWindow> dlg;
PrepareImageDlg(dlg);
dlg.Title("Image");
dlg.Breaker(dlg.cx);
dlg.Breaker(dlg.cy);
Image img = c.image;
dlg.cx <<= img.GetWidth();
dlg.cy <<= img.GetHeight();
dlg.name <<= c.name;
dlg.exp <<= c.exp;
for(;;) {
switch(dlg.Run()) {
case IDCANCEL:
c.image = img;
Reset();
return;
case IDOK:
if(!CheckName(dlg)) break;
c.name = ~dlg.name;
c.exp = ~dlg.exp;
SetRes(c.image, ~dlg.resolution);
ilist.Set(1, sFormatImageName(c.name, c.image, c.exp));
int q = ilist.GetKey();
Reset();
SyncList();
GoTo(q);
return;
}
int r = c.image.GetResolution();
c.image = CreateImage(Size(minmax((int)~dlg.cx, 1, 8192), minmax((int)~dlg.cy, 1, 8192)), Null);
UPP::Copy(c.image, Point(0, 0), img, img.GetSize());
SetRes(c.image, r);
Reset();
}
}
void IconDes::RemoveImage()
{
if(!IsCurrent() || !PromptYesNo("Remove current image?"))
return;
int ii = ilist.GetKey();
while(removed.GetCount() > 12)
removed.Remove(0);
Slot& r = removed.Add();
r = slot[ii];
if(r.image.GetWidth() <= 128 && r.image.GetHeight() <= 128)
r.base_image = Rescale(r.image, Size(16, 16));
else
r.base_image = IconDesImg::LargeImage();
slot.Remove(ii);
ilist.KillCursor();
SyncList();
if(ii < slot.GetCount())
GoTo(ii);
else
ilist.GoEnd();
}
void IconDes::ChangeSlot(int d)
{
if(!IsCurrent())
return;
int c = ilist.GetCursor();
d = c + d;
if(d >= 0 && d < ilist.GetCount())
ilist.SetCursor(d);
}
void IconDes::ListMenu(Bar& bar)
{
using namespace IconDesKeys;
if(single_mode)
bar.Add(IsCurrent(), AK_RESIZE_SINGLE, IconDesImg::Edit(), THISBACK(EditImage));
else {
bar.Add(AK_INSERT_IMAGE, IconDesImg::Insert(), THISBACK(InsertImage));
bar.Add(IsCurrent(), AK_IMAGE, IconDesImg::Edit(), THISBACK(EditImage));
bar.Add(IsCurrent(), AK_REMOVE_IMAGE, IconDesImg::Remove(), THISBACK(RemoveImage));
bar.Add(IsCurrent(), AK_DUPLICATE, IconDesImg::Duplicate(), THISBACK(Duplicate));
bar.Add(AK_INSERT_CLIP, IconDesImg::InsertPaste(), THISBACK(InsertPaste));
bar.Add(AK_INSERT_FILE, IconDesImg::InsertFile(), THISBACK(InsertFile));
bar.Add(AK_INSERT_IML, IconDesImg::InsertIml(), THISBACK(InsertIml));
bar.Add(AK_EXPORT_PNGS, IconDesImg::ExportPngs(), THISBACK(ExportPngs));
bar.Separator();
int q = ilist.GetKey();
bar.Add(IsCurrent() && q > 0, AK_MOVE_UP, IconDesImg::MoveUp(),
THISBACK1(MoveSlot, -1));
bar.Add(IsCurrent() && q < slot.GetCount() - 1, AK_MOVE_DOWN, IconDesImg::MoveDown(),
THISBACK1(MoveSlot, 1));
if(removed.GetCount()) {
bar.Separator();
for(int i = removed.GetCount() - 1; i >= 0; i--) {
Slot& r = removed[i];
bar.Add("Insert " + sFormatImageName(r.name, r.image, r.exp), r.base_image,
THISBACK1(InsertRemoved, i));
}
}
}
bar.Separator();
EditBar(bar);
ListMenuEx(bar);
}
void IconDes::ListMenuEx(Bar& bar) {}
void IconDes::Clear()
{
ilist.Clear();
slot.Clear();
Reset();
}
void IconDes::AddImage(const String& name, const Image& image, bool exp)
{
int q = slot.GetCount();
Slot& c = slot.Add();
c.name = name;
c.image = image;
c.exp = exp;
ilist.Add(q, sFormatImageName(c.name, c.image, c.exp), c.image);
ilist.GoBegin();
}
int IconDes::GetCount() const
{
return slot.GetCount();
}
Image IconDes::GetImage(int ii) const
{
return slot[ii].image;
}
String IconDes::GetName(int ii) const
{
return slot[ii].name;
}
bool IconDes::FindName(const String& name)
{
for(int i = 0; i < slot.GetCount(); i++)
if(slot[i].name == name) {
GoTo(i);
return true;
}
return false;
}
String IconDes::GetCurrentName() const
{
if(ilist.IsCursor())
return GetName(ilist.GetKey());
return String();
}
bool IconDes::GetExport(int ii) const
{
return slot[ii].exp;
}
void IconDes::MoveSlot(int d)
{
if(!IsCurrent())
return;
int c = ilist.GetKey();
d = c + d;
if(d >= 0 && d < slot.GetCount()) {
slot.Swap(c, d);
search <<= Null;
SyncList();
GoTo(d);
}
}
void IconDes::DnDInsert(int line, PasteClip& d)
{
if(GetInternalPtr<ArrayCtrl>(d, "icondes-icon") == &ilist && IsCurrent() &&
line >= 0 && line <= slot.GetCount() && d.Accept()) {
int c = ilist.GetKey();
slot.Move(c, line);
if(c <= line)
line--;
search <<= Null;
SyncList();
GoTo(line);
}
}
void IconDes::Drag()
{
ilist.DoDragAndDrop(InternalClip(ilist, "icondes-icon"), ilist.GetDragSample(), DND_MOVE);
}
}