/* * GDI pen objects * * Copyright 1993 Alexandre Julliard * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include #include "windef.h" #include "winbase.h" #include "wingdi.h" #include "gdi_private.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(gdi); /* GDI logical pen object */ typedef struct { struct gdi_obj_header obj; struct brush_pattern pattern; EXTLOGPEN logpen; } PENOBJ; static HGDIOBJ PEN_SelectObject( HGDIOBJ handle, HDC hdc ); static INT PEN_GetObject( HGDIOBJ handle, INT count, LPVOID buffer ); static BOOL PEN_DeleteObject( HGDIOBJ handle ); static const struct gdi_obj_funcs pen_funcs = { PEN_SelectObject, /* pSelectObject */ PEN_GetObject, /* pGetObjectA */ PEN_GetObject, /* pGetObjectW */ NULL, /* pUnrealizeObject */ PEN_DeleteObject /* pDeleteObject */ }; /*********************************************************************** * CreatePen (GDI32.@) */ HPEN WINAPI CreatePen( INT style, INT width, COLORREF color ) { LOGPEN logpen; TRACE("%d %d %06x\n", style, width, color ); logpen.lopnStyle = style; logpen.lopnWidth.x = width; logpen.lopnWidth.y = 0; logpen.lopnColor = color; return CreatePenIndirect( &logpen ); } /*********************************************************************** * CreatePenIndirect (GDI32.@) */ HPEN WINAPI CreatePenIndirect( const LOGPEN * pen ) { PENOBJ * penPtr; HPEN hpen; if (pen->lopnStyle == PS_NULL) { hpen = GetStockObject(NULL_PEN); if (hpen) return hpen; } if (!(penPtr = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*penPtr) ))) return 0; penPtr->logpen.elpPenStyle = pen->lopnStyle; penPtr->logpen.elpWidth = abs(pen->lopnWidth.x); penPtr->logpen.elpColor = pen->lopnColor; penPtr->logpen.elpBrushStyle = BS_SOLID; switch (pen->lopnStyle) { case PS_SOLID: case PS_DASH: case PS_DOT: case PS_DASHDOT: case PS_DASHDOTDOT: case PS_INSIDEFRAME: break; case PS_NULL: penPtr->logpen.elpWidth = 1; penPtr->logpen.elpColor = 0; break; default: penPtr->logpen.elpPenStyle = PS_SOLID; break; } if (!(hpen = alloc_gdi_handle( &penPtr->obj, OBJ_PEN, &pen_funcs ))) HeapFree( GetProcessHeap(), 0, penPtr ); return hpen; } /*********************************************************************** * ExtCreatePen (GDI32.@) */ HPEN WINAPI ExtCreatePen( DWORD style, DWORD width, const LOGBRUSH * brush, DWORD style_count, const DWORD *style_bits ) { PENOBJ *penPtr = NULL; HPEN hpen; LOGBRUSH logbrush; if ((style_count || style_bits) && (style & PS_STYLE_MASK) != PS_USERSTYLE) goto invalid; switch (style & PS_STYLE_MASK) { case PS_NULL: return CreatePen( PS_NULL, 0, brush->lbColor ); case PS_SOLID: case PS_DASH: case PS_DOT: case PS_DASHDOT: case PS_DASHDOTDOT: break; case PS_USERSTYLE: if (((INT)style_count) <= 0) return 0; if ((style_count > 16) || !style_bits) goto invalid; if ((style & PS_TYPE_MASK) == PS_GEOMETRIC) { UINT i; BOOL has_neg = FALSE, all_zero = TRUE; for(i = 0; (i < style_count) && !has_neg; i++) { has_neg = has_neg || (((INT)(style_bits[i])) < 0); all_zero = all_zero && (style_bits[i] == 0); } if (all_zero || has_neg) goto invalid; } break; case PS_INSIDEFRAME: /* applicable only for geometric pens */ if ((style & PS_TYPE_MASK) != PS_GEOMETRIC) goto invalid; break; case PS_ALTERNATE: /* applicable only for cosmetic pens */ if ((style & PS_TYPE_MASK) == PS_GEOMETRIC) goto invalid; break; default: SetLastError(ERROR_INVALID_PARAMETER); return 0; } if ((style & PS_TYPE_MASK) == PS_GEOMETRIC) { if (brush->lbStyle == BS_NULL) return CreatePen( PS_NULL, 0, 0 ); } else { if (width != 1) goto invalid; if (brush->lbStyle != BS_SOLID) goto invalid; } if (!(penPtr = HeapAlloc(GetProcessHeap(), 0, FIELD_OFFSET(PENOBJ,logpen.elpStyleEntry[style_count])))) return 0; logbrush = *brush; if (!store_brush_pattern( &logbrush, &penPtr->pattern )) goto invalid; if (logbrush.lbStyle == BS_DIBPATTERN) logbrush.lbStyle = BS_DIBPATTERNPT; penPtr->logpen.elpPenStyle = style; penPtr->logpen.elpWidth = abs((int)width); penPtr->logpen.elpBrushStyle = logbrush.lbStyle; penPtr->logpen.elpColor = logbrush.lbColor; penPtr->logpen.elpHatch = brush->lbHatch; penPtr->logpen.elpNumEntries = style_count; memcpy(penPtr->logpen.elpStyleEntry, style_bits, style_count * sizeof(DWORD)); if (!(hpen = alloc_gdi_handle( &penPtr->obj, OBJ_EXTPEN, &pen_funcs ))) { free_brush_pattern( &penPtr->pattern ); HeapFree( GetProcessHeap(), 0, penPtr ); } return hpen; invalid: HeapFree( GetProcessHeap(), 0, penPtr ); SetLastError( ERROR_INVALID_PARAMETER ); return 0; } /*********************************************************************** * PEN_SelectObject */ static HGDIOBJ PEN_SelectObject( HGDIOBJ handle, HDC hdc ) { PENOBJ *pen; HGDIOBJ ret = 0; DC *dc = get_dc_ptr( hdc ); WORD type; if (!dc) { SetLastError( ERROR_INVALID_HANDLE ); return 0; } if ((pen = get_any_obj_ptr( handle, &type ))) { struct brush_pattern *pattern; PHYSDEV physdev = GET_DC_PHYSDEV( dc, pSelectPen ); switch (type) { case OBJ_PEN: pattern = NULL; break; case OBJ_EXTPEN: pattern = &pen->pattern; if (!pattern->info) pattern = NULL; break; default: GDI_ReleaseObj( handle ); release_dc_ptr( dc ); return 0; } GDI_inc_ref_count( handle ); GDI_ReleaseObj( handle ); if (!physdev->funcs->pSelectPen( physdev, handle, pattern )) { GDI_dec_ref_count( handle ); } else { ret = dc->hPen; dc->hPen = handle; GDI_dec_ref_count( ret ); } } release_dc_ptr( dc ); return ret; } /*********************************************************************** * PEN_DeleteObject */ static BOOL PEN_DeleteObject( HGDIOBJ handle ) { PENOBJ *pen = free_gdi_handle( handle ); if (!pen) return FALSE; free_brush_pattern( &pen->pattern ); HeapFree( GetProcessHeap(), 0, pen ); return TRUE; } /*********************************************************************** * PEN_GetObject */ static INT PEN_GetObject( HGDIOBJ handle, INT count, LPVOID buffer ) { WORD type; PENOBJ *pen = get_any_obj_ptr( handle, &type ); INT ret = 0; if (!pen) return 0; switch (type) { case OBJ_PEN: { LOGPEN *lp; if (!buffer) ret = sizeof(LOGPEN); else if (count < sizeof(LOGPEN)) ret = 0; else if ((pen->logpen.elpPenStyle & PS_STYLE_MASK) == PS_NULL && count == sizeof(EXTLOGPEN)) { EXTLOGPEN *elp = buffer; *elp = pen->logpen; elp->elpWidth = 0; ret = sizeof(EXTLOGPEN); } else { lp = buffer; lp->lopnStyle = pen->logpen.elpPenStyle; lp->lopnColor = pen->logpen.elpColor; lp->lopnWidth.x = pen->logpen.elpWidth; lp->lopnWidth.y = 0; ret = sizeof(LOGPEN); } break; } case OBJ_EXTPEN: ret = sizeof(EXTLOGPEN) + pen->logpen.elpNumEntries * sizeof(DWORD) - sizeof(pen->logpen.elpStyleEntry); if (buffer) { if (count < ret) ret = 0; else memcpy(buffer, &pen->logpen, ret); } break; } GDI_ReleaseObj( handle ); return ret; }