mirror of
https://github.com/ultimatepp/ultimatepp.git
synced 2026-05-16 06:05:58 -06:00
311 lines
8.8 KiB
C++
311 lines
8.8 KiB
C++
#include "CtrlCore.h"
|
|
|
|
#ifdef GUI_X11
|
|
|
|
NAMESPACE_UPP
|
|
|
|
#define LLOG(x) // LOG(x)
|
|
#define LTIMING(x) // TIMING(x)
|
|
|
|
void SystemDraw::BeginOp()
|
|
{
|
|
Cloff f = cloff.Top();
|
|
Vector<Rect> newclip;
|
|
newclip <<= clip.Top();
|
|
f.clipi = clip.GetCount();
|
|
clip.Add() = newclip;
|
|
cloff.Add(f);
|
|
}
|
|
|
|
void SystemDraw::OffsetOp(Point p)
|
|
{
|
|
Cloff f = cloff.Top();
|
|
f.offseti = offset.GetCount();
|
|
actual_offset += p;
|
|
drawingclip -= p;
|
|
offset.Add(actual_offset);
|
|
cloff.Add(f);
|
|
}
|
|
|
|
bool SystemDraw::ClipOp(const Rect& r)
|
|
{
|
|
LLOG("SystemDraw::ClipOp(" << r << ")");
|
|
Cloff f = cloff.Top();
|
|
bool ch = false;
|
|
Vector<Rect> newclip = Intersect(clip.Top(), r + actual_offset, ch);
|
|
if(ch) {
|
|
f.clipi = clip.GetCount();
|
|
clip.Add() = newclip;
|
|
}
|
|
cloff.Add(f);
|
|
if(ch)
|
|
SetClip();
|
|
return clip.Top().GetCount();
|
|
}
|
|
|
|
bool SystemDraw::ClipoffOp(const Rect& r)
|
|
{
|
|
LLOG("SystemDraw::ClipOffOp(" << r << ")");
|
|
Cloff f = cloff.Top();
|
|
bool ch = false;
|
|
Vector<Rect> newclip = Intersect(clip.Top(), r + actual_offset, ch);
|
|
if(ch) {
|
|
f.clipi = clip.GetCount();
|
|
clip.Add() = newclip;
|
|
}
|
|
f.offseti = offset.GetCount();
|
|
actual_offset += r.TopLeft();
|
|
drawingclip -= r.TopLeft();
|
|
offset.Add(actual_offset);
|
|
cloff.Add(f);
|
|
if(ch)
|
|
SetClip();
|
|
return clip.Top().GetCount();
|
|
}
|
|
|
|
void SystemDraw::EndOp()
|
|
{
|
|
ASSERT(cloff.GetCount());
|
|
cloff.Drop();
|
|
actual_offset = offset[cloff.Top().offseti];
|
|
clip.SetCount(cloff.Top().clipi + 1);
|
|
SetClip();
|
|
}
|
|
|
|
bool SystemDraw::ExcludeClipOp(const Rect& r)
|
|
{
|
|
LLOG("SystemDraw::ExcludeClipOp(" << r << ")");
|
|
CloneClip();
|
|
Vector<Rect>& cl = clip.Top();
|
|
bool ch = false;
|
|
Vector<Rect> ncl = Subtract(cl, r + actual_offset, ch);
|
|
if(ch) {
|
|
cl = ncl;
|
|
SetClip();
|
|
}
|
|
return clip.Top().GetCount();
|
|
}
|
|
|
|
bool SystemDraw::IntersectClipOp(const Rect& r)
|
|
{
|
|
CloneClip();
|
|
Vector<Rect>& cl = clip.Top();
|
|
bool ch = false;
|
|
Vector<Rect> ncl = Intersect(cl, r + actual_offset, ch);
|
|
if(ch) {
|
|
cl = ncl;
|
|
SetClip();
|
|
}
|
|
return clip.Top().GetCount();
|
|
}
|
|
|
|
bool SystemDraw::IsPaintingOp(const Rect& r) const
|
|
{
|
|
LTIMING("IsPaintingOp");
|
|
Rect rr = r + actual_offset;
|
|
const Vector<Rect>& cl = clip[cloff.Top().clipi];
|
|
for(int i = 0; i < cl.GetCount(); i++)
|
|
if(cl[i].Intersects(rr))
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
Rect SystemDraw::GetPaintRect() const
|
|
{
|
|
return drawingclip;
|
|
}
|
|
|
|
void SystemDraw::DrawRectOp(int x, int y, int cx, int cy, Color color)
|
|
{
|
|
LTIMING("DrawRect");
|
|
GuiLock __;
|
|
LLOG("DrawRect " << RectC(x, y, cx, cy) << ": " << color);
|
|
if(IsNull(color)) return;
|
|
if(cx <= 0 || cy <= 0) return;
|
|
if(color == InvertColor) {
|
|
XSetFunction(Xdisplay, gc, GXinvert);
|
|
XFillRectangle(Xdisplay, dw, gc, x + actual_offset.x, y + actual_offset.y, cx, cy);
|
|
XSetFunction(Xdisplay, gc, GXcopy);
|
|
}
|
|
else {
|
|
SetForeground(color);
|
|
XFillRectangle(Xdisplay, dw, gc, x + actual_offset.x, y + actual_offset.y, cx, cy);
|
|
}
|
|
}
|
|
|
|
void SystemDraw::DrawLineOp(int x1, int y1, int x2, int y2, int width, Color color)
|
|
{
|
|
GuiLock __;
|
|
if(IsNull(width) || IsNull(color)) return;
|
|
SetLineStyle(width);
|
|
SetForeground(color);
|
|
XDrawLine(Xdisplay, dw, gc,
|
|
x1 + actual_offset.x, y1 + actual_offset.y,
|
|
x2 + actual_offset.x, y2 + actual_offset.y);
|
|
}
|
|
|
|
void SystemDraw::DrawPolyPolylineOp(const Point *vertices, int vertex_count,
|
|
const int *counts, int count_count,
|
|
int width, Color color, Color doxor)
|
|
{
|
|
GuiLock __;
|
|
ASSERT(count_count > 0 && vertex_count > 0);
|
|
if(vertex_count < 2 || IsNull(color))
|
|
return;
|
|
SetLineStyle(width);
|
|
XGCValues gcv_old, gcv_new;
|
|
XGetGCValues(Xdisplay, GetGC(), GCForeground | GCFunction, &gcv_old);
|
|
gcv_new.function = IsNull(doxor) ? X11_ROP2_COPY : X11_ROP2_XOR;
|
|
gcv_new.foreground = GetXPixel(color) ^ (IsNull(doxor) ? 0 : GetXPixel(doxor));
|
|
XChangeGC(Xdisplay, GetGC(), GCForeground | GCFunction, &gcv_new);
|
|
enum { REQUEST_LENGTH = 256 }; // X server XDrawLines request length (heuristic)
|
|
Point offset = GetOffset();
|
|
if(vertex_count == 2)
|
|
XDrawLine(Xdisplay, GetDrawable(), GetGC(),
|
|
vertices[0].x + offset.x, vertices[0].y + offset.y,
|
|
vertices[1].x + offset.x, vertices[1].y + offset.y);
|
|
else if(count_count == 1 || vertex_count > count_count * (REQUEST_LENGTH + 2)) {
|
|
for(; --count_count >= 0; counts++)
|
|
{
|
|
Buffer<XPoint> part(*counts);
|
|
for(XPoint *vo = part, *ve = vo + *counts; vo < ve; vo++, vertices++)
|
|
{
|
|
vo -> x = (short)(vertices -> x + offset.x);
|
|
vo -> y = (short)(vertices -> y + offset.y);
|
|
}
|
|
XDrawLines(Xdisplay, GetDrawable(), GetGC(), part, *counts, CoordModeOrigin);
|
|
}
|
|
}
|
|
else {
|
|
int segment_count = vertex_count - count_count;
|
|
Buffer<XSegment> segments(segment_count);
|
|
XSegment *so = segments;
|
|
while(--count_count >= 0)
|
|
{
|
|
const Point *end = vertices + *counts++;
|
|
so -> x1 = (short)(vertices -> x + offset.x);
|
|
so -> y1 = (short)(vertices -> y + offset.y);
|
|
vertices++;
|
|
so -> x2 = (short)(vertices -> x + offset.x);
|
|
so -> y2 = (short)(vertices -> y + offset.y);
|
|
so++;
|
|
while(++vertices < end) {
|
|
so -> x1 = so[-1].x2;
|
|
so -> y1 = so[-1].y2;
|
|
so -> x2 = (short)(vertices -> x + offset.x);
|
|
so -> y2 = (short)(vertices -> y + offset.y);
|
|
so++;
|
|
}
|
|
}
|
|
ASSERT(so == segments + segment_count);
|
|
XDrawSegments(Xdisplay, GetDrawable(), GetGC(), segments, segment_count);
|
|
}
|
|
XChangeGC(Xdisplay, GetGC(), GCForeground | GCFunction, &gcv_old);
|
|
}
|
|
|
|
static void DrawPolyPolyPolygonRaw(SystemDraw& draw, const Point *vertices, int vertex_count,
|
|
const int *subpolygon_counts, int subpolygon_count_count, const int *, int)
|
|
{
|
|
GuiLock __;
|
|
Point offset = draw.GetOffset();
|
|
const Point *in = vertices;
|
|
for(int i = 0; i < subpolygon_count_count; i++) {
|
|
int n = subpolygon_counts[i];
|
|
Buffer<XPoint> out_points(n);
|
|
XPoint *t = out_points;
|
|
XPoint *e = t + n;
|
|
while(t < e) {
|
|
t->x = (short)(in->x + offset.x);
|
|
t->y = (short)(in->y + offset.y);
|
|
t++;
|
|
in++;
|
|
}
|
|
XFillPolygon(Xdisplay, draw.GetDrawable(), draw.GetGC(), out_points, n, Nonconvex, CoordModeOrigin);
|
|
}
|
|
}
|
|
|
|
void SystemDraw::DrawPolyPolyPolygonOp(const Point *vertices, int vertex_count,
|
|
const int *subpolygon_counts, int subpolygon_count_count,
|
|
const int *disjunct_polygon_counts, int disjunct_polygon_count_count,
|
|
Color color, int width, Color outline, uint64 pattern, Color doxor)
|
|
{
|
|
GuiLock __;
|
|
if(vertex_count == 0)
|
|
return;
|
|
|
|
if(!IsNull(outline)) SetLineStyle(width); // Added
|
|
bool is_xor = !IsNull(doxor);
|
|
XGCValues gcv_old, gcv_new;
|
|
XGetGCValues(Xdisplay, GetGC(), GCForeground | GCFunction, &gcv_old);
|
|
unsigned xor_pixel = (is_xor ? GetXPixel(doxor) : 0);
|
|
if(!IsNull(color))
|
|
{
|
|
gcv_new.foreground = GetXPixel(color) ^ xor_pixel;
|
|
gcv_new.function = is_xor ? X11_ROP2_XOR : X11_ROP2_COPY;
|
|
XChangeGC(Xdisplay, GetGC(), GCForeground | GCFunction, &gcv_new);
|
|
DrawPolyPolyPolygonRaw(*this, vertices, vertex_count,
|
|
subpolygon_counts, subpolygon_count_count,
|
|
disjunct_polygon_counts, disjunct_polygon_count_count);
|
|
}
|
|
if(!IsNull(outline))
|
|
{
|
|
gcv_new.foreground = GetXPixel(outline) ^ xor_pixel;
|
|
XChangeGC(Xdisplay, GetGC(), GCForeground, &gcv_new);
|
|
Point offset = GetOffset();
|
|
for(const int *sp = subpolygon_counts, *se = sp + subpolygon_count_count; sp < se; sp++)
|
|
{
|
|
Buffer<XPoint> segment(*sp + 1);
|
|
XPoint *out = segment;
|
|
for(const Point *end = vertices + *sp; vertices < end; vertices++, out++)
|
|
{
|
|
out -> x = (short)(vertices -> x + offset.x);
|
|
out -> y = (short)(vertices -> y + offset.y);
|
|
}
|
|
*out = segment[0];
|
|
XDrawLines(Xdisplay, GetDrawable(), GetGC(), segment, *sp + 1, CoordModeOrigin);
|
|
}
|
|
}
|
|
XChangeGC(Xdisplay, GetGC(), GCForeground | GCFunction, &gcv_old);
|
|
}
|
|
|
|
void SystemDraw::DrawEllipseOp(const Rect& r, Color color, int pen, Color pencolor)
|
|
{
|
|
GuiLock __;
|
|
SetLineStyle(pen);
|
|
if(!IsNull(color)) {
|
|
SetForeground(color);
|
|
XFillArc(Xdisplay, dw, gc, r.left + actual_offset.x, r.top + actual_offset.y,
|
|
r.Width() - 1, r.Height() - 1, 0, 360 * 64);
|
|
}
|
|
if(!IsNull(pencolor) && !IsNull(pen)) {
|
|
SetForeground(pencolor);
|
|
XDrawArc(Xdisplay, dw, gc, r.left + actual_offset.x, r.top + actual_offset.y,
|
|
r.Width() - 1, r.Height() - 1, 0, 360 * 64);
|
|
}
|
|
}
|
|
|
|
void SystemDraw::DrawArcOp(const Rect& rc, Point start, Point end, int width, Color color)
|
|
{
|
|
GuiLock __;
|
|
SetLineStyle(width);
|
|
XGCValues gcv, gcv_old;
|
|
XGetGCValues(Xdisplay, GetGC(), GCForeground, &gcv_old);
|
|
Point offset = GetOffset();
|
|
gcv.foreground = GetXPixel(color);
|
|
XChangeGC(Xdisplay, GetGC(), GCForeground, &gcv);
|
|
Point centre = rc.CenterPoint();
|
|
int angle1 = fround(360 * 64 / (2 * M_PI) *
|
|
atan2(centre.y - start.y, start.x - centre.x));
|
|
int angle2 = fround(360 * 64 / (2 * M_PI) *
|
|
atan2(centre.y - end.y, end.x - centre.x));
|
|
if(angle2 <= angle1)
|
|
angle2 += 360 * 64;
|
|
angle2 -= angle1;
|
|
XDrawArc(Xdisplay, GetDrawable(), GetGC(), rc.left + offset.x, rc.top + offset.y,
|
|
rc.Width(), rc.Height(), angle1, angle2);
|
|
XChangeGC(Xdisplay, GetGC(), GCForeground, &gcv_old);
|
|
}
|
|
|
|
END_UPP_NAMESPACE
|
|
|
|
#endif
|