/** * cairo_win32_surface_get_image: * @surface: a #cairo_surface_t * * Returns a #cairo_surface_t image surface that refers to the same bits * as the DIB of the Win32 surface. If the passed-in win32 surface * is not a DIB surface, %NULL is returned. * * Return value: a #cairo_surface_t (owned by the win32 #cairo_surface_t), * or %NULL if the win32 surface is not a DIB. * * Since: 1.4 **/ cairo_surface_t * cairo_win32_surface_get_image (cairo_surface_t *surface) { if (! _cairo_surface_is_win32 (surface)) { return _cairo_surface_create_in_error (_cairo_error (CAIRO_STATUS_SURFACE_TYPE_MISMATCH)); } GdiFlush(); return to_win32_display_surface(surface)->image; }
cairo_warn cairo_int_status_t _cairo_dwrite_scaled_show_glyphs(void *scaled_font, cairo_operator_t op, const cairo_pattern_t *pattern, cairo_surface_t *generic_surface, int source_x, int source_y, int dest_x, int dest_y, unsigned int width, unsigned int height, cairo_glyph_t *glyphs, int num_glyphs, cairo_region_t *clip_region, int *remaining_glyphs) { cairo_win32_surface_t *surface = (cairo_win32_surface_t *)generic_surface; cairo_int_status_t status; if (width == 0 || height == 0) return (cairo_int_status_t)CAIRO_STATUS_SUCCESS; if (_cairo_surface_is_win32 (generic_surface) && surface->format == CAIRO_FORMAT_RGB24 && op == CAIRO_OPERATOR_OVER) { //XXX: we need to set the clip region here status = (cairo_int_status_t)_cairo_dwrite_show_glyphs_on_surface (surface, op, pattern, glyphs, num_glyphs, (cairo_scaled_font_t*)scaled_font, NULL); return status; } else { cairo_dwrite_scaled_font_t *dwritesf = static_cast<cairo_dwrite_scaled_font_t*>(scaled_font); UINT16 *indices = new UINT16[num_glyphs]; DWRITE_GLYPH_OFFSET *offsets = new DWRITE_GLYPH_OFFSET[num_glyphs]; FLOAT *advances = new FLOAT[num_glyphs]; BOOL transform = FALSE; DWRITE_GLYPH_RUN run; run.bidiLevel = 0; run.fontFace = ((cairo_dwrite_font_face_t*)dwritesf->base.font_face)->dwriteface; run.glyphIndices = indices; run.glyphCount = num_glyphs; run.isSideways = FALSE; run.glyphOffsets = offsets; run.glyphAdvances = advances; IDWriteGlyphRunAnalysis *analysis; if (dwritesf->mat.xy == 0 && dwritesf->mat.yx == 0 && dwritesf->mat.xx == dwritesf->base.font_matrix.xx && dwritesf->mat.yy == dwritesf->base.font_matrix.yy) { for (int i = 0; i < num_glyphs; i++) { indices[i] = (WORD) glyphs[i].index; // Since we will multiply by our ctm matrix later for rotation effects // and such, adjust positions by the inverse matrix now. offsets[i].ascenderOffset = (FLOAT)dest_y - (FLOAT)glyphs[i].y; offsets[i].advanceOffset = (FLOAT)glyphs[i].x - dest_x; advances[i] = 0.0; } run.fontEmSize = (FLOAT)dwritesf->base.font_matrix.yy; } else { transform = TRUE; for (int i = 0; i < num_glyphs; i++) { indices[i] = (WORD) glyphs[i].index; double x = glyphs[i].x - dest_x; double y = glyphs[i].y - dest_y; cairo_matrix_transform_point(&dwritesf->mat_inverse, &x, &y); // Since we will multiply by our ctm matrix later for rotation effects // and such, adjust positions by the inverse matrix now. offsets[i].ascenderOffset = -(FLOAT)y; offsets[i].advanceOffset = (FLOAT)x; advances[i] = 0.0; } run.fontEmSize = 1.0f; } if (!transform) { DWriteFactory::Instance()->CreateGlyphRunAnalysis(&run, 1.0f, NULL, DWRITE_RENDERING_MODE_CLEARTYPE_NATURAL_SYMMETRIC, DWRITE_MEASURING_MODE_NATURAL, 0, 0, &analysis); } else { DWRITE_MATRIX dwmatrix = _cairo_dwrite_matrix_from_matrix(&dwritesf->mat); DWriteFactory::Instance()->CreateGlyphRunAnalysis(&run, 1.0f, &dwmatrix, DWRITE_RENDERING_MODE_CLEARTYPE_NATURAL_SYMMETRIC, DWRITE_MEASURING_MODE_NATURAL, 0, 0, &analysis); } RECT r; r.left = 0; r.top = 0; r.right = width; r.bottom = height; BYTE *surface = new BYTE[width * height * 3]; analysis->CreateAlphaTexture(DWRITE_TEXTURE_CLEARTYPE_3x1, &r, surface, width * height * 3); cairo_image_surface_t *mask_surface = (cairo_image_surface_t*)cairo_image_surface_create(CAIRO_FORMAT_ARGB32, width, height); cairo_surface_flush(&mask_surface->base); for (unsigned int y = 0; y < height; y++) { for (unsigned int x = 0; x < width; x++) { mask_surface->data[y * mask_surface->stride + x * 4] = surface[y * width * 3 + x * 3 + 1]; mask_surface->data[y * mask_surface->stride + x * 4 + 1] = surface[y * width * 3 + x * 3 + 1]; mask_surface->data[y * mask_surface->stride + x * 4 + 2] = surface[y * width * 3 + x * 3 + 1]; mask_surface->data[y * mask_surface->stride + x * 4 + 3] = surface[y * width * 3 + x * 3 + 1]; } } cairo_surface_mark_dirty(&mask_surface->base); pixman_image_set_component_alpha(mask_surface->pixman_image, 1); cairo_surface_pattern_t mask; _cairo_pattern_init_for_surface (&mask, &mask_surface->base); status = (cairo_int_status_t)_cairo_surface_composite (op, pattern, &mask.base, generic_surface, source_x, source_y, 0, 0, dest_x, dest_y, width, height, clip_region); _cairo_pattern_fini (&mask.base); analysis->Release(); delete [] surface; delete [] indices; delete [] offsets; delete [] advances; cairo_surface_destroy (&mask_surface->base); *remaining_glyphs = 0; return (cairo_int_status_t)CAIRO_STATUS_SUCCESS; } }