-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathAnnotation.py
125 lines (103 loc) · 4.62 KB
/
Annotation.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
# python3
#
# Copyright 2019 The TensorFlow Authors. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""An annotation library that draws overlays on the Pi camera preview.
Annotations include bounding boxes and text overlays.
Annotations support partial opacity, however only with respect to the content in
the preview. A transparent fill value will cover up previously drawn overlay
under it, but not the camera content under it. A color of None can be given,
which will then not cover up overlay content drawn under the region.
Note: Overlays do not persist through to the storage layer so images saved from
the camera, will not contain overlays.
"""
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
from PIL import Image
from PIL import ImageDraw
def _round_up(value, n):
"""Rounds up the given value to the next number divisible by n.
Args:
value: int to be rounded up.
n: the number that should be divisible into value.
Returns:
the result of value rounded up to the next multiple of n.
"""
return n * ((value + (n - 1)) // n)
def _round_buffer_dims(dims):
"""Appropriately rounds the given dimensions for image overlaying.
As per the PiCamera.add_overlay documentation, the source data must have a
width rounded up to the nearest multiple of 32, and the height rounded up to
the nearest multiple of 16. This does that for the given image dimensions.
Args:
dims: image dimensions.
Returns:
the rounded-up dimensions in a tuple.
"""
width, height = dims
return _round_up(width, 32), _round_up(height, 16)
class Annotator:
"""Utility for managing annotations on the camera preview."""
def __init__(self, camera, default_color=None):
"""Initializes Annotator parameters.
Args:
camera: picamera.PiCamera camera object to overlay on top of.
default_color: PIL.ImageColor (with alpha) default for the drawn content.
"""
self._camera = camera
self._dims = camera.resolution
self._buffer_dims = _round_buffer_dims(self._dims)
self._buffer = Image.new('RGBA', self._buffer_dims)
self._overlay = None
self._draw = ImageDraw.Draw(self._buffer)
self._default_color = default_color or (0xFF, 0, 0, 0xFF)
def update(self):
"""Draws any changes to the image buffer onto the overlay."""
# For some reason, simply updating the current overlay causes
# PiCameraMMALError every time we update. To avoid that, we create a new
# overlay each time we want to update.
# We use a temp overlay object because if we remove the current overlay
# first, it causes flickering (the overlay visibly disappears for a moment).
temp_overlay = self._camera.add_overlay(
self._buffer.tobytes(), format='rgba', layer=3, size=self._buffer_dims)
if self._overlay is not None:
self._camera.remove_overlay(self._overlay)
self._overlay = temp_overlay
self._overlay.update(self._buffer.tobytes())
def clear(self):
"""Clears the contents of the overlay, leaving only the plain background."""
self._draw.rectangle((0, 0) + self._dims, fill=(0, 0, 0, 0x00))
def bounding_box(self, rect, outline=None, fill=None):
"""Draws a bounding box around the specified rectangle.
Args:
rect: (x1, y1, x2, y2) rectangle to be drawn, where (x1, y1) and (x2, y2)
are opposite corners of the desired rectangle.
outline: PIL.ImageColor with which to draw the outline (defaults to the
Annotator default_color).
fill: PIL.ImageColor with which to fill the rectangle (defaults to None,
which will *not* cover up drawings under the region).
"""
outline = outline or self._default_color
self._draw.rectangle(rect, fill=fill, outline=outline)
def text(self, location, text, color=None):
"""Draws the given text at the given location.
Args:
location: (x, y) point at which to draw the text (upper left corner).
text: string to be drawn.
color: PIL.ImageColor to draw the string in (defaults to the Annotator
default_color).
"""
color = color or self._default_color
self._draw.text(location, text, fill=color)