1.1.0版本
This commit is contained in:
288
utils/data_range_converter.py
Normal file
288
utils/data_range_converter.py
Normal file
@@ -0,0 +1,288 @@
|
||||
# -*- coding: UTF-8 -*-
|
||||
"""
|
||||
数据范围转换器
|
||||
将 Full Range (0-255) 转换为 Limited Range (16-235)
|
||||
|
||||
使用方法:
|
||||
from utils.data_range_converter import DataRangeConverter
|
||||
|
||||
converter = DataRangeConverter()
|
||||
converted_params = converter.convert(pattern_params, "Limited")
|
||||
"""
|
||||
|
||||
|
||||
class DataRangeConverter:
|
||||
"""数据范围转换器"""
|
||||
|
||||
def __init__(self, verbose=True):
|
||||
"""
|
||||
初始化转换器
|
||||
|
||||
Args:
|
||||
verbose: 是否打印详细日志
|
||||
"""
|
||||
self.verbose = verbose
|
||||
|
||||
def convert_value(self, value):
|
||||
"""
|
||||
将单个 Full Range 值转换为 Limited Range
|
||||
|
||||
转换公式:
|
||||
limited = 16 + (value / 255) × (235 - 16)
|
||||
|
||||
Args:
|
||||
value: Full Range 值 (0-255)
|
||||
|
||||
Returns:
|
||||
int: Limited Range 值 (16-235)
|
||||
"""
|
||||
# 边界值直接映射
|
||||
if value == 0:
|
||||
return 16
|
||||
elif value == 255:
|
||||
return 235
|
||||
else:
|
||||
# 线性映射
|
||||
limited = 16 + round((value / 255.0) * (235 - 16))
|
||||
# 限制范围
|
||||
return max(16, min(235, limited))
|
||||
|
||||
def convert_rgb(self, r, g, b):
|
||||
"""
|
||||
转换单个 RGB 值
|
||||
|
||||
Args:
|
||||
r, g, b: Full Range RGB 值 (0-255)
|
||||
|
||||
Returns:
|
||||
tuple: Limited Range RGB 值 (16-235)
|
||||
"""
|
||||
return (self.convert_value(r), self.convert_value(g), self.convert_value(b))
|
||||
|
||||
def convert(self, pattern_params, data_range="Full"):
|
||||
"""
|
||||
转换图案参数列表
|
||||
|
||||
Args:
|
||||
pattern_params: 图案参数列表 [[r,g,b], [r,g,b], ...]
|
||||
data_range: "Full" 或 "Limited"
|
||||
|
||||
Returns:
|
||||
list: 转换后的图案参数列表
|
||||
"""
|
||||
# Full Range 不需要转换
|
||||
if data_range == "Full":
|
||||
if self.verbose:
|
||||
print("✓ 使用 Full Range (0-255),无需转换")
|
||||
return pattern_params
|
||||
|
||||
# Limited Range 需要转换
|
||||
if data_range == "Limited":
|
||||
if self.verbose:
|
||||
self._print_header()
|
||||
|
||||
converted = []
|
||||
for i, rgb in enumerate(pattern_params):
|
||||
# 获取原始值
|
||||
r_orig, g_orig, b_orig = rgb[0], rgb[1], rgb[2]
|
||||
|
||||
# 转换
|
||||
r_new, g_new, b_new = self.convert_rgb(r_orig, g_orig, b_orig)
|
||||
|
||||
# 保存
|
||||
converted.append([r_new, g_new, b_new])
|
||||
|
||||
# 打印日志(关键值)
|
||||
if self.verbose:
|
||||
self._print_conversion(
|
||||
i, r_orig, g_orig, b_orig, r_new, g_new, b_new
|
||||
)
|
||||
|
||||
if self.verbose:
|
||||
self._print_footer(len(pattern_params))
|
||||
|
||||
return converted
|
||||
|
||||
# 未知范围,返回原始值
|
||||
else:
|
||||
if self.verbose:
|
||||
print(f"⚠ 未知的数据范围: {data_range},使用原始值")
|
||||
return pattern_params
|
||||
|
||||
def _print_header(self):
|
||||
"""打印转换头部信息"""
|
||||
print("=" * 80)
|
||||
print("【数据范围转换】Limited Range (16-235)")
|
||||
print(" 转换公式: 16 + (value / 255) × (235 - 16)")
|
||||
print("=" * 80)
|
||||
|
||||
def _print_conversion(self, index, r_orig, g_orig, b_orig, r_new, g_new, b_new):
|
||||
"""
|
||||
打印转换日志
|
||||
|
||||
策略:只打印关键值,避免刷屏
|
||||
- 第一个和最后一个图案
|
||||
- 0, 128, 255 等关键值
|
||||
"""
|
||||
# 判断是否需要打印
|
||||
should_print = False
|
||||
|
||||
# 第一个和最后一个
|
||||
if index == 0:
|
||||
should_print = True
|
||||
label = "黑色"
|
||||
elif index == len([]) - 1: # 需要在外部判断
|
||||
should_print = True
|
||||
label = "白色"
|
||||
# 关键 RGB 值
|
||||
elif (
|
||||
r_orig in [0, 128, 255]
|
||||
or g_orig in [0, 128, 255]
|
||||
or b_orig in [0, 128, 255]
|
||||
):
|
||||
should_print = True
|
||||
if r_orig == 128:
|
||||
label = "50% 灰"
|
||||
else:
|
||||
label = ""
|
||||
|
||||
if should_print:
|
||||
diff = abs(r_new - r_orig)
|
||||
print(
|
||||
f" 图案 {index+1:2d} {label:8s}: "
|
||||
f"RGB({r_orig:3d},{g_orig:3d},{b_orig:3d}) → "
|
||||
f"RGB({r_new:3d},{g_new:3d},{b_new:3d}) "
|
||||
f"(差值: {diff:+3d})"
|
||||
)
|
||||
|
||||
def _print_footer(self, total_count):
|
||||
"""打印转换尾部信息"""
|
||||
print(f"✓ 转换完成,共 {total_count} 个图案")
|
||||
print("=" * 80)
|
||||
|
||||
def get_info(self):
|
||||
"""获取转换器信息"""
|
||||
return {
|
||||
"name": "Data Range Converter",
|
||||
"version": "1.0.0",
|
||||
"full_range": "0-255",
|
||||
"limited_range": "16-235",
|
||||
"formula": "16 + (value / 255) × 219",
|
||||
}
|
||||
|
||||
|
||||
# ========== 便捷函数 ==========
|
||||
|
||||
|
||||
def convert_pattern_params(pattern_params, data_range="Full", verbose=True):
|
||||
"""
|
||||
便捷函数:转换图案参数
|
||||
|
||||
Args:
|
||||
pattern_params: 图案参数列表 [[r,g,b], [r,g,b], ...]
|
||||
data_range: "Full" 或 "Limited"
|
||||
verbose: 是否打印日志
|
||||
|
||||
Returns:
|
||||
list: 转换后的图案参数列表
|
||||
|
||||
示例:
|
||||
>>> from utils.data_range_converter import convert_pattern_params
|
||||
>>> params = [[0,0,0], [255,255,255]]
|
||||
>>> converted = convert_pattern_params(params, "Limited")
|
||||
[[16,16,16], [235,235,235]]
|
||||
"""
|
||||
converter = DataRangeConverter(verbose=verbose)
|
||||
return converter.convert(pattern_params, data_range)
|
||||
|
||||
|
||||
def convert_single_rgb(r, g, b, data_range="Full"):
|
||||
"""
|
||||
便捷函数:转换单个 RGB 值
|
||||
|
||||
Args:
|
||||
r, g, b: RGB 值 (0-255)
|
||||
data_range: "Full" 或 "Limited"
|
||||
|
||||
Returns:
|
||||
tuple: 转换后的 RGB 值
|
||||
|
||||
示例:
|
||||
>>> from utils.data_range_converter import convert_single_rgb
|
||||
>>> r, g, b = convert_single_rgb(0, 0, 0, "Limited")
|
||||
(16, 16, 16)
|
||||
"""
|
||||
if data_range == "Full":
|
||||
return (r, g, b)
|
||||
|
||||
converter = DataRangeConverter(verbose=False)
|
||||
return converter.convert_rgb(r, g, b)
|
||||
|
||||
|
||||
# ========== 测试代码 ==========
|
||||
|
||||
if __name__ == "__main__":
|
||||
"""测试转换器"""
|
||||
|
||||
print("=" * 80)
|
||||
print("数据范围转换器 - 测试")
|
||||
print("=" * 80)
|
||||
|
||||
# 测试 1: 基本转换
|
||||
print("\n[测试 1] 基本转换...")
|
||||
converter = DataRangeConverter(verbose=False)
|
||||
|
||||
test_values = [0, 16, 64, 128, 192, 235, 255]
|
||||
print(" Full Range → Limited Range:")
|
||||
for v in test_values:
|
||||
limited = converter.convert_value(v)
|
||||
diff = limited - v
|
||||
print(f" {v:3d} → {limited:3d} (差值: {diff:+3d})")
|
||||
|
||||
# 测试 2: RGB 转换
|
||||
print("\n[测试 2] RGB 转换...")
|
||||
test_rgb = [
|
||||
(0, 0, 0),
|
||||
(128, 128, 128),
|
||||
(255, 255, 255),
|
||||
]
|
||||
|
||||
for r, g, b in test_rgb:
|
||||
r_new, g_new, b_new = converter.convert_rgb(r, g, b)
|
||||
print(f" RGB({r},{g},{b}) → RGB({r_new},{g_new},{b_new})")
|
||||
|
||||
# 测试 3: 完整转换流程
|
||||
print("\n[测试 3] 完整转换流程...")
|
||||
pattern_params = [
|
||||
[255, 255, 255], # 100% 白
|
||||
[230, 230, 230], # 90%
|
||||
[204, 204, 204], # 80%
|
||||
[128, 128, 128], # 50%
|
||||
[0, 0, 0], # 0% 黑
|
||||
]
|
||||
|
||||
converted = converter.convert(pattern_params, "Limited")
|
||||
|
||||
print("\n 对比:")
|
||||
for i, (orig, conv) in enumerate(zip(pattern_params, converted)):
|
||||
print(f" [{i+1}] {orig} → {conv}")
|
||||
|
||||
# 测试 4: 便捷函数
|
||||
print("\n[测试 4] 便捷函数...")
|
||||
result = convert_pattern_params(
|
||||
[[0, 0, 0], [255, 255, 255]], "Limited", verbose=False
|
||||
)
|
||||
print(f" 结果: {result}")
|
||||
|
||||
r, g, b = convert_single_rgb(128, 128, 128, "Limited")
|
||||
print(f" RGB(128,128,128) → RGB({r},{g},{b})")
|
||||
|
||||
# 测试 5: 获取信息
|
||||
print("\n[测试 5] 转换器信息...")
|
||||
info = converter.get_info()
|
||||
for key, value in info.items():
|
||||
print(f" {key}: {value}")
|
||||
|
||||
print("\n" + "=" * 80)
|
||||
print("测试完成")
|
||||
print("=" * 80)
|
||||
Reference in New Issue
Block a user