1
/*
2
 * Copyright 2008 Chris Wilson
3
 *
4
 * Permission to use, copy, modify, distribute, and sell this software
5
 * and its documentation for any purpose is hereby granted without
6
 * fee, provided that the above copyright notice appear in all copies
7
 * and that both that copyright notice and this permission notice
8
 * appear in supporting documentation, and that the name of
9
 * Chris Wilson not be used in advertising or publicity pertaining to
10
 * distribution of the software without specific, written prior
11
 * permission. Chris Wilson makes no representations about the
12
 * suitability of this software for any purpose.  It is provided "as
13
 * is" without express or implied warranty.
14
 *
15
 * CHRIS WILSON DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS
16
 * SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
17
 * FITNESS, IN NO EVENT SHALL CHRIS WILSON BE LIABLE FOR ANY SPECIAL,
18
 * INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
19
 * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
20
 * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
21
 * IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
22
 *
23
 * Author: Chris Wilson <chris@chris-wilson.co.uk>
24
 */
25

            
26
/*
27
 * Test case derived from the bug report by Michel Iwaniec:
28
 * https://lists.cairographics.org/archives/cairo/2008-November/015660.html
29
 */
30

            
31
#include "cairo-test.h"
32

            
33
static cairo_surface_t *
34
3
create_source (cairo_surface_t *target, int width, int height)
35
{
36
    cairo_surface_t *similar;
37
    cairo_t *cr;
38

            
39
3
    similar = cairo_image_surface_create (CAIRO_FORMAT_RGB24,
40
					    width, height);
41
3
    cr = cairo_create (similar);
42
3
    cairo_surface_destroy (similar);
43

            
44
3
    cairo_set_source_rgb (cr, 1, 1, 1);
45
3
    cairo_rectangle (cr,
46
3
		     width - 4, height - 4,
47
		     2, 2);
48
3
    cairo_fill (cr);
49
3
    cairo_set_source_rgb (cr, 1, 0, 0);
50
3
    cairo_rectangle (cr,
51
3
		     width - 2, height - 4,
52
		     2, 2);
53
3
    cairo_fill (cr);
54
3
    cairo_set_source_rgb (cr, 0, 1, 0);
55
3
    cairo_rectangle (cr,
56
3
		     width - 4, height - 2,
57
		     2, 2);
58
3
    cairo_fill (cr);
59
3
    cairo_set_source_rgb (cr, 0, 0, 1);
60
3
    cairo_rectangle (cr,
61
3
		     width - 2, height - 2,
62
		     2, 2);
63
3
    cairo_fill (cr);
64

            
65
3
    similar = cairo_surface_reference (cairo_get_target (cr));
66
3
    cairo_destroy (cr);
67

            
68
3
    return similar;
69
}
70

            
71
static void
72
24
draw_grid (cairo_t *cr, cairo_pattern_t *pattern, int dst_x, int dst_y)
73
{
74
    cairo_matrix_t m;
75

            
76
24
    cairo_save (cr);
77
24
    cairo_translate (cr, dst_x, dst_y);
78
24
    cairo_scale (cr, 16, 16);
79
24
    cairo_rotate (cr, 1);
80

            
81
24
    cairo_matrix_init_translate (&m, 2560-4, 1280-4);
82
24
    cairo_pattern_set_matrix (pattern, &m);
83
24
    cairo_set_source (cr, pattern);
84
24
    cairo_rectangle (cr, 0, 0, 4, 4);
85
24
    cairo_fill (cr);
86

            
87
24
    cairo_set_source_rgb (cr, .7, .7, .7);
88
24
    cairo_set_line_width (cr, 1./16);
89
24
    cairo_move_to (cr, 0, 0);
90
24
    cairo_line_to (cr, 4, 0);
91
24
    cairo_move_to (cr, 0, 2);
92
24
    cairo_line_to (cr, 4, 2);
93
24
    cairo_move_to (cr, 0, 4);
94
24
    cairo_line_to (cr, 4, 4);
95
24
    cairo_move_to (cr, 0, 0);
96
24
    cairo_line_to (cr, 0, 4);
97
24
    cairo_move_to (cr, 2, 0);
98
24
    cairo_line_to (cr, 2, 4);
99
24
    cairo_move_to (cr, 4, 0);
100
24
    cairo_line_to (cr, 4, 4);
101
24
    cairo_stroke (cr);
102

            
103
24
    cairo_restore (cr);
104
24
}
105

            
106
static cairo_test_status_t
107
3
draw (cairo_t *cr, int width, int height)
108
{
109
    cairo_surface_t *source;
110
    cairo_pattern_t *pattern;
111

            
112
3
    cairo_paint (cr);
113

            
114
3
    source = create_source (cairo_get_target (cr), 2560, 1280);
115
3
    pattern = cairo_pattern_create_for_surface (source);
116
3
    cairo_surface_destroy (source);
117

            
118
3
    cairo_pattern_set_filter (pattern, CAIRO_FILTER_NEAREST);
119
3
    cairo_pattern_set_extend (pattern, CAIRO_EXTEND_NONE);
120

            
121
3
    draw_grid (cr, pattern, 50, 0);
122
3
    draw_grid (cr, pattern, 130, 0);
123
3
    draw_grid (cr, pattern, 210, 0);
124
3
    draw_grid (cr, pattern, 290, 0);
125

            
126
3
    draw_grid (cr, pattern, 50,  230);
127
3
    draw_grid (cr, pattern, 130, 230);
128
3
    draw_grid (cr, pattern, 210, 230);
129
3
    draw_grid (cr, pattern, 290, 230);
130

            
131
3
    cairo_pattern_destroy (pattern);
132

            
133
3
    return CAIRO_TEST_SUCCESS;
134
}
135

            
136
/* XFAIL: loss of precision converting a cairo matrix to */
137
1
CAIRO_TEST (scale_offset_image,
138
	    "Tests drawing surfaces under various scales and transforms",
139
	    "surface, scale-offset", /* keywords */
140
	    NULL, /* requirements */
141
	    320, 320,
142
	    NULL, draw)
143