mirror of
https://github.com/python-pillow/Pillow.git
synced 2025-01-26 01:04:29 +03:00
Merge pull request #5694 from radarhere/polygon_width
This commit is contained in:
commit
c857366d39
BIN
Tests/images/imagedraw/triangle_right_width.png
Normal file
BIN
Tests/images/imagedraw/triangle_right_width.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 472 B |
BIN
Tests/images/imagedraw/triangle_right_width_no_fill.png
Normal file
BIN
Tests/images/imagedraw/triangle_right_width_no_fill.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 479 B |
|
@ -945,6 +945,18 @@ def test_triangle_right():
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"fill, suffix",
|
||||||
|
((BLACK, "width"), (None, "width_no_fill")),
|
||||||
|
)
|
||||||
|
def test_triangle_right_width(fill, suffix):
|
||||||
|
img, draw = create_base_image_draw((100, 100))
|
||||||
|
draw.polygon([(15, 25), (85, 25), (50, 60)], fill, WHITE, width=5)
|
||||||
|
assert_image_equal_tofile(
|
||||||
|
img, os.path.join(IMAGES_PATH, "triangle_right_" + suffix + ".png")
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def test_line_horizontal():
|
def test_line_horizontal():
|
||||||
img, draw = create_base_image_draw((20, 20))
|
img, draw = create_base_image_draw((20, 20))
|
||||||
draw.line((5, 5, 14, 5), BLACK, 2)
|
draw.line((5, 5, 14, 5), BLACK, 2)
|
||||||
|
|
|
@ -233,13 +233,35 @@ class ImageDraw:
|
||||||
if ink is not None:
|
if ink is not None:
|
||||||
self.draw.draw_points(xy, ink)
|
self.draw.draw_points(xy, ink)
|
||||||
|
|
||||||
def polygon(self, xy, fill=None, outline=None):
|
def polygon(self, xy, fill=None, outline=None, width=1):
|
||||||
"""Draw a polygon."""
|
"""Draw a polygon."""
|
||||||
ink, fill = self._getink(outline, fill)
|
ink, fill = self._getink(outline, fill)
|
||||||
if fill is not None:
|
if fill is not None:
|
||||||
self.draw.draw_polygon(xy, fill, 1)
|
self.draw.draw_polygon(xy, fill, 1)
|
||||||
if ink is not None and ink != fill:
|
if ink is not None and ink != fill and width != 0:
|
||||||
self.draw.draw_polygon(xy, ink, 0)
|
if width == 1:
|
||||||
|
self.draw.draw_polygon(xy, ink, 0, width)
|
||||||
|
else:
|
||||||
|
# To avoid expanding the polygon outwards,
|
||||||
|
# use the fill as a mask
|
||||||
|
mask = Image.new("1", self.im.size)
|
||||||
|
mask_ink = self._getink(1)[0]
|
||||||
|
|
||||||
|
fill_im = mask.copy()
|
||||||
|
draw = Draw(fill_im)
|
||||||
|
draw.draw.draw_polygon(xy, mask_ink, 1)
|
||||||
|
|
||||||
|
ink_im = mask.copy()
|
||||||
|
draw = Draw(ink_im)
|
||||||
|
width = width * 2 - 1
|
||||||
|
draw.draw.draw_polygon(xy, mask_ink, 0, width)
|
||||||
|
|
||||||
|
mask.paste(ink_im, mask=fill_im)
|
||||||
|
|
||||||
|
im = Image.new(self.mode, self.im.size)
|
||||||
|
draw = Draw(im)
|
||||||
|
draw.draw.draw_polygon(xy, ink, 0, width)
|
||||||
|
self.im.paste(im.im, (0, 0) + im.size, mask.im)
|
||||||
|
|
||||||
def regular_polygon(
|
def regular_polygon(
|
||||||
self, bounding_circle, n_sides, rotation=0, fill=None, outline=None
|
self, bounding_circle, n_sides, rotation=0, fill=None, outline=None
|
||||||
|
|
|
@ -3124,7 +3124,8 @@ _draw_polygon(ImagingDrawObject *self, PyObject *args) {
|
||||||
PyObject *data;
|
PyObject *data;
|
||||||
int ink;
|
int ink;
|
||||||
int fill = 0;
|
int fill = 0;
|
||||||
if (!PyArg_ParseTuple(args, "Oi|i", &data, &ink, &fill)) {
|
int width = 0;
|
||||||
|
if (!PyArg_ParseTuple(args, "Oi|ii", &data, &ink, &fill, &width)) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3153,7 +3154,7 @@ _draw_polygon(ImagingDrawObject *self, PyObject *args) {
|
||||||
|
|
||||||
free(xy);
|
free(xy);
|
||||||
|
|
||||||
if (ImagingDrawPolygon(self->image->image, n, ixy, &ink, fill, self->blend) < 0) {
|
if (ImagingDrawPolygon(self->image->image, n, ixy, &ink, fill, width, self->blend) < 0) {
|
||||||
free(ixy);
|
free(ixy);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
|
@ -742,7 +742,7 @@ ImagingDrawRectangle(
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
ImagingDrawPolygon(Imaging im, int count, int *xy, const void *ink_, int fill, int op) {
|
ImagingDrawPolygon(Imaging im, int count, int *xy, const void *ink_, int fill, int width, int op) {
|
||||||
int i, n, x0, y0, x1, y1;
|
int i, n, x0, y0, x1, y1;
|
||||||
DRAW *draw;
|
DRAW *draw;
|
||||||
INT32 ink;
|
INT32 ink;
|
||||||
|
@ -790,10 +790,17 @@ ImagingDrawPolygon(Imaging im, int count, int *xy, const void *ink_, int fill, i
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
/* Outline */
|
/* Outline */
|
||||||
for (i = 0; i < count - 1; i++) {
|
if (width == 1) {
|
||||||
draw->line(im, xy[i * 2], xy[i * 2 + 1], xy[i * 2 + 2], xy[i * 2 + 3], ink);
|
for (i = 0; i < count - 1; i++) {
|
||||||
|
draw->line(im, xy[i * 2], xy[i * 2 + 1], xy[i * 2 + 2], xy[i * 2 + 3], ink);
|
||||||
|
}
|
||||||
|
draw->line(im, xy[i * 2], xy[i * 2 + 1], xy[0], xy[1], ink);
|
||||||
|
} else {
|
||||||
|
for (i = 0; i < count - 1; i++) {
|
||||||
|
ImagingDrawWideLine(im, xy[i * 2], xy[i * 2 + 1], xy[i * 2 + 2], xy[i * 2 + 3], ink_, width, op);
|
||||||
|
}
|
||||||
|
ImagingDrawWideLine(im, xy[i * 2], xy[i * 2 + 1], xy[0], xy[1], ink_, width, op);
|
||||||
}
|
}
|
||||||
draw->line(im, xy[i * 2], xy[i * 2 + 1], xy[0], xy[1], ink);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -487,7 +487,7 @@ ImagingDrawPieslice(
|
||||||
extern int
|
extern int
|
||||||
ImagingDrawPoint(Imaging im, int x, int y, const void *ink, int op);
|
ImagingDrawPoint(Imaging im, int x, int y, const void *ink, int op);
|
||||||
extern int
|
extern int
|
||||||
ImagingDrawPolygon(Imaging im, int points, int *xy, const void *ink, int fill, int op);
|
ImagingDrawPolygon(Imaging im, int points, int *xy, const void *ink, int fill, int width, int op);
|
||||||
extern int
|
extern int
|
||||||
ImagingDrawRectangle(
|
ImagingDrawRectangle(
|
||||||
Imaging im,
|
Imaging im,
|
||||||
|
|
Loading…
Reference in New Issue
Block a user