1
/*
2
 * Copyright © 2011 Intel Corporation
3
 *
4
 * Permission is hereby granted, free of charge, to any person
5
 * obtaining a copy of this software and associated documentation
6
 * files (the "Software"), to deal in the Software without
7
 * restriction, including without limitation the rights to use, copy,
8
 * modify, merge, publish, distribute, sublicense, and/or sell copies
9
 * of the Software, and to permit persons to whom the Software is
10
 * furnished to do so, subject to the following conditions:
11
 *
12
 * The above copyright notice and this permission notice shall be
13
 * included in all copies or substantial portions of the Software.
14
 *
15
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
17
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
18
 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
19
 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
20
 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
21
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
22
 * SOFTWARE.
23
 *
24
 * Author: Chris Wilson <chris@chris-wilson.co.uk>
25
 */
26

            
27
#include "cairo-test.h"
28

            
29
#define LINE_WIDTH	30.
30
#define SIZE		(2 * LINE_WIDTH)
31
#define PAD		(2 * LINE_WIDTH)
32

            
33
static void
34
324
make_path (cairo_t *cr, double theta)
35
{
36
324
    double line_width = cairo_get_line_width (cr) / 4;
37

            
38
324
    cairo_move_to (cr, 0, 0);
39
324
    cairo_rel_curve_to (cr,
40
			SIZE/3, -SIZE/4,
41
			SIZE/3, -SIZE/4,
42
			SIZE, 0);
43

            
44
324
    cairo_rel_line_to (cr,
45
324
		       cos (theta) * line_width,
46
324
		       sin (theta) * line_width);
47
324
}
48

            
49
static void
50
108
draw_joins (cairo_t *cr, double theta)
51
{
52
108
    make_path (cr, theta);
53
108
    cairo_set_line_join (cr, CAIRO_LINE_JOIN_BEVEL);
54
108
    cairo_stroke (cr);
55

            
56
108
    cairo_translate (cr, SIZE + PAD, 0.);
57

            
58
108
    make_path (cr, theta);
59
108
    cairo_set_line_join (cr, CAIRO_LINE_JOIN_ROUND);
60
108
    cairo_stroke (cr);
61

            
62
108
    cairo_translate (cr, SIZE + PAD, 0.);
63

            
64
108
    make_path (cr, theta);
65
108
    cairo_set_line_join (cr, CAIRO_LINE_JOIN_MITER);
66
108
    cairo_stroke (cr);
67

            
68
108
    cairo_translate (cr, SIZE + PAD, 0.);
69
108
}
70

            
71
static void
72
36
draw_caps_joins (cairo_t *cr, double theta)
73
{
74
36
    cairo_translate (cr, PAD, 0);
75
36
    cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
76
36
    draw_joins (cr, theta);
77

            
78
36
    cairo_translate (cr, PAD, 0);
79
36
    cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);
80
36
    draw_joins (cr, theta);
81

            
82
36
    cairo_translate (cr, PAD, 0);
83
36
    cairo_set_line_cap (cr, CAIRO_LINE_CAP_SQUARE);
84
36
    draw_joins (cr, theta);
85
36
}
86

            
87
static cairo_test_status_t
88
3
draw (cairo_t *cr, int width, int height)
89
{
90
3
    const double theta[] = {
91
	-M_PI/2, -M_PI/4, 0, M_PI/8, M_PI/3, M_PI
92
    };
93
    unsigned int t;
94

            
95
    /* We draw in the default black, so paint white first. */
96
3
    cairo_save (cr);
97
3
    cairo_set_source_rgb (cr, 1.0, 1.0, 1.0); /* white */
98
3
    cairo_paint (cr);
99
3
    cairo_restore (cr);
100

            
101
3
    cairo_set_line_width (cr, LINE_WIDTH);
102

            
103
21
    for (t = 0; t < ARRAY_LENGTH(theta); t++) {
104
18
	cairo_save (cr);
105
18
	cairo_translate (cr, 0, t * (SIZE + PAD) + PAD);
106
18
	draw_caps_joins (cr, theta[t]);
107
18
	cairo_restore (cr);
108

            
109
18
	cairo_save (cr);
110
	/* and reflect to generate the opposite vertex ordering */
111
18
	cairo_translate (cr, 0, height - t * (SIZE + PAD) - PAD);
112
18
	cairo_scale (cr, 1, -1);
113
18
	draw_caps_joins (cr, theta[t]);
114
18
	cairo_restore (cr);
115
    }
116

            
117
3
    return CAIRO_TEST_SUCCESS;
118
}
119

            
120
1
CAIRO_TEST (caps_tails_curve,
121
	    "Test caps and joins on short tail segments",
122
	    "stroke, cap, join", /* keywords */
123
	    NULL, /* requirements */
124
	    9 * (PAD + SIZE) + 4*PAD,
125
	    12 * (PAD + SIZE) + PAD,
126
	    NULL, draw)
127