|
|
|
""" |
|
Plotting utils |
|
""" |
|
|
|
import os |
|
from pathlib import Path |
|
|
|
import cv2 |
|
|
|
import numpy as np |
|
import torch |
|
from PIL import Image, ImageDraw, ImageFont |
|
|
|
from .general import (LOGGER, clip_coords, increment_path, is_ascii, is_chinese, |
|
user_config_dir, xywh2xyxy, xyxy2xywh) |
|
|
|
|
|
CONFIG_DIR = user_config_dir() |
|
RANK = int(os.getenv('RANK', -1)) |
|
|
|
|
|
|
|
class Colors: |
|
|
|
def __init__(self): |
|
|
|
hex = ('FF3838', 'FF9D97', 'FF701F', 'FFB21D', 'CFD231', '48F90A', '92CC17', '3DDB86', '1A9334', '00D4BB', |
|
'2C99A8', '00C2FF', '344593', '6473FF', '0018EC', '8438FF', '520085', 'CB38FF', 'FF95C8', 'FF37C7') |
|
self.palette = [self.hex2rgb('#' + c) for c in hex] |
|
self.n = len(self.palette) |
|
|
|
def __call__(self, i, bgr=False): |
|
c = self.palette[int(i) % self.n] |
|
return (c[2], c[1], c[0]) if bgr else c |
|
|
|
@staticmethod |
|
def hex2rgb(h): |
|
return tuple(int(h[1 + i:1 + i + 2], 16) for i in (0, 2, 4)) |
|
|
|
|
|
colors = Colors() |
|
|
|
|
|
def check_font(font='Arial.ttf', size=10): |
|
|
|
font = Path(font) |
|
font = font if font.exists() else (CONFIG_DIR / font.name) |
|
try: |
|
return ImageFont.truetype(str(font) if font.exists() else font.name, size) |
|
except Exception as e: |
|
url = "https://ultralytics.com/assets/" + font.name |
|
LOGGER.info(f'Downloading {url} to {font}...') |
|
torch.hub.download_url_to_file(url, str(font), progress=False) |
|
try: |
|
return ImageFont.truetype(str(font), size) |
|
except TypeError: |
|
pass |
|
|
|
class Annotator: |
|
if RANK in (-1, 0): |
|
check_font() |
|
|
|
|
|
def __init__(self, im, line_width=None, font_size=None, font='Arial.ttf', pil=False, example='abc'): |
|
assert im.data.contiguous, 'Image not contiguous. Apply np.ascontiguousarray(im) to Annotator() input images.' |
|
self.pil = pil or not is_ascii(example) or is_chinese(example) |
|
if self.pil: |
|
self.im = im if isinstance(im, Image.Image) else Image.fromarray(im) |
|
self.draw = ImageDraw.Draw(self.im) |
|
self.font = check_font(font='Arial.Unicode.ttf' if is_chinese(example) else font, |
|
size=font_size or max(round(sum(self.im.size) / 2 * 0.035), 12)) |
|
else: |
|
self.im = im |
|
self.lw = line_width or max(round(sum(im.shape) / 2 * 0.003), 2) |
|
|
|
def box_label(self, box, label='', color=(128, 128, 128), txt_color=(255, 255, 255)): |
|
|
|
if self.pil or not is_ascii(label): |
|
self.draw.rectangle(box, width=self.lw, outline=color) |
|
if label: |
|
w, h = self.font.getsize(label) |
|
outside = box[1] - h >= 0 |
|
self.draw.rectangle([box[0], |
|
box[1] - h if outside else box[1], |
|
box[0] + w + 1, |
|
box[1] + 1 if outside else box[1] + h + 1], fill=color) |
|
|
|
self.draw.text((box[0], box[1] - h if outside else box[1]), label, fill=txt_color, font=self.font) |
|
else: |
|
p1, p2 = (int(box[0]), int(box[1])), (int(box[2]), int(box[3])) |
|
cv2.rectangle(self.im, p1, p2, color, thickness=self.lw, lineType=cv2.LINE_AA) |
|
if label: |
|
tf = max(self.lw - 1, 1) |
|
w, h = cv2.getTextSize(label, 0, fontScale=self.lw / 3, thickness=tf)[0] |
|
outside = p1[1] - h - 3 >= 0 |
|
p2 = p1[0] + w, p1[1] - h - 3 if outside else p1[1] + h + 3 |
|
cv2.rectangle(self.im, p1, p2, color, -1, cv2.LINE_AA) |
|
cv2.putText(self.im, label, (p1[0], p1[1] - 2 if outside else p1[1] + h + 2), 0, self.lw / 3, txt_color, |
|
thickness=tf, lineType=cv2.LINE_AA) |
|
|
|
def rectangle(self, xy, fill=None, outline=None, width=1): |
|
|
|
self.draw.rectangle(xy, fill, outline, width) |
|
|
|
def text(self, xy, text, txt_color=(255, 255, 255)): |
|
|
|
w, h = self.font.getsize(text) |
|
self.draw.text((xy[0], xy[1] - h + 1), text, fill=txt_color, font=self.font) |
|
|
|
def result(self): |
|
|
|
return np.asarray(self.im) |
|
|
|
|
|
def save_one_box(xyxy, im, file='image.jpg', gain=1.02, pad=10, square=False, BGR=False, save=True): |
|
|
|
xyxy = torch.tensor(xyxy).view(-1, 4) |
|
b = xyxy2xywh(xyxy) |
|
if square: |
|
b[:, 2:] = b[:, 2:].max(1)[0].unsqueeze(1) |
|
b[:, 2:] = b[:, 2:] * gain + pad |
|
xyxy = xywh2xyxy(b).long() |
|
clip_coords(xyxy, im.shape) |
|
crop = im[int(xyxy[0, 1]):int(xyxy[0, 3]), int(xyxy[0, 0]):int(xyxy[0, 2]), ::(1 if BGR else -1)] |
|
if save: |
|
file.parent.mkdir(parents=True, exist_ok=True) |
|
cv2.imwrite(str(increment_path(file).with_suffix('.jpg')), crop) |
|
return crop |
|
|