/uppsrc/CtrlCore/DrawOpX11.cpp
https://github.com/ultimatepp/mirror · C++ · 311 lines · 289 code · 22 blank · 0 comment · 44 complexity · feb661c68ab39cde89de0798daeb150b MD5 · raw file
- #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() = pick(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() = pick(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() = pick(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 = pick(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 = pick(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);
- }
- }
- #endif