/WPFImage/AForge.Imaging/Filters/FilterIterator.cs

# · C# · 299 lines · 107 code · 24 blank · 168 comment · 7 complexity · bec4eb19c1b5494a7dac4fc9e4a96d54 MD5 · raw file

  1. // AForge Image Processing Library
  2. // AForge.NET framework
  3. // http://www.aforgenet.com/framework/
  4. //
  5. // Copyright Š Andrew Kirillov, 2005-2009
  6. // andrew.kirillov@aforgenet.com
  7. //
  8. namespace AForge.Imaging.Filters
  9. {
  10. using System;
  11. using System.Collections.Generic;
  12. using System.Drawing;
  13. using System.Drawing.Imaging;
  14. /// <summary>
  15. /// Filter iterator.
  16. /// </summary>
  17. ///
  18. /// <remarks><para>Filter iterator performs specified amount of filter's iterations.
  19. /// The filter take the specified <see cref="BaseFilter">base filter</see> and applies it
  20. /// to source image <see cref="Iterations">specified amount of times</see>.</para>
  21. ///
  22. /// <para><note>The filter itself does not have any restrictions to pixel format of source
  23. /// image. This is set by <see cref="BaseFilter">base filter</see>.</note></para>
  24. ///
  25. /// <para><note>The filter does image processing using only <see cref="IFilter"/>
  26. /// interface of the specified <see cref="BaseFilter">base filter</see>. This means
  27. /// that this filter may not utilize all potential features of the base filter, like
  28. /// in-place processing (see <see cref="IInPlaceFilter"/>) and region based processing
  29. /// (see <see cref="IInPlacePartialFilter"/>). To utilize those features, it is required to
  30. /// do filter's iteration manually.</note></para>
  31. ///
  32. /// <para>Sample usage (morphological thinning):</para>
  33. /// <code>
  34. /// // create filter sequence
  35. /// FiltersSequence filterSequence = new FiltersSequence( );
  36. /// // add 8 thinning filters with different structuring elements
  37. /// filterSequence.Add( new HitAndMiss(
  38. /// new short [,] { { 0, 0, 0 }, { -1, 1, -1 }, { 1, 1, 1 } },
  39. /// HitAndMiss.Modes.Thinning ) );
  40. /// filterSequence.Add( new HitAndMiss(
  41. /// new short [,] { { -1, 0, 0 }, { 1, 1, 0 }, { -1, 1, -1 } },
  42. /// HitAndMiss.Modes.Thinning ) );
  43. /// filterSequence.Add( new HitAndMiss(
  44. /// new short [,] { { 1, -1, 0 }, { 1, 1, 0 }, { 1, -1, 0 } },
  45. /// HitAndMiss.Modes.Thinning ) );
  46. /// filterSequence.Add( new HitAndMiss(
  47. /// new short [,] { { -1, 1, -1 }, { 1, 1, 0 }, { -1, 0, 0 } },
  48. /// HitAndMiss.Modes.Thinning ) );
  49. /// filterSequence.Add( new HitAndMiss(
  50. /// new short [,] { { 1, 1, 1 }, { -1, 1, -1 }, { 0, 0, 0 } },
  51. /// HitAndMiss.Modes.Thinning ) );
  52. /// filterSequence.Add( new HitAndMiss(
  53. /// new short [,] { { -1, 1, -1 }, { 0, 1, 1 }, { 0, 0, -1 } },
  54. /// HitAndMiss.Modes.Thinning ) );
  55. /// filterSequence.Add( new HitAndMiss(
  56. /// new short [,] { { 0, -1, 1 }, { 0, 1, 1 }, { 0, -1, 1 } },
  57. /// HitAndMiss.Modes.Thinning ) );
  58. /// filterSequence.Add( new HitAndMiss(
  59. /// new short [,] { { 0, 0, -1 }, { 0, 1, 1 }, { -1, 1, -1 } },
  60. /// HitAndMiss.Modes.Thinning ) );
  61. /// // create filter iterator for 10 iterations
  62. /// FilterIterator filter = new FilterIterator( filterSequence, 10 );
  63. /// // apply the filter
  64. /// Bitmap newImage = filter.Apply( image );
  65. /// </code>
  66. ///
  67. /// <para><b>Initial image:</b></para>
  68. /// <img src="img/imaging/sample14.png" width="150" height="150" />
  69. /// <para><b>Result image:</b></para>
  70. /// <img src="img/imaging/thinning.png" width="150" height="150" />
  71. /// </remarks>
  72. ///
  73. [Serializable] public class FilterIterator : IFilter, IFilterInformation
  74. {
  75. private IFilter baseFilter;
  76. private int iterations = 1;
  77. public List<System.Drawing.Imaging.PixelFormat> GetValidPixelFormats()
  78. {
  79. var formats = FormatTransalations;
  80. List<System.Drawing.Imaging.PixelFormat> results = new List<PixelFormat>();
  81. if (formats != null)
  82. {
  83. foreach (var kvp in formats)
  84. {
  85. results.Add(kvp.Key);
  86. }
  87. return results;
  88. }
  89. results.Add(PixelFormat.Format24bppRgb);
  90. return results;
  91. }
  92. public bool IsValidPixelFormat(PixelFormat pixelFormat)
  93. {
  94. return FormatTransalations.ContainsKey(pixelFormat);
  95. }
  96. /// <summary>
  97. /// Format translations dictionary.
  98. /// </summary>
  99. ///
  100. /// <remarks><para>See <see cref="IFilterInformation.FormatTransalations"/>
  101. /// documentation for additional information.</para>
  102. ///
  103. /// <para><note>The filter provides format translation dictionary taken from
  104. /// <see cref="BaseFilter"/> filter.</note></para>
  105. /// </remarks>
  106. ///
  107. public Dictionary<PixelFormat, PixelFormat> FormatTransalations
  108. {
  109. get { return ( (IFilterInformation) baseFilter).FormatTransalations; }
  110. }
  111. /// <summary>
  112. /// Base filter.
  113. /// </summary>
  114. ///
  115. /// <remarks><para>The base filter is the filter to be applied specified amount of iterations to
  116. /// a specified image.</para></remarks>
  117. ///
  118. public IFilter BaseFilter
  119. {
  120. get { return baseFilter; }
  121. set { baseFilter = value; }
  122. }
  123. /// <summary>
  124. /// Iterations amount, [1, 255].
  125. /// </summary>
  126. ///
  127. /// <remarks><para>The amount of times to apply specified filter to a specified image.</para>
  128. ///
  129. /// <para>Default value is set to <b>1</b>.</para>
  130. /// </remarks>
  131. ///
  132. public int Iterations
  133. {
  134. get { return iterations; }
  135. set { iterations = Math.Max( 1, Math.Min( 255, value ) ); }
  136. }
  137. /// <summary>
  138. /// Initializes a new instance of the <see cref="FilterIterator"/> class.
  139. /// </summary>
  140. ///
  141. /// <param name="baseFilter">Filter to iterate.</param>
  142. ///
  143. public FilterIterator( IFilter baseFilter )
  144. {
  145. this.baseFilter = baseFilter;
  146. }
  147. /// <summary>
  148. /// Initializes a new instance of the <see cref="FilterIterator"/> class.
  149. /// </summary>
  150. ///
  151. /// <param name="baseFilter">Filter to iterate.</param>
  152. /// <param name="iterations">Iterations amount.</param>
  153. ///
  154. public FilterIterator( IFilter baseFilter, int iterations )
  155. {
  156. this.baseFilter = baseFilter;
  157. this.iterations = iterations;
  158. }
  159. /// <summary>
  160. /// Apply filter to an image.
  161. /// </summary>
  162. ///
  163. /// <param name="image">Source image to apply filter to.</param>
  164. ///
  165. /// <returns>Returns filter's result obtained by applying the filter to
  166. /// the source image.</returns>
  167. ///
  168. /// <remarks>The method keeps the source image unchanged and returns the
  169. /// the result of image processing filter as new image.</remarks>
  170. ///
  171. public Bitmap Apply( Bitmap image )
  172. {
  173. // lock source bitmap data
  174. BitmapData imageData = image.LockBits(
  175. new Rectangle( 0, 0, image.Width, image.Height ),
  176. ImageLockMode.ReadOnly, image.PixelFormat );
  177. // apply the filter
  178. Bitmap dstImage = Apply( imageData );
  179. // unlock source image
  180. image.UnlockBits( imageData );
  181. return dstImage;
  182. }
  183. /// <summary>
  184. /// Apply filter to an image.
  185. /// </summary>
  186. ///
  187. /// <param name="imageData">Source image to apply filter to.</param>
  188. ///
  189. /// <returns>Returns filter's result obtained by applying the filter to
  190. /// the source image.</returns>
  191. ///
  192. /// <remarks>The filter accepts bitmap data as input and returns the result
  193. /// of image processing filter as new image. The source image data are kept
  194. /// unchanged.</remarks>
  195. ///
  196. public Bitmap Apply( BitmapData imageData )
  197. {
  198. // initial iteration
  199. Bitmap dstImg = baseFilter.Apply( imageData );
  200. Bitmap tmpImg;
  201. // continue to iterate
  202. for ( int i = 1; i < iterations; i++ )
  203. {
  204. tmpImg = dstImg;
  205. dstImg = baseFilter.Apply( tmpImg );
  206. tmpImg.Dispose( );
  207. }
  208. return dstImg;
  209. }
  210. /// <summary>
  211. /// Apply filter to an image in unmanaged memory.
  212. /// </summary>
  213. ///
  214. /// <param name="image">Source image in unmanaged memory to apply filter to.</param>
  215. ///
  216. /// <returns>Returns filter's result obtained by applying the filter to
  217. /// the source image.</returns>
  218. ///
  219. /// <remarks>The method keeps the source image unchanged and returns the
  220. /// the result of image processing filter as new image.</remarks>
  221. ///
  222. public UnmanagedImage Apply( UnmanagedImage image )
  223. {
  224. // initial iteration
  225. UnmanagedImage dstImg = baseFilter.Apply( image );
  226. UnmanagedImage tmpImg;
  227. // continue to iterate
  228. for ( int i = 1; i < iterations; i++ )
  229. {
  230. tmpImg = dstImg;
  231. dstImg = baseFilter.Apply( tmpImg );
  232. tmpImg.Dispose( );
  233. }
  234. return dstImg;
  235. }
  236. /// <summary>
  237. /// Apply filter to an image in unmanaged memory.
  238. /// </summary>
  239. ///
  240. /// <param name="sourceImage">Source image in unmanaged memory to apply filter to.</param>
  241. /// <param name="destinationImage">Destination image in unmanaged memory to put result into.</param>
  242. ///
  243. /// <remarks><para>The method keeps the source image unchanged and puts result of image processing
  244. /// into destination image.</para>
  245. ///
  246. /// <para><note>The destination image must have the same width and height as source image. Also
  247. /// destination image must have pixel format, which is expected by particular filter (see
  248. /// <see cref="FormatTransalations"/> property for information about pixel format conversions).</note></para>
  249. /// </remarks>
  250. ///
  251. public void Apply( UnmanagedImage sourceImage, UnmanagedImage destinationImage )
  252. {
  253. if ( iterations == 1 )
  254. {
  255. baseFilter.Apply( sourceImage, destinationImage );
  256. }
  257. else
  258. {
  259. // initial iteration
  260. UnmanagedImage dstImg = baseFilter.Apply( sourceImage );
  261. UnmanagedImage tmpImg;
  262. iterations--;
  263. // continue to iterate
  264. for ( int i = 1; i < iterations; i++ )
  265. {
  266. tmpImg = dstImg;
  267. dstImg = baseFilter.Apply( tmpImg );
  268. tmpImg.Dispose( );
  269. }
  270. // put result of last iteration into the specified destination
  271. baseFilter.Apply( dstImg, destinationImage );
  272. }
  273. }
  274. }
  275. }