seg_transforms.py 42 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057
  1. # coding: utf8
  2. # copyright (c) 2020 PaddlePaddle Authors. All Rights Reserve.
  3. #
  4. # Licensed under the Apache License, Version 2.0 (the "License");
  5. # you may not use this file except in compliance with the License.
  6. # You may obtain a copy of the License at
  7. #
  8. # http://www.apache.org/licenses/LICENSE-2.0
  9. #
  10. # Unless required by applicable law or agreed to in writing, software
  11. # distributed under the License is distributed on an "AS IS" BASIS,
  12. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. # See the License for the specific language governing permissions and
  14. # limitations under the License.
  15. from .ops import *
  16. import random
  17. import os.path as osp
  18. import numpy as np
  19. from PIL import Image
  20. import cv2
  21. from collections import OrderedDict
  22. class Compose:
  23. """根据数据预处理/增强算子对输入数据进行操作。
  24. 所有操作的输入图像流形状均是[H, W, C],其中H为图像高,W为图像宽,C为图像通道数。
  25. Args:
  26. transforms (list): 数据预处理/增强算子。
  27. Raises:
  28. TypeError: transforms不是list对象
  29. ValueError: transforms元素个数小于1。
  30. """
  31. def __init__(self, transforms):
  32. if not isinstance(transforms, list):
  33. raise TypeError('The transforms must be a list!')
  34. if len(transforms) < 1:
  35. raise ValueError('The length of transforms ' + \
  36. 'must be equal or larger than 1!')
  37. self.transforms = transforms
  38. self.to_rgb = False
  39. def __call__(self, im, im_info=None, label=None):
  40. """
  41. Args:
  42. im (str/np.ndarray): 图像路径/图像np.ndarray数据。
  43. im_info (list): 存储图像reisze或padding前的shape信息,如
  44. [('resize', [200, 300]), ('padding', [400, 600])]表示
  45. 图像在过resize前shape为(200, 300), 过padding前shape为
  46. (400, 600)
  47. label (str/np.ndarray): 标注图像路径/标注图像np.ndarray数据。
  48. Returns:
  49. tuple: 根据网络所需字段所组成的tuple;字段由transforms中的最后一个数据预处理操作决定。
  50. """
  51. if im_info is None:
  52. im_info = list()
  53. try:
  54. im = cv2.imread(im).astype('float32')
  55. except:
  56. raise ValueError('Can\'t read The image file {}!'.format(im))
  57. if self.to_rgb:
  58. im = cv2.cvtColor(im, cv2.COLOR_BGR2RGB)
  59. if label is not None:
  60. if not isinstance(label, np.ndarray):
  61. label = np.asarray(Image.open(label))
  62. for op in self.transforms:
  63. outputs = op(im, im_info, label)
  64. im = outputs[0]
  65. if len(outputs) >= 2:
  66. im_info = outputs[1]
  67. if len(outputs) == 3:
  68. label = outputs[2]
  69. return outputs
  70. class RandomHorizontalFlip:
  71. """以一定的概率对图像进行水平翻转。当存在标注图像时,则同步进行翻转。
  72. Args:
  73. prob (float): 随机水平翻转的概率。默认值为0.5。
  74. """
  75. def __init__(self, prob=0.5):
  76. self.prob = prob
  77. def __call__(self, im, im_info=None, label=None):
  78. """
  79. Args:
  80. im (np.ndarray): 图像np.ndarray数据。
  81. im_info (list): 存储图像reisze或padding前的shape信息,如
  82. [('resize', [200, 300]), ('padding', [400, 600])]表示
  83. 图像在过resize前shape为(200, 300), 过padding前shape为
  84. (400, 600)
  85. label (np.ndarray): 标注图像np.ndarray数据。
  86. Returns:
  87. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  88. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  89. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  90. """
  91. if random.random() < self.prob:
  92. im = horizontal_flip(im)
  93. if label is not None:
  94. label = horizontal_flip(label)
  95. if label is None:
  96. return (im, im_info)
  97. else:
  98. return (im, im_info, label)
  99. class RandomVerticalFlip:
  100. """以一定的概率对图像进行垂直翻转。当存在标注图像时,则同步进行翻转。
  101. Args:
  102. prob (float): 随机垂直翻转的概率。默认值为0.1。
  103. """
  104. def __init__(self, prob=0.1):
  105. self.prob = prob
  106. def __call__(self, im, im_info=None, label=None):
  107. """
  108. Args:
  109. im (np.ndarray): 图像np.ndarray数据。
  110. im_info (list): 存储图像reisze或padding前的shape信息,如
  111. [('resize', [200, 300]), ('padding', [400, 600])]表示
  112. 图像在过resize前shape为(200, 300), 过padding前shape为
  113. (400, 600)
  114. label (np.ndarray): 标注图像np.ndarray数据。
  115. Returns:
  116. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  117. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  118. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  119. """
  120. if random.random() < self.prob:
  121. im = vertical_flip(im)
  122. if label is not None:
  123. label = vertical_flip(label)
  124. if label is None:
  125. return (im, im_info)
  126. else:
  127. return (im, im_info, label)
  128. class Resize:
  129. """调整图像大小(resize),当存在标注图像时,则同步进行处理。
  130. - 当目标大小(target_size)类型为int时,根据插值方式,
  131. 将图像resize为[target_size, target_size]。
  132. - 当目标大小(target_size)类型为list或tuple时,根据插值方式,
  133. 将图像resize为target_size, target_size的输入应为[w, h]或(w, h)。
  134. Args:
  135. target_size (int|list|tuple): 目标大小。
  136. interp (str): resize的插值方式,与opencv的插值方式对应,
  137. 可选的值为['NEAREST', 'LINEAR', 'CUBIC', 'AREA', 'LANCZOS4'],默认为"LINEAR"。
  138. Raises:
  139. TypeError: target_size不是int/list/tuple。
  140. ValueError: target_size为list/tuple时元素个数不等于2。
  141. AssertionError: interp的取值不在['NEAREST', 'LINEAR', 'CUBIC', 'AREA', 'LANCZOS4']之内。
  142. """
  143. # The interpolation mode
  144. interp_dict = {
  145. 'NEAREST': cv2.INTER_NEAREST,
  146. 'LINEAR': cv2.INTER_LINEAR,
  147. 'CUBIC': cv2.INTER_CUBIC,
  148. 'AREA': cv2.INTER_AREA,
  149. 'LANCZOS4': cv2.INTER_LANCZOS4
  150. }
  151. def __init__(self, target_size, interp='LINEAR'):
  152. self.interp = interp
  153. assert interp in self.interp_dict, "interp should be one of {}".format(
  154. interp_dict.keys())
  155. if isinstance(target_size, list) or isinstance(target_size, tuple):
  156. if len(target_size) != 2:
  157. raise ValueError(
  158. 'when target is list or tuple, it should include 2 elements, but it is {}'
  159. .format(target_size))
  160. elif not isinstance(target_size, int):
  161. raise TypeError(
  162. "Type of target_size is invalid. Must be Integer or List or tuple, now is {}"
  163. .format(type(target_size)))
  164. self.target_size = target_size
  165. def __call__(self, im, im_info=None, label=None):
  166. """
  167. Args:
  168. im (np.ndarray): 图像np.ndarray数据。
  169. im_info (list): 存储图像reisze或padding前的shape信息,如
  170. [('resize', [200, 300]), ('padding', [400, 600])]表示
  171. 图像在过resize前shape为(200, 300), 过padding前shape为
  172. (400, 600)
  173. label (np.ndarray): 标注图像np.ndarray数据。
  174. Returns:
  175. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  176. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  177. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  178. 其中,im_info跟新字段为:
  179. -shape_before_resize (tuple): 保存resize之前图像的形状(h, w)。
  180. Raises:
  181. ZeroDivisionError: im的短边为0。
  182. TypeError: im不是np.ndarray数据。
  183. ValueError: im不是3维nd.ndarray。
  184. """
  185. if im_info is None:
  186. im_info = OrderedDict()
  187. im_info.append(('resize', im.shape[:2]))
  188. if not isinstance(im, np.ndarray):
  189. raise TypeError("ResizeImage: image type is not np.ndarray.")
  190. if len(im.shape) != 3:
  191. raise ValueError('ResizeImage: image is not 3-dimensional.')
  192. im_shape = im.shape
  193. im_size_min = np.min(im_shape[0:2])
  194. im_size_max = np.max(im_shape[0:2])
  195. if float(im_size_min) == 0:
  196. raise ZeroDivisionError('ResizeImage: min size of image is 0')
  197. if isinstance(self.target_size, int):
  198. resize_w = self.target_size
  199. resize_h = self.target_size
  200. else:
  201. resize_w = self.target_size[0]
  202. resize_h = self.target_size[1]
  203. im_scale_x = float(resize_w) / float(im_shape[1])
  204. im_scale_y = float(resize_h) / float(im_shape[0])
  205. im = cv2.resize(
  206. im,
  207. None,
  208. None,
  209. fx=im_scale_x,
  210. fy=im_scale_y,
  211. interpolation=self.interp_dict[self.interp])
  212. if label is not None:
  213. label = cv2.resize(
  214. label,
  215. None,
  216. None,
  217. fx=im_scale_x,
  218. fy=im_scale_y,
  219. interpolation=self.interp_dict['NEAREST'])
  220. if label is None:
  221. return (im, im_info)
  222. else:
  223. return (im, im_info, label)
  224. class ResizeByLong:
  225. """对图像长边resize到固定值,短边按比例进行缩放。当存在标注图像时,则同步进行处理。
  226. Args:
  227. long_size (int): resize后图像的长边大小。
  228. """
  229. def __init__(self, long_size):
  230. self.long_size = long_size
  231. def __call__(self, im, im_info=None, label=None):
  232. """
  233. Args:
  234. im (np.ndarray): 图像np.ndarray数据。
  235. im_info (list): 存储图像reisze或padding前的shape信息,如
  236. [('resize', [200, 300]), ('padding', [400, 600])]表示
  237. 图像在过resize前shape为(200, 300), 过padding前shape为
  238. (400, 600)
  239. label (np.ndarray): 标注图像np.ndarray数据。
  240. Returns:
  241. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  242. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  243. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  244. 其中,im_info新增字段为:
  245. -shape_before_resize (tuple): 保存resize之前图像的形状(h, w)。
  246. """
  247. if im_info is None:
  248. im_info = OrderedDict()
  249. im_info.append(('resize', im.shape[:2]))
  250. im = resize_long(im, self.long_size)
  251. if label is not None:
  252. label = resize_long(label, self.long_size, cv2.INTER_NEAREST)
  253. if label is None:
  254. return (im, im_info)
  255. else:
  256. return (im, im_info, label)
  257. class ResizeByShort:
  258. """根据图像的短边调整图像大小(resize)。
  259. 1. 获取图像的长边和短边长度。
  260. 2. 根据短边与short_size的比例,计算长边的目标长度,
  261. 此时高、宽的resize比例为short_size/原图短边长度。
  262. 3. 如果max_size>0,调整resize比例:
  263. 如果长边的目标长度>max_size,则高、宽的resize比例为max_size/原图长边长度。
  264. 4. 根据调整大小的比例对图像进行resize。
  265. Args:
  266. target_size (int): 短边目标长度。默认为800。
  267. max_size (int): 长边目标长度的最大限制。默认为1333。
  268. Raises:
  269. TypeError: 形参数据类型不满足需求。
  270. """
  271. def __init__(self, short_size=800, max_size=1333):
  272. self.max_size = int(max_size)
  273. if not isinstance(short_size, int):
  274. raise TypeError(
  275. "Type of short_size is invalid. Must be Integer, now is {}".
  276. format(type(short_size)))
  277. self.short_size = short_size
  278. if not (isinstance(self.max_size, int)):
  279. raise TypeError("max_size: input type is invalid.")
  280. def __call__(self, im, im_info=None, label=None):
  281. """
  282. Args:
  283. im (numnp.ndarraypy): 图像np.ndarray数据。
  284. im_info (dict, 可选): 存储与图像相关的信息。
  285. label (np.ndarray): 标注图像np.ndarray数据。
  286. Returns:
  287. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  288. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  289. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  290. 其中,im_info更新字段为:
  291. -shape_before_resize (tuple): 保存resize之前图像的形状(h, w)。
  292. Raises:
  293. TypeError: 形参数据类型不满足需求。
  294. ValueError: 数据长度不匹配。
  295. """
  296. if im_info is None:
  297. im_info = OrderedDict()
  298. if not isinstance(im, np.ndarray):
  299. raise TypeError("ResizeByShort: image type is not numpy.")
  300. if len(im.shape) != 3:
  301. raise ValueError('ResizeByShort: image is not 3-dimensional.')
  302. im_info.append(('resize', im.shape[:2]))
  303. im_short_size = min(im.shape[0], im.shape[1])
  304. im_long_size = max(im.shape[0], im.shape[1])
  305. scale = float(self.short_size) / im_short_size
  306. if self.max_size > 0 and np.round(
  307. scale * im_long_size) > self.max_size:
  308. scale = float(self.max_size) / float(im_long_size)
  309. resized_width = int(round(im.shape[1] * scale))
  310. resized_height = int(round(im.shape[0] * scale))
  311. im = cv2.resize(
  312. im, (resized_width, resized_height),
  313. interpolation=cv2.INTER_NEAREST)
  314. if label is not None:
  315. im = cv2.resize(
  316. label, (resized_width, resized_height),
  317. interpolation=cv2.INTER_NEAREST)
  318. if label is None:
  319. return (im, im_info)
  320. else:
  321. return (im, im_info, label)
  322. class ResizeRangeScaling:
  323. """对图像长边随机resize到指定范围内,短边按比例进行缩放。当存在标注图像时,则同步进行处理。
  324. Args:
  325. min_value (int): 图像长边resize后的最小值。默认值400。
  326. max_value (int): 图像长边resize后的最大值。默认值600。
  327. Raises:
  328. ValueError: min_value大于max_value
  329. """
  330. def __init__(self, min_value=400, max_value=600):
  331. if min_value > max_value:
  332. raise ValueError('min_value must be less than max_value, '
  333. 'but they are {} and {}.'.format(
  334. min_value, max_value))
  335. self.min_value = min_value
  336. self.max_value = max_value
  337. def __call__(self, im, im_info=None, label=None):
  338. """
  339. Args:
  340. im (np.ndarray): 图像np.ndarray数据。
  341. im_info (list): 存储图像reisze或padding前的shape信息,如
  342. [('resize', [200, 300]), ('padding', [400, 600])]表示
  343. 图像在过resize前shape为(200, 300), 过padding前shape为
  344. (400, 600)
  345. label (np.ndarray): 标注图像np.ndarray数据。
  346. Returns:
  347. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  348. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  349. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  350. """
  351. if self.min_value == self.max_value:
  352. random_size = self.max_value
  353. else:
  354. random_size = int(
  355. np.random.uniform(self.min_value, self.max_value) + 0.5)
  356. im = resize_long(im, random_size, cv2.INTER_LINEAR)
  357. if label is not None:
  358. label = resize_long(label, random_size, cv2.INTER_NEAREST)
  359. if label is None:
  360. return (im, im_info)
  361. else:
  362. return (im, im_info, label)
  363. class ResizeStepScaling:
  364. """对图像按照某一个比例resize,这个比例以scale_step_size为步长
  365. 在[min_scale_factor, max_scale_factor]随机变动。当存在标注图像时,则同步进行处理。
  366. Args:
  367. min_scale_factor(float), resize最小尺度。默认值0.75。
  368. max_scale_factor (float), resize最大尺度。默认值1.25。
  369. scale_step_size (float), resize尺度范围间隔。默认值0.25。
  370. Raises:
  371. ValueError: min_scale_factor大于max_scale_factor
  372. """
  373. def __init__(self,
  374. min_scale_factor=0.75,
  375. max_scale_factor=1.25,
  376. scale_step_size=0.25):
  377. if min_scale_factor > max_scale_factor:
  378. raise ValueError(
  379. 'min_scale_factor must be less than max_scale_factor, '
  380. 'but they are {} and {}.'.format(min_scale_factor,
  381. max_scale_factor))
  382. self.min_scale_factor = min_scale_factor
  383. self.max_scale_factor = max_scale_factor
  384. self.scale_step_size = scale_step_size
  385. def __call__(self, im, im_info=None, label=None):
  386. """
  387. Args:
  388. im (np.ndarray): 图像np.ndarray数据。
  389. im_info (list): 存储图像reisze或padding前的shape信息,如
  390. [('resize', [200, 300]), ('padding', [400, 600])]表示
  391. 图像在过resize前shape为(200, 300), 过padding前shape为
  392. (400, 600)
  393. label (np.ndarray): 标注图像np.ndarray数据。
  394. Returns:
  395. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  396. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  397. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  398. """
  399. if self.min_scale_factor == self.max_scale_factor:
  400. scale_factor = self.min_scale_factor
  401. elif self.scale_step_size == 0:
  402. scale_factor = np.random.uniform(self.min_scale_factor,
  403. self.max_scale_factor)
  404. else:
  405. num_steps = int((self.max_scale_factor - self.min_scale_factor) /
  406. self.scale_step_size + 1)
  407. scale_factors = np.linspace(self.min_scale_factor,
  408. self.max_scale_factor,
  409. num_steps).tolist()
  410. np.random.shuffle(scale_factors)
  411. scale_factor = scale_factors[0]
  412. im = cv2.resize(
  413. im, (0, 0),
  414. fx=scale_factor,
  415. fy=scale_factor,
  416. interpolation=cv2.INTER_LINEAR)
  417. if label is not None:
  418. label = cv2.resize(
  419. label, (0, 0),
  420. fx=scale_factor,
  421. fy=scale_factor,
  422. interpolation=cv2.INTER_NEAREST)
  423. if label is None:
  424. return (im, im_info)
  425. else:
  426. return (im, im_info, label)
  427. class Normalize:
  428. """对图像进行标准化。
  429. 1.尺度缩放到 [0,1]。
  430. 2.对图像进行减均值除以标准差操作。
  431. Args:
  432. mean (list): 图像数据集的均值。默认值[0.5, 0.5, 0.5]。
  433. std (list): 图像数据集的标准差。默认值[0.5, 0.5, 0.5]。
  434. Raises:
  435. ValueError: mean或std不是list对象。std包含0。
  436. """
  437. def __init__(self, mean=[0.5, 0.5, 0.5], std=[0.5, 0.5, 0.5]):
  438. self.mean = mean
  439. self.std = std
  440. if not (isinstance(self.mean, list) and isinstance(self.std, list)):
  441. raise ValueError("{}: input type is invalid.".format(self))
  442. from functools import reduce
  443. if reduce(lambda x, y: x * y, self.std) == 0:
  444. raise ValueError('{}: std is invalid!'.format(self))
  445. def __call__(self, im, im_info=None, label=None):
  446. """
  447. Args:
  448. im (np.ndarray): 图像np.ndarray数据。
  449. im_info (list): 存储图像reisze或padding前的shape信息,如
  450. [('resize', [200, 300]), ('padding', [400, 600])]表示
  451. 图像在过resize前shape为(200, 300), 过padding前shape为
  452. (400, 600)
  453. label (np.ndarray): 标注图像np.ndarray数据。
  454. Returns:
  455. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  456. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  457. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  458. """
  459. mean = np.array(self.mean)[np.newaxis, np.newaxis, :]
  460. std = np.array(self.std)[np.newaxis, np.newaxis, :]
  461. im = normalize(im, mean, std)
  462. if label is None:
  463. return (im, im_info)
  464. else:
  465. return (im, im_info, label)
  466. class Padding:
  467. """对图像或标注图像进行padding,padding方向为右和下。
  468. 根据提供的值对图像或标注图像进行padding操作。
  469. Args:
  470. target_size (int|list|tuple): padding后图像的大小。
  471. im_padding_value (list): 图像padding的值。默认为[127.5, 127.5, 127.5]。
  472. label_padding_value (int): 标注图像padding的值。默认值为255。
  473. Raises:
  474. TypeError: target_size不是int|list|tuple。
  475. ValueError: target_size为list|tuple时元素个数不等于2。
  476. """
  477. def __init__(self,
  478. target_size,
  479. im_padding_value=[127.5, 127.5, 127.5],
  480. label_padding_value=255):
  481. if isinstance(target_size, list) or isinstance(target_size, tuple):
  482. if len(target_size) != 2:
  483. raise ValueError(
  484. 'when target is list or tuple, it should include 2 elements, but it is {}'
  485. .format(target_size))
  486. elif not isinstance(target_size, int):
  487. raise TypeError(
  488. "Type of target_size is invalid. Must be Integer or List or tuple, now is {}"
  489. .format(type(target_size)))
  490. self.target_size = target_size
  491. self.im_padding_value = im_padding_value
  492. self.label_padding_value = label_padding_value
  493. def __call__(self, im, im_info=None, label=None):
  494. """
  495. Args:
  496. im (np.ndarray): 图像np.ndarray数据。
  497. im_info (list): 存储图像reisze或padding前的shape信息,如
  498. [('resize', [200, 300]), ('padding', [400, 600])]表示
  499. 图像在过resize前shape为(200, 300), 过padding前shape为
  500. (400, 600)
  501. label (np.ndarray): 标注图像np.ndarray数据。
  502. Returns:
  503. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  504. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  505. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  506. 其中,im_info新增字段为:
  507. -shape_before_padding (tuple): 保存padding之前图像的形状(h, w)。
  508. Raises:
  509. ValueError: 输入图像im或label的形状大于目标值
  510. """
  511. if im_info is None:
  512. im_info = OrderedDict()
  513. im_info.append(('padding', im.shape[:2]))
  514. im_height, im_width = im.shape[0], im.shape[1]
  515. if isinstance(self.target_size, int):
  516. target_height = self.target_size
  517. target_width = self.target_size
  518. else:
  519. target_height = self.target_size[1]
  520. target_width = self.target_size[0]
  521. pad_height = target_height - im_height
  522. pad_width = target_width - im_width
  523. if pad_height < 0 or pad_width < 0:
  524. raise ValueError(
  525. 'the size of image should be less than target_size, but the size of image ({}, {}), is larger than target_size ({}, {})'
  526. .format(im_width, im_height, target_width, target_height))
  527. else:
  528. im = cv2.copyMakeBorder(
  529. im,
  530. 0,
  531. pad_height,
  532. 0,
  533. pad_width,
  534. cv2.BORDER_CONSTANT,
  535. value=self.im_padding_value)
  536. if label is not None:
  537. label = cv2.copyMakeBorder(
  538. label,
  539. 0,
  540. pad_height,
  541. 0,
  542. pad_width,
  543. cv2.BORDER_CONSTANT,
  544. value=self.label_padding_value)
  545. if label is None:
  546. return (im, im_info)
  547. else:
  548. return (im, im_info, label)
  549. class RandomPaddingCrop:
  550. """对图像和标注图进行随机裁剪,当所需要的裁剪尺寸大于原图时,则进行padding操作。
  551. Args:
  552. crop_size (int|list|tuple): 裁剪图像大小。默认为512。
  553. im_padding_value (list): 图像padding的值。默认为[127.5, 127.5, 127.5]。
  554. label_padding_value (int): 标注图像padding的值。默认值为255。
  555. Raises:
  556. TypeError: crop_size不是int/list/tuple。
  557. ValueError: target_size为list/tuple时元素个数不等于2。
  558. """
  559. def __init__(self,
  560. crop_size=512,
  561. im_padding_value=[127.5, 127.5, 127.5],
  562. label_padding_value=255):
  563. if isinstance(crop_size, list) or isinstance(crop_size, tuple):
  564. if len(crop_size) != 2:
  565. raise ValueError(
  566. 'when crop_size is list or tuple, it should include 2 elements, but it is {}'
  567. .format(crop_size))
  568. elif not isinstance(crop_size, int):
  569. raise TypeError(
  570. "Type of crop_size is invalid. Must be Integer or List or tuple, now is {}"
  571. .format(type(crop_size)))
  572. self.crop_size = crop_size
  573. self.im_padding_value = im_padding_value
  574. self.label_padding_value = label_padding_value
  575. def __call__(self, im, im_info=None, label=None):
  576. """
  577. Args:
  578. im (np.ndarray): 图像np.ndarray数据。
  579. im_info (list): 存储图像reisze或padding前的shape信息,如
  580. [('resize', [200, 300]), ('padding', [400, 600])]表示
  581. 图像在过resize前shape为(200, 300), 过padding前shape为
  582. (400, 600)
  583. label (np.ndarray): 标注图像np.ndarray数据。
  584. Returns:
  585. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  586. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  587. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  588. """
  589. if isinstance(self.crop_size, int):
  590. crop_width = self.crop_size
  591. crop_height = self.crop_size
  592. else:
  593. crop_width = self.crop_size[0]
  594. crop_height = self.crop_size[1]
  595. img_height = im.shape[0]
  596. img_width = im.shape[1]
  597. if img_height == crop_height and img_width == crop_width:
  598. if label is None:
  599. return (im, im_info)
  600. else:
  601. return (im, im_info, label)
  602. else:
  603. pad_height = max(crop_height - img_height, 0)
  604. pad_width = max(crop_width - img_width, 0)
  605. if (pad_height > 0 or pad_width > 0):
  606. im = cv2.copyMakeBorder(
  607. im,
  608. 0,
  609. pad_height,
  610. 0,
  611. pad_width,
  612. cv2.BORDER_CONSTANT,
  613. value=self.im_padding_value)
  614. if label is not None:
  615. label = cv2.copyMakeBorder(
  616. label,
  617. 0,
  618. pad_height,
  619. 0,
  620. pad_width,
  621. cv2.BORDER_CONSTANT,
  622. value=self.label_padding_value)
  623. img_height = im.shape[0]
  624. img_width = im.shape[1]
  625. if crop_height > 0 and crop_width > 0:
  626. h_off = np.random.randint(img_height - crop_height + 1)
  627. w_off = np.random.randint(img_width - crop_width + 1)
  628. im = im[h_off:(crop_height + h_off), w_off:(
  629. w_off + crop_width), :]
  630. if label is not None:
  631. label = label[h_off:(crop_height + h_off), w_off:(
  632. w_off + crop_width)]
  633. if label is None:
  634. return (im, im_info)
  635. else:
  636. return (im, im_info, label)
  637. class RandomBlur:
  638. """以一定的概率对图像进行高斯模糊。
  639. Args:
  640. prob (float): 图像模糊概率。默认为0.1。
  641. """
  642. def __init__(self, prob=0.1):
  643. self.prob = prob
  644. def __call__(self, im, im_info=None, label=None):
  645. """
  646. Args:
  647. im (np.ndarray): 图像np.ndarray数据。
  648. im_info (list): 存储图像reisze或padding前的shape信息,如
  649. [('resize', [200, 300]), ('padding', [400, 600])]表示
  650. 图像在过resize前shape为(200, 300), 过padding前shape为
  651. (400, 600)
  652. label (np.ndarray): 标注图像np.ndarray数据。
  653. Returns:
  654. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  655. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  656. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  657. """
  658. if self.prob <= 0:
  659. n = 0
  660. elif self.prob >= 1:
  661. n = 1
  662. else:
  663. n = int(1.0 / self.prob)
  664. if n > 0:
  665. if np.random.randint(0, n) == 0:
  666. radius = np.random.randint(3, 10)
  667. if radius % 2 != 1:
  668. radius = radius + 1
  669. if radius > 9:
  670. radius = 9
  671. im = cv2.GaussianBlur(im, (radius, radius), 0, 0)
  672. if label is None:
  673. return (im, im_info)
  674. else:
  675. return (im, im_info, label)
  676. class RandomRotate:
  677. """对图像进行随机旋转, 模型训练时的数据增强操作。
  678. 在旋转区间[-rotate_range, rotate_range]内,对图像进行随机旋转,当存在标注图像时,同步进行,
  679. 并对旋转后的图像和标注图像进行相应的padding。
  680. Args:
  681. rotate_range (float): 最大旋转角度。默认为15度。
  682. im_padding_value (list): 图像padding的值。默认为[127.5, 127.5, 127.5]。
  683. label_padding_value (int): 标注图像padding的值。默认为255。
  684. """
  685. def __init__(self,
  686. rotate_range=15,
  687. im_padding_value=[127.5, 127.5, 127.5],
  688. label_padding_value=255):
  689. self.rotate_range = rotate_range
  690. self.im_padding_value = im_padding_value
  691. self.label_padding_value = label_padding_value
  692. def __call__(self, im, im_info=None, label=None):
  693. """
  694. Args:
  695. im (np.ndarray): 图像np.ndarray数据。
  696. im_info (list): 存储图像reisze或padding前的shape信息,如
  697. [('resize', [200, 300]), ('padding', [400, 600])]表示
  698. 图像在过resize前shape为(200, 300), 过padding前shape为
  699. (400, 600)
  700. label (np.ndarray): 标注图像np.ndarray数据。
  701. Returns:
  702. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  703. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  704. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  705. """
  706. if self.rotate_range > 0:
  707. (h, w) = im.shape[:2]
  708. do_rotation = np.random.uniform(-self.rotate_range,
  709. self.rotate_range)
  710. pc = (w // 2, h // 2)
  711. r = cv2.getRotationMatrix2D(pc, do_rotation, 1.0)
  712. cos = np.abs(r[0, 0])
  713. sin = np.abs(r[0, 1])
  714. nw = int((h * sin) + (w * cos))
  715. nh = int((h * cos) + (w * sin))
  716. (cx, cy) = pc
  717. r[0, 2] += (nw / 2) - cx
  718. r[1, 2] += (nh / 2) - cy
  719. dsize = (nw, nh)
  720. im = cv2.warpAffine(
  721. im,
  722. r,
  723. dsize=dsize,
  724. flags=cv2.INTER_LINEAR,
  725. borderMode=cv2.BORDER_CONSTANT,
  726. borderValue=self.im_padding_value)
  727. label = cv2.warpAffine(
  728. label,
  729. r,
  730. dsize=dsize,
  731. flags=cv2.INTER_NEAREST,
  732. borderMode=cv2.BORDER_CONSTANT,
  733. borderValue=self.label_padding_value)
  734. if label is None:
  735. return (im, im_info)
  736. else:
  737. return (im, im_info, label)
  738. class RandomScaleAspect:
  739. """裁剪并resize回原始尺寸的图像和标注图像。
  740. 按照一定的面积比和宽高比对图像进行裁剪,并reszie回原始图像的图像,当存在标注图时,同步进行。
  741. Args:
  742. min_scale (float):裁取图像占原始图像的面积比,取值[0,1],为0时则返回原图。默认为0.5。
  743. aspect_ratio (float): 裁取图像的宽高比范围,非负值,为0时返回原图。默认为0.33。
  744. """
  745. def __init__(self, min_scale=0.5, aspect_ratio=0.33):
  746. self.min_scale = min_scale
  747. self.aspect_ratio = aspect_ratio
  748. def __call__(self, im, im_info=None, label=None):
  749. """
  750. Args:
  751. im (np.ndarray): 图像np.ndarray数据。
  752. im_info (list): 存储图像reisze或padding前的shape信息,如
  753. [('resize', [200, 300]), ('padding', [400, 600])]表示
  754. 图像在过resize前shape为(200, 300), 过padding前shape为
  755. (400, 600)
  756. label (np.ndarray): 标注图像np.ndarray数据。
  757. Returns:
  758. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  759. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  760. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  761. """
  762. if self.min_scale != 0 and self.aspect_ratio != 0:
  763. img_height = im.shape[0]
  764. img_width = im.shape[1]
  765. for i in range(0, 10):
  766. area = img_height * img_width
  767. target_area = area * np.random.uniform(self.min_scale, 1.0)
  768. aspectRatio = np.random.uniform(self.aspect_ratio,
  769. 1.0 / self.aspect_ratio)
  770. dw = int(np.sqrt(target_area * 1.0 * aspectRatio))
  771. dh = int(np.sqrt(target_area * 1.0 / aspectRatio))
  772. if (np.random.randint(10) < 5):
  773. tmp = dw
  774. dw = dh
  775. dh = tmp
  776. if (dh < img_height and dw < img_width):
  777. h1 = np.random.randint(0, img_height - dh)
  778. w1 = np.random.randint(0, img_width - dw)
  779. im = im[h1:(h1 + dh), w1:(w1 + dw), :]
  780. label = label[h1:(h1 + dh), w1:(w1 + dw)]
  781. im = cv2.resize(
  782. im, (img_width, img_height),
  783. interpolation=cv2.INTER_LINEAR)
  784. label = cv2.resize(
  785. label, (img_width, img_height),
  786. interpolation=cv2.INTER_NEAREST)
  787. break
  788. if label is None:
  789. return (im, im_info)
  790. else:
  791. return (im, im_info, label)
  792. class RandomDistort:
  793. """对图像进行随机失真。
  794. 1. 对变换的操作顺序进行随机化操作。
  795. 2. 按照1中的顺序以一定的概率对图像进行随机像素内容变换。
  796. Args:
  797. brightness_range (float): 明亮度因子的范围。默认为0.5。
  798. brightness_prob (float): 随机调整明亮度的概率。默认为0.5。
  799. contrast_range (float): 对比度因子的范围。默认为0.5。
  800. contrast_prob (float): 随机调整对比度的概率。默认为0.5。
  801. saturation_range (float): 饱和度因子的范围。默认为0.5。
  802. saturation_prob (float): 随机调整饱和度的概率。默认为0.5。
  803. hue_range (int): 色调因子的范围。默认为18。
  804. hue_prob (float): 随机调整色调的概率。默认为0.5。
  805. """
  806. def __init__(self,
  807. brightness_range=0.5,
  808. brightness_prob=0.5,
  809. contrast_range=0.5,
  810. contrast_prob=0.5,
  811. saturation_range=0.5,
  812. saturation_prob=0.5,
  813. hue_range=18,
  814. hue_prob=0.5):
  815. self.brightness_range = brightness_range
  816. self.brightness_prob = brightness_prob
  817. self.contrast_range = contrast_range
  818. self.contrast_prob = contrast_prob
  819. self.saturation_range = saturation_range
  820. self.saturation_prob = saturation_prob
  821. self.hue_range = hue_range
  822. self.hue_prob = hue_prob
  823. def __call__(self, im, im_info=None, label=None):
  824. """
  825. Args:
  826. im (np.ndarray): 图像np.ndarray数据。
  827. im_info (list): 存储图像reisze或padding前的shape信息,如
  828. [('resize', [200, 300]), ('padding', [400, 600])]表示
  829. 图像在过resize前shape为(200, 300), 过padding前shape为
  830. (400, 600)
  831. label (np.ndarray): 标注图像np.ndarray数据。
  832. Returns:
  833. tuple: 当label为空时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  834. 当label不为空时,返回的tuple为(im, im_info, label),分别对应图像np.ndarray数据、
  835. 存储与图像相关信息的字典和标注图像np.ndarray数据。
  836. """
  837. brightness_lower = 1 - self.brightness_range
  838. brightness_upper = 1 + self.brightness_range
  839. contrast_lower = 1 - self.contrast_range
  840. contrast_upper = 1 + self.contrast_range
  841. saturation_lower = 1 - self.saturation_range
  842. saturation_upper = 1 + self.saturation_range
  843. hue_lower = -self.hue_range
  844. hue_upper = self.hue_range
  845. ops = [brightness, contrast, saturation, hue]
  846. random.shuffle(ops)
  847. params_dict = {
  848. 'brightness': {
  849. 'brightness_lower': brightness_lower,
  850. 'brightness_upper': brightness_upper
  851. },
  852. 'contrast': {
  853. 'contrast_lower': contrast_lower,
  854. 'contrast_upper': contrast_upper
  855. },
  856. 'saturation': {
  857. 'saturation_lower': saturation_lower,
  858. 'saturation_upper': saturation_upper
  859. },
  860. 'hue': {
  861. 'hue_lower': hue_lower,
  862. 'hue_upper': hue_upper
  863. }
  864. }
  865. prob_dict = {
  866. 'brightness': self.brightness_prob,
  867. 'contrast': self.contrast_prob,
  868. 'saturation': self.saturation_prob,
  869. 'hue': self.hue_prob
  870. }
  871. for id in range(4):
  872. params = params_dict[ops[id].__name__]
  873. prob = prob_dict[ops[id].__name__]
  874. params['im'] = im
  875. if np.random.uniform(0, 1) < prob:
  876. im = ops[id](**params)
  877. if label is None:
  878. return (im, im_info)
  879. else:
  880. return (im, im_info, label)
  881. class ArrangeSegmenter:
  882. """获取训练/验证/预测所需的信息。
  883. Args:
  884. mode (str): 指定数据用于何种用途,取值范围为['train', 'eval', 'test', 'quant']。
  885. Raises:
  886. ValueError: mode的取值不在['train', 'eval', 'test', 'quant']之内
  887. """
  888. def __init__(self, mode):
  889. if mode not in ['train', 'eval', 'test', 'quant']:
  890. raise ValueError(
  891. "mode should be defined as one of ['train', 'eval', 'test', 'quant']!"
  892. )
  893. self.mode = mode
  894. def __call__(self, im, im_info, label=None):
  895. """
  896. Args:
  897. im (np.ndarray): 图像np.ndarray数据。
  898. im_info (list): 存储图像reisze或padding前的shape信息,如
  899. [('resize', [200, 300]), ('padding', [400, 600])]表示
  900. 图像在过resize前shape为(200, 300), 过padding前shape为
  901. (400, 600)
  902. label (np.ndarray): 标注图像np.ndarray数据。
  903. Returns:
  904. tuple: 当mode为'train'或'eval'时,返回的tuple为(im, label),分别对应图像np.ndarray数据、存储与图像相关信息的字典;
  905. 当mode为'test'时,返回的tuple为(im, im_info),分别对应图像np.ndarray数据、存储与图像相关信息的字典;当mode为
  906. 'quant'时,返回的tuple为(im,),为图像np.ndarray数据。
  907. """
  908. im = permute(im, False)
  909. if self.mode == 'train' or self.mode == 'eval':
  910. label = label[np.newaxis, :, :]
  911. return (im, label)
  912. elif self.mode == 'test':
  913. return (im, im_info)
  914. else:
  915. return (im, )