#include "Draw.h" NAMESPACE_UPP #ifdef PLATFORM_WIN32 #define LLOG(x) // LOG(x) #define LTIMING(x) // RTIMING(x) void Draw::BeginOp() { LTIMING("Begin"); DrawLock __; Cloff& w = cloff.Add(); w.org = actual_offset; w.hrgn = CreateRectRgn(0, 0, 0, 0); ASSERT(w.hrgn); int q = ::GetClipRgn(handle, w.hrgn); ASSERT(q >= 0); if(q == 0) { DeleteObject(w.hrgn); w.hrgn = NULL; } } void Draw::OffsetOp(Point p) { DrawLock __; Begin(); actual_offset += p; LTIMING("Offset"); SetOrg(); } bool Draw::ClipOp(const Rect& r) { DrawLock __; Begin(); LTIMING("Clip"); return IntersectClip(r); } bool Draw::ClipoffOp(const Rect& r) { DrawLock __; Begin(); LTIMING("Clipoff"); LLOG("ClipoffOp " << r << ", GetClip() = " << GetClip() << ", actual_offset = " << actual_offset); actual_offset += r.TopLeft(); bool q = IntersectClip(r); SetOrg(); LLOG("//ClipoffOp, GetClip() = " << GetClip() << ", actual_offset = " << actual_offset); return q; } void Draw::EndOp() { DrawLock __; LTIMING("End"); ASSERT(cloff.GetCount()); Cloff& w = cloff.Top(); actual_offset = w.org; ::SelectClipRgn(handle, w.hrgn); SetOrg(); if(w.hrgn) ::DeleteObject(w.hrgn); cloff.Drop(); } bool Draw::ExcludeClipOp(const Rect& r) { DrawLock __; #ifdef PLATFORM_WINCE int q = ExcludeClipRect(handle, r.left, r.top, r.right, r.bottom); #else LTIMING("ExcludeClip"); Rect rr = LPtoDP(r); HRGN hrgn = ::CreateRectRgnIndirect(rr); int q = ::ExtSelectClipRgn(handle, hrgn, RGN_DIFF); ASSERT(q != ERROR); ::DeleteObject(hrgn); #endif return q == SIMPLEREGION || q == COMPLEXREGION; } bool Draw::IntersectClipOp(const Rect& r) { DrawLock __; #ifdef PLATFORM_WINCE int q = IntersectClipRect(handle, r.left, r.top, r.right, r.bottom); #else LTIMING("Intersect"); Rect rr = LPtoDP(r); HRGN hrgn = ::CreateRectRgnIndirect(rr); int q = ::ExtSelectClipRgn(handle, hrgn, RGN_AND); ASSERT(q != ERROR); ::DeleteObject(hrgn); #endif return q == SIMPLEREGION || q == COMPLEXREGION; } Rect Draw::GetClipOp() const { DrawLock __; Rect r; ::GetClipBox(handle, r); return r; } bool Draw::IsPaintingOp(const Rect& r) const { DrawLock __; LTIMING("IsPainting"); LLOG("Draw::IsPaintingOp r: " << r); return ::RectVisible(handle, r); } void Draw::DrawRectOp(int x, int y, int cx, int cy, Color color) { DrawLock __; LTIMING("DrawRect"); LLOG("DrawRect " << RectC(x, y, cx, cy) << ": " << color); if(IsNull(color)) return; if(cx <= 0 || cy <= 0) return; if(color == InvertColor) ::PatBlt(handle, x, y, cx, cy, DSTINVERT); else { SetColor(color); ::PatBlt(handle, x, y, cx, cy, PATCOPY); } } void Draw::DrawLineOp(int x1, int y1, int x2, int y2, int width, Color color) { DrawLock __; if(IsNull(width) || IsNull(color)) return; SetDrawPen(width, color); ::MoveToEx(handle, x1, y1, NULL); ::LineTo(handle, x2, y2); } #ifndef PLATFORM_WINCE void Draw::DrawPolyPolylineOp(const Point *vertices, int vertex_count, const int *counts, int count_count, int width, Color color, Color doxor) { DrawLock __; ASSERT(count_count > 0 && vertex_count > 0); if(vertex_count < 2 || IsNull(color)) return; bool is_xor = !IsNull(doxor); if(is_xor) color = Color(color.GetR() ^ doxor.GetR(), color.GetG() ^ doxor.GetG(), color.GetB() ^ doxor.GetB()); if(is_xor) SetROP2(GetHandle(), R2_XORPEN); SetDrawPen(width, color); if(count_count == 1) ::Polyline(GetHandle(), (const POINT *)vertices, vertex_count); else ::PolyPolyline(GetHandle(), (const POINT *)vertices, (const dword *)counts, count_count); if(is_xor) SetROP2(GetHandle(), R2_COPYPEN); } static void DrawPolyPolyPolygonRaw( Draw& draw, const Point *vertices, int vertex_count, const int *subpolygon_counts, int subpolygon_count_count, const int *disjunct_polygon_counts, int disjunct_polygon_count_count) { DrawLock __; for(int i = 0; i < disjunct_polygon_count_count; i++, disjunct_polygon_counts++) { int poly = *disjunct_polygon_counts; int sub = 1; if(*subpolygon_counts < poly) if(disjunct_polygon_count_count > 1) { const int *se = subpolygon_counts; int total = 0; while(total < poly) total += *se++; sub = (int)(se - subpolygon_counts); } else sub = subpolygon_count_count; ASSERT(sizeof(POINT) == sizeof(Point)); // modify algorithm when not if(sub == 1) Polygon(draw, (const POINT *)vertices, poly); else PolyPolygon(draw, (const POINT *)vertices, subpolygon_counts, sub); vertices += poly; subpolygon_counts += sub; } } void Draw::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) { DrawLock __; if(vertex_count == 0) return; bool is_xor = !IsNull(doxor); HDC hdc = GetHandle(); if(pattern) { int old_rop = GetROP2(hdc); HGDIOBJ old_brush = GetCurrentObject(hdc, OBJ_BRUSH); word wpat[8] = { (byte)(pattern >> 56), (byte)(pattern >> 48), (byte)(pattern >> 40), (byte)(pattern >> 32), (byte)(pattern >> 24), (byte)(pattern >> 16), (byte)(pattern >> 8), (byte)(pattern >> 0), }; HBITMAP bitmap = CreateBitmap(8, 8, 1, 1, wpat); HBRUSH brush = ::CreatePatternBrush(bitmap); COLORREF old_bk = GetBkColor(hdc); COLORREF old_fg = GetTextColor(hdc); if(!is_xor) { SetROP2(hdc, R2_MASKPEN); SelectObject(hdc, brush); SetTextColor(hdc, Black()); SetBkColor(hdc, White()); SetDrawPen(PEN_NULL, Black); DrawPolyPolyPolygonRaw(*this, vertices, vertex_count, subpolygon_counts, subpolygon_count_count, disjunct_polygon_counts, disjunct_polygon_count_count); SetROP2(hdc, R2_MERGEPEN); SetTextColor(hdc, color); SetBkColor(hdc, Black()); } else { SetROP2(hdc, R2_XORPEN); SetTextColor(hdc, COLORREF(color) ^ COLORREF(doxor)); SelectObject(hdc, brush); } DrawPolyPolyPolygonRaw(*this, vertices, vertex_count, subpolygon_counts, subpolygon_count_count, disjunct_polygon_counts, disjunct_polygon_count_count); SelectObject(hdc, old_brush); SetTextColor(hdc, old_fg); SetBkColor(hdc, old_bk); SetROP2(hdc, old_rop); DeleteObject(brush); DeleteObject(bitmap); if(!IsNull(outline)) { SetColor(Null); SetDrawPen(width, outline); ASSERT(sizeof(POINT) == sizeof(Point)); DrawPolyPolyPolygonRaw(*this, vertices, vertex_count, subpolygon_counts, subpolygon_count_count, disjunct_polygon_counts, disjunct_polygon_count_count); } } else { // simple fill SetDrawPen(IsNull(outline) ? PEN_NULL : width, Nvl(outline, Black)); int old_rop2; if(is_xor) { color = Color(color.GetR() ^ doxor.GetR(), color.GetG() ^ doxor.GetG(), color.GetB() ^ doxor.GetB()); old_rop2 = SetROP2(hdc, R2_XORPEN); } SetColor(color); DrawPolyPolyPolygonRaw(*this, vertices, vertex_count, subpolygon_counts, subpolygon_count_count, disjunct_polygon_counts, disjunct_polygon_count_count); if(is_xor) SetROP2(hdc, old_rop2); } } void Draw::DrawArcOp(const Rect& rc, Point start, Point end, int width, Color color) { DrawLock __; SetDrawPen(width, color); ::Arc(GetHandle(), rc.left, rc.top, rc.right, rc.bottom, start.x, start.y, end.x, end.y); } #endif void Draw::DrawEllipseOp(const Rect& r, Color color, int width, Color pencolor) { DrawLock __; SetColor(color); SetDrawPen(width, pencolor); ::Ellipse(GetHandle(), r.left, r.top, r.right, r.bottom); } #endif END_UPP_NAMESPACE