21 #include "config_auto.h"
65 max_noise_count_(static_cast<int>(kMaxSmallNeighboursPerPix *
66 gridsize * gridsize)),
67 noise_density_(
NULL) {
74 delete noise_density_;
95 BLOBNBOX_IT blob_it(&blob_block->
blobs);
96 for (blob_it.mark_cycle_pt(); !blob_it.cycled_list(); blob_it.forward()) {
99 perimeter_area_ratio *= perimeter_area_ratio / blob->
enclosed_area();
105 noise_density_ = ComputeNoiseDensity(debug, photo_map, &good_grid);
109 pixWrite(
"junknoisemask.png", pix, IFF_PNG);
112 #ifndef GRAPHICS_DISABLED
116 #endif // GRAPHICS_DISABLED
119 MarkAndDeleteNonTextBlobs(&blob_block->
large_blobs,
120 kMaxLargeOverlapsWithSmall,
122 MarkAndDeleteNonTextBlobs(&blob_block->
blobs, kMaxMediumOverlapsWithSmall,
127 MarkAndDeleteNonTextBlobs(&blob_block->
large_blobs,
128 kMaxLargeOverlapsWithMedium,
132 MarkAndDeleteNonTextBlobs(&blob_block->
noise_blobs, -1,
134 MarkAndDeleteNonTextBlobs(&blob_block->
small_blobs, -1,
136 MarkAndDeleteNonTextBlobs(&blob_block->
blobs, -1,
139 #ifndef GRAPHICS_DISABLED
141 #endif // GRAPHICS_DISABLED
142 pixWrite(
"junkccphotomask.png", pix, IFF_PNG);
143 #ifndef GRAPHICS_DISABLED
146 #endif // GRAPHICS_DISABLED
158 IntGrid* CCNonTextDetect::ComputeNoiseDensity(
bool debug, Pix* photo_map,
167 int height = pixGetHeight(photo_map);
168 int photo_offset =
IntCastRounded(max_noise_count_ * kPhotoOffsetFraction);
172 if (max_noise_count_ < noise + photo_offset &&
173 noise <= max_noise_count_) {
177 int bottom = height - y *
gridsize();
181 noise_density->
SetGridCell(x, y, noise + photo_offset);
184 if (debug && noise > max_noise_count_ &&
186 tprintf(
"At %d, %d, noise = %d, good=%d, orig=%d, thr=%d\n",
192 if (noise > max_noise_count_ &&
194 noise_counts->
GridCellValue(x, y) * kOriginalNoiseMultiple <=
202 return noise_density;
208 static TBOX AttemptBoxExpansion(
const TBOX& box,
const IntGrid& noise_density,
210 TBOX expanded_box(box);
211 expanded_box.set_right(box.
right() + pad);
212 if (!noise_density.AnyZeroInRect(expanded_box))
216 if (!noise_density.AnyZeroInRect(expanded_box))
220 if (!noise_density.AnyZeroInRect(expanded_box))
224 if (!noise_density.AnyZeroInRect(expanded_box))
227 expanded_box.
pad(kNoisePadding, kNoisePadding);
228 if (!noise_density.AnyZeroInRect(expanded_box))
247 void CCNonTextDetect::MarkAndDeleteNonTextBlobs(BLOBNBOX_LIST* blobs,
248 int max_blob_overlaps,
253 BLOBNBOX_IT blob_it(blobs);
254 BLOBNBOX_LIST dead_blobs;
255 BLOBNBOX_IT dead_it(&dead_blobs);
256 for (blob_it.mark_cycle_pt(); !blob_it.cycled_list(); blob_it.forward()) {
260 (max_blob_overlaps < 0 ||
261 !BlobOverlapsTooMuch(blob, max_blob_overlaps))) {
263 #ifndef GRAPHICS_DISABLED
265 blob->
plot(win, ok_color, ok_color);
266 #endif // GRAPHICS_DISABLED
272 pixRasterop(nontext_mask, box.
left(), imageheight - box.
top(),
275 pixDestroy(&blob_pix);
281 box = AttemptBoxExpansion(box, *noise_density_,
gridsize());
284 pixRasterop(nontext_mask, box.
left(), imageheight - box.
top(),
287 #ifndef GRAPHICS_DISABLED
290 #endif // GRAPHICS_DISABLED
296 delete blob->
cblob();
297 dead_it.add_to_end(blob_it.extract());
304 bool CCNonTextDetect::BlobOverlapsTooMuch(
BLOBNBOX* blob,
int max_overlaps) {
309 rsearch.StartRectSearch(box);
310 rsearch.SetUniqueMode(
true);
312 int overlap_count = 0;
313 while (overlap_count <= max_overlaps &&
314 (neighbour = rsearch.NextRectSearch()) !=
NULL) {
317 if (overlap_count > max_overlaps)
GridSearch< BLOBNBOX, BLOBNBOX_CLIST, BLOBNBOX_C_IT > BlobGridSearch
const double kMinGoodTextPARatio
static bool BoundsWithinRect(Pix *pix, int *x_start, int *y_start, int *x_end, int *y_end)
const int kMaxLargeOverlapsWithMedium
int GridCellValue(int grid_x, int grid_y) const
void InsertBlobList(BLOBNBOX_LIST *blobs)
Pix * ThresholdToPix(int threshold) const
void pad(int xpad, int ypad)
BLOBNBOX_LIST small_blobs
void InsertBBox(bool h_spread, bool v_spread, BLOBNBOX *bbox)
bool AnyZeroInRect(const TBOX &rect) const
IntGrid * NeighbourhoodSum() const
inT16 y() const
access_function
bool RectMostlyOverThreshold(const TBOX &rect, int threshold) const
SVEvent * AwaitEvent(SVEventType type)
BLOBNBOX_LIST noise_blobs
const int kMaxMediumOverlapsWithSmall
const ICOORD & bleft() const
bool major_overlap(const TBOX &box) const
IntGrid * CountCellElements()
Pix * ComputeNonTextMask(bool debug, Pix *photo_map, TO_BLOCK *blob_block)
inT32 enclosed_area() const
ScrollView * MakeWindow(int x, int y, const char *window_name)
virtual ~CCNonTextDetect()
const double kMaxSmallNeighboursPerPix
const int kOriginalNoiseMultiple
int IntCastRounded(double x)
void SetGridCell(int grid_x, int grid_y, int value)
inT16 x() const
access function
const ICOORD & tright() const
void plot(ScrollView *window, ScrollView::Color blob_colour, ScrollView::Color child_colour)
const TBOX & bounding_box() const
const int kMaxLargeOverlapsWithSmall
CCNonTextDetect(int gridsize, const ICOORD &bleft, const ICOORD &tright)
BLOBNBOX_LIST large_blobs
const double kPhotoOffsetFraction