#!/usr/bin/env python # -*- coding: utf-8 -*- from PIL import Image import io import numpy as np import cv2 import base64 from hivision.plugin.watermark import Watermarker, WatermarkerStyles def save_image_dpi_to_bytes(image: np.ndarray, output_image_path: str = None, dpi: int = 300): """ 设置图像的DPI(每英寸点数)并返回字节流 :param image: numpy.ndarray, 输入的图像数组 :param output_image_path: Path to save the resized image. 保存调整大小后的图像的路径。 :param dpi: int, 要设置的DPI值,默认为300 """ image = Image.fromarray(image) # 创建一个字节流对象 byte_stream = io.BytesIO() # 将图像保存到字节流 image.save(byte_stream, format="PNG", dpi=(dpi, dpi)) # 获取字节流的内容 image_bytes = byte_stream.getvalue() # Save the image to the output path if output_image_path: with open(output_image_path, "wb") as f: f.write(image_bytes) return image_bytes def resize_image_to_kb(input_image: np.ndarray, output_image_path: str = None, target_size_kb: int = 100, dpi: int = 300): """ Resize an image to a target size in KB. 将图像调整大小至目标文件大小(KB)。 :param input_image_path: Path to the input image. 输入图像的路径。 :param output_image_path: Path to save the resized image. 保存调整大小后的图像的路径。 :param target_size_kb: Target size in KB. 目标文件大小(KB)。 Example: resize_image_to_kb('input_image.jpg', 'output_image.jpg', 50) """ if isinstance(input_image, np.ndarray): img = Image.fromarray(input_image) elif isinstance(input_image, Image.Image): img = input_image else: raise ValueError("input_image must be a NumPy array or PIL Image.") # Convert image to RGB mode if it's not if img.mode != "RGB": img = img.convert("RGB") # Initial quality value quality = 95 while True: # Create a BytesIO object to hold the image data in memory img_byte_arr = io.BytesIO() # Save the image to the BytesIO object with the current quality img.save(img_byte_arr, format="JPEG", quality=quality, dpi=(dpi, dpi)) # Get the size of the image in KB img_size_kb = len(img_byte_arr.getvalue()) / 1024 # Check if the image size is within the target size if img_size_kb <= target_size_kb or quality == 1: # If the image is smaller than the target size, add padding if img_size_kb < target_size_kb: padding_size = int( (target_size_kb * 1024) - len(img_byte_arr.getvalue()) ) padding = b"\x00" * padding_size img_byte_arr.write(padding) # Save the image to the output path if output_image_path: with open(output_image_path, "wb") as f: f.write(img_byte_arr.getvalue()) return img_byte_arr.getvalue() # Reduce the quality if the image is still too large quality -= 5 # Ensure quality does not go below 1 if quality < 1: quality = 1 def resize_image_to_kb_base64(input_image, target_size_kb, mode="exact"): """ Resize an image to a target size in KB and return it as a base64 encoded string. 将图像调整大小至目标文件大小(KB)并返回base64编码的字符串。 :param input_image: Input image as a NumPy array or PIL Image. 输入图像,可以是NumPy数组或PIL图像。 :param target_size_kb: Target size in KB. 目标文件大小(KB)。 :param mode: Mode of resizing ('exact', 'max', 'min'). 模式:'exact'(精确大小)、'max'(不大于)、'min'(不小于)。 :return: Base64 encoded string of the resized image. 调整大小后的图像的base64编码字符串。 """ if isinstance(input_image, np.ndarray): img = Image.fromarray(input_image) elif isinstance(input_image, Image.Image): img = input_image else: raise ValueError("input_image must be a NumPy array or PIL Image.") # Convert image to RGB mode if it's not if img.mode != "RGB": img = img.convert("RGB") # Initial quality value quality = 95 while True: # Create a BytesIO object to hold the image data in memory img_byte_arr = io.BytesIO() # Save the image to the BytesIO object with the current quality img.save(img_byte_arr, format="JPEG", quality=quality) # Get the size of the image in KB img_size_kb = len(img_byte_arr.getvalue()) / 1024 # Check based on the mode if mode == "exact": # If the image size is equal to the target size, we can return it if img_size_kb == target_size_kb: break # If the image is smaller than the target size, add padding elif img_size_kb < target_size_kb: padding_size = int( (target_size_kb * 1024) - len(img_byte_arr.getvalue()) ) padding = b"\x00" * padding_size img_byte_arr.write(padding) break elif mode == "max": # If the image size is within the target size, we can return it if img_size_kb <= target_size_kb or quality == 1: break elif mode == "min": # If the image size is greater than or equal to the target size, we can return it if img_size_kb >= target_size_kb: break # Reduce the quality if the image is still too large quality -= 5 # Ensure quality does not go below 1 if quality < 1: quality = 1 # Encode the image data to base64 img_base64 = base64.b64encode(img_byte_arr.getvalue()).decode("utf-8") return "data:image/png;base64," + img_base64 def numpy_2_base64(img: np.ndarray) -> str: _, buffer = cv2.imencode(".png", img) base64_image = base64.b64encode(buffer).decode("utf-8") return "data:image/png;base64," + base64_image def base64_2_numpy(base64_image: str) -> np.ndarray: # Remove the data URL prefix if present if base64_image.startswith('data:image'): base64_image = base64_image.split(',')[1] # Decode base64 string to bytes img_bytes = base64.b64decode(base64_image) # Convert bytes to numpy array img_array = np.frombuffer(img_bytes, dtype=np.uint8) # Decode the image array img = cv2.imdecode(img_array, cv2.IMREAD_UNCHANGED) return img # 字节流转base64 def bytes_2_base64(img_byte_arr: bytes) -> str: base64_image = base64.b64encode(img_byte_arr).decode("utf-8") return "data:image/png;base64," + base64_image def save_numpy_image(numpy_img, file_path): # 检查数组的形状 if numpy_img.shape[2] == 4: # 将 BGR 转换为 RGB,并保留透明通道 rgb_img = np.concatenate( (np.flip(numpy_img[:, :, :3], axis=-1), numpy_img[:, :, 3:]), axis=-1 ).astype(np.uint8) img = Image.fromarray(rgb_img, mode="RGBA") else: # 将 BGR 转换为 RGB rgb_img = np.flip(numpy_img, axis=-1).astype(np.uint8) img = Image.fromarray(rgb_img, mode="RGB") img.save(file_path) def numpy_to_bytes(numpy_img): img = Image.fromarray(numpy_img) img_byte_arr = io.BytesIO() img.save(img_byte_arr, format="PNG") img_byte_arr.seek(0) return img_byte_arr def hex_to_rgb(value): value = value.lstrip("#") length = len(value) return tuple( int(value[i : i + length // 3], 16) for i in range(0, length, length // 3) ) def generate_gradient(start_color, width, height, mode="updown"): # 定义背景颜色 end_color = (255, 255, 255) # 白色 # 创建一个空白图像 r_out = np.zeros((height, width), dtype=int) g_out = np.zeros((height, width), dtype=int) b_out = np.zeros((height, width), dtype=int) if mode == "updown": # 生成上下渐变色 for y in range(height): r = int( (y / height) * end_color[0] + ((height - y) / height) * start_color[0] ) g = int( (y / height) * end_color[1] + ((height - y) / height) * start_color[1] ) b = int( (y / height) * end_color[2] + ((height - y) / height) * start_color[2] ) r_out[y, :] = r g_out[y, :] = g b_out[y, :] = b else: # 生成中心渐变色 img = np.zeros((height, width, 3)) # 定义椭圆中心和半径 center = (width // 2, height // 2) end_axies = max(height, width) # 定义渐变色 end_color = (255, 255, 255) # 绘制椭圆 for y in range(end_axies): axes = (end_axies - y, end_axies - y) r = int( (y / end_axies) * end_color[0] + ((end_axies - y) / end_axies) * start_color[0] ) g = int( (y / end_axies) * end_color[1] + ((end_axies - y) / end_axies) * start_color[1] ) b = int( (y / end_axies) * end_color[2] + ((end_axies - y) / end_axies) * start_color[2] ) cv2.ellipse(img, center, axes, 0, 0, 360, (b, g, r), -1) b_out, g_out, r_out = cv2.split(np.uint64(img)) return r_out, g_out, b_out def add_background(input_image, bgr=(0, 0, 0), mode="pure_color"): """ 本函数的功能为为透明图像加上背景。 :param input_image: numpy.array(4 channels), 透明图像 :param bgr: tuple, 合成纯色底时的 BGR 值 :param new_background: numpy.array(3 channels),合成自定义图像底时的背景图 :return: output: 合成好的输出图像 """ height, width = input_image.shape[0], input_image.shape[1] try: b, g, r, a = cv2.split(input_image) except ValueError: raise ValueError( "The input image must have 4 channels. 输入图像必须有4个通道,即透明图像。" ) a_cal = a / 255 if mode == "pure_color": # 纯色填充 b2 = np.full([height, width], bgr[0], dtype=int) g2 = np.full([height, width], bgr[1], dtype=int) r2 = np.full([height, width], bgr[2], dtype=int) elif mode == "updown_gradient": b2, g2, r2 = generate_gradient(bgr, width, height, mode="updown") else: b2, g2, r2 = generate_gradient(bgr, width, height, mode="center") output = cv2.merge( ((b - b2) * a_cal + b2, (g - g2) * a_cal + g2, (r - r2) * a_cal + r2) ) return output def add_background_with_image(input_image: np.ndarray, background_image: np.ndarray) -> np.ndarray: """ 本函数的功能为为透明图像加上背景。 :param input_image: numpy.array(4 channels), 透明图像 :param background_image: numpy.array(3 channels), 背景图像 :return: output: 合成好的输出图像 """ height, width = input_image.shape[:2] try: b, g, r, a = cv2.split(input_image) except ValueError: raise ValueError( "The input image must have 4 channels. 输入图像必须有4个通道,即透明图像。" ) # 确保背景图像与输入图像大小一致 background_image = cv2.resize(background_image, (width, height), cv2.INTER_AREA) background_image = cv2.cvtColor(background_image, cv2.COLOR_BGR2RGB) b2, g2, r2 = cv2.split(background_image) a_cal = a / 255.0 # 修正混合公式 output = cv2.merge( (b * a_cal + b2 * (1 - a_cal), g * a_cal + g2 * (1 - a_cal), r * a_cal + r2 * (1 - a_cal)) ) return output.astype(np.uint8) def add_watermark( image, text, size=50, opacity=0.5, angle=45, color="#8B8B1B", space=75 ): image = Image.fromarray(image) watermarker = Watermarker( input_image=image, text=text, style=WatermarkerStyles.STRIPED, angle=angle, color=color, opacity=opacity, size=size, space=space, ) return np.array(watermarker.image.convert("RGB"))