tesseract  3.05.02
topitch.h File Reference
#include "blobbox.h"

Go to the source code of this file.

Namespaces

 tesseract
 

Functions

void compute_fixed_pitch (ICOORD page_tr, TO_BLOCK_LIST *port_blocks, float gradient, FCOORD rotation, BOOL8 testing_on)
 
void fix_row_pitch (TO_ROW *bad_row, TO_BLOCK *bad_block, TO_BLOCK_LIST *blocks, inT32 row_target, inT32 block_target)
 
void compute_block_pitch (TO_BLOCK *block, FCOORD rotation, inT32 block_index, BOOL8 testing_on)
 
BOOL8 compute_rows_pitch (TO_BLOCK *block, inT32 block_index, BOOL8 testing_on)
 
BOOL8 try_doc_fixed (ICOORD page_tr, TO_BLOCK_LIST *port_blocks, float gradient)
 
BOOL8 try_block_fixed (TO_BLOCK *block, inT32 block_index)
 
BOOL8 try_rows_fixed (TO_BLOCK *block, inT32 block_index, BOOL8 testing_on)
 
void print_block_counts (TO_BLOCK *block, inT32 block_index)
 
void count_block_votes (TO_BLOCK *block, inT32 &def_fixed, inT32 &def_prop, inT32 &maybe_fixed, inT32 &maybe_prop, inT32 &corr_fixed, inT32 &corr_prop, inT32 &dunno)
 
BOOL8 row_pitch_stats (TO_ROW *row, inT32 maxwidth, BOOL8 testing_on)
 
BOOL8 find_row_pitch (TO_ROW *row, inT32 maxwidth, inT32 dm_gap, TO_BLOCK *block, inT32 block_index, inT32 row_index, BOOL8 testing_on)
 
BOOL8 fixed_pitch_row (TO_ROW *row, BLOCK *block, inT32 block_index)
 
BOOL8 count_pitch_stats (TO_ROW *row, STATS *gap_stats, STATS *pitch_stats, float initial_pitch, float min_space, BOOL8 ignore_outsize, BOOL8 split_outsize, inT32 dm_gap)
 
float tune_row_pitch (TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float &initial_pitch, float &best_sp_sd, inT16 &best_mid_cuts, ICOORDELT_LIST *best_cells, BOOL8 testing_on)
 
float tune_row_pitch2 (TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float &initial_pitch, float &best_sp_sd, inT16 &best_mid_cuts, ICOORDELT_LIST *best_cells, BOOL8 testing_on)
 
float compute_pitch_sd (TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float initial_pitch, float &sp_sd, inT16 &mid_cuts, ICOORDELT_LIST *row_cells, BOOL8 testing_on, inT16 start=0, inT16 end=0)
 
float compute_pitch_sd2 (TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float initial_pitch, inT16 &occupation, inT16 &mid_cuts, ICOORDELT_LIST *row_cells, BOOL8 testing_on, inT16 start=0, inT16 end=0)
 
void print_pitch_sd (TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float initial_pitch)
 
void find_repeated_chars (TO_BLOCK *block, BOOL8 testing_on)
 
void plot_fp_word (TO_BLOCK *block, float pitch, float nonspace)
 

Variables

bool textord_debug_pitch_test = FALSE
 
bool textord_debug_pitch_metric = FALSE
 
bool textord_show_row_cuts = FALSE
 
bool textord_show_page_cuts = FALSE
 
bool textord_pitch_cheat = FALSE
 
bool textord_blockndoc_fixed = TRUE
 
bool textord_fast_pitch_test = FALSE
 
double textord_projection_scale = 0.125
 
double textord_balance_factor = 2.0
 

Function Documentation

◆ compute_block_pitch()

void compute_block_pitch ( TO_BLOCK block,
FCOORD  rotation,
inT32  block_index,
BOOL8  testing_on 
)

Definition at line 308 of file topitch.cpp.

311  { // correct orientation
312  TBOX block_box; //bounding box
313 
314  block_box = block->block->bounding_box ();
315  if (testing_on && textord_debug_pitch_test) {
316  tprintf ("Block %d at (%d,%d)->(%d,%d)\n",
317  block_index,
318  block_box.left (), block_box.bottom (),
319  block_box.right (), block_box.top ());
320  }
321  block->min_space = (inT32) floor (block->xheight
323  block->max_nonspace = (inT32) ceil (block->xheight
325  block->fixed_pitch = 0.0f;
326  block->space_size = (float) block->min_space;
327  block->kern_size = (float) block->max_nonspace;
328  block->pr_nonsp = block->xheight * words_default_prop_nonspace;
330  if (!block->get_rows ()->empty ()) {
331  ASSERT_HOST (block->xheight > 0);
332  find_repeated_chars(block, textord_show_initial_words && testing_on);
333 #ifndef GRAPHICS_DISABLED
334  if (textord_show_initial_words && testing_on)
335  //overlap_picture_ops(TRUE);
337 #endif
338  compute_rows_pitch(block,
339  block_index,
340  textord_debug_pitch_test &&testing_on);
341  }
342 }
inT32 min_space
Definition: blobbox.h:788
static void Update()
Definition: scrollview.cpp:715
EXTERN double textord_words_default_minspace
Definition: tovars.cpp:48
BLOCK * block
Definition: blobbox.h:773
EXTERN double textord_spacesize_ratioprop
Definition: tovars.cpp:80
float kern_size
Definition: blobbox.h:786
inT16 bottom() const
Definition: rect.h:61
inT32 max_nonspace
Definition: blobbox.h:789
float pr_space
Definition: blobbox.h:792
BOOL8 compute_rows_pitch(TO_BLOCK *block, inT32 block_index, BOOL8 testing_on)
Definition: topitch.cpp:351
float xheight
Definition: blobbox.h:784
inT16 left() const
Definition: rect.h:68
TO_ROW_LIST * get_rows()
Definition: blobbox.h:700
EXTERN double textord_words_default_nonspace
Definition: tovars.cpp:51
float space_size
Definition: blobbox.h:787
int inT32
Definition: host.h:35
#define tprintf(...)
Definition: tprintf.h:31
EXTERN bool textord_debug_pitch_test
Definition: topitch.cpp:44
inT16 top() const
Definition: rect.h:54
Definition: rect.h:30
inT16 right() const
Definition: rect.h:75
float fixed_pitch
Definition: blobbox.h:785
void find_repeated_chars(TO_BLOCK *block, BOOL8 testing_on)
Definition: topitch.cpp:1756
float pr_nonsp
Definition: blobbox.h:793
#define ASSERT_HOST(x)
Definition: errcode.h:84
EXTERN double words_default_prop_nonspace
Definition: tovars.cpp:72
void bounding_box(ICOORD &bottom_left, ICOORD &top_right) const
get box
Definition: pdblock.h:59
EXTERN bool textord_show_initial_words
Definition: tovars.cpp:25

◆ compute_fixed_pitch()

void compute_fixed_pitch ( ICOORD  page_tr,
TO_BLOCK_LIST *  port_blocks,
float  gradient,
FCOORD  rotation,
BOOL8  testing_on 
)

Definition at line 73 of file topitch.cpp.

77  { // correct orientation
78  TO_BLOCK_IT block_it; //iterator
79  TO_BLOCK *block; //current block;
80  TO_ROW_IT row_it; //row iterator
81  TO_ROW *row; //current row
82  int block_index; //block number
83  int row_index; //row number
84 
85 #ifndef GRAPHICS_DISABLED
86  if (textord_show_initial_words && testing_on) {
87  if (to_win == NULL)
88  create_to_win(page_tr);
89  }
90 #endif
91 
92  block_it.set_to_list (port_blocks);
93  block_index = 1;
94  for (block_it.mark_cycle_pt (); !block_it.cycled_list ();
95  block_it.forward ()) {
96  block = block_it.data ();
97  compute_block_pitch(block, rotation, block_index, testing_on);
98  block_index++;
99  }
100 
101  if (!try_doc_fixed (page_tr, port_blocks, gradient)) {
102  block_index = 1;
103  for (block_it.mark_cycle_pt (); !block_it.cycled_list ();
104  block_it.forward ()) {
105  block = block_it.data ();
106  if (!try_block_fixed (block, block_index))
107  try_rows_fixed(block, block_index, testing_on);
108  block_index++;
109  }
110  }
111 
112  block_index = 1;
113  for (block_it.mark_cycle_pt(); !block_it.cycled_list();
114  block_it.forward()) {
115  block = block_it.data ();
116  POLY_BLOCK* pb = block->block->poly_block();
117  if (pb != NULL && !pb->IsText()) continue; // Non-text doesn't exist!
118  row_it.set_to_list (block->get_rows ());
119  row_index = 1;
120  for (row_it.mark_cycle_pt (); !row_it.cycled_list (); row_it.forward ()) {
121  row = row_it.data ();
122  fix_row_pitch(row, block, port_blocks, row_index, block_index);
123  row_index++;
124  }
125  block_index++;
126  }
127 #ifndef GRAPHICS_DISABLED
128  if (textord_show_initial_words && testing_on) {
130  }
131 #endif
132 }
ScrollView * create_to_win(ICOORD page_tr)
Definition: drawtord.cpp:47
static void Update()
Definition: scrollview.cpp:715
BOOL8 try_doc_fixed(ICOORD page_tr, TO_BLOCK_LIST *port_blocks, float gradient)
Definition: topitch.cpp:395
BLOCK * block
Definition: blobbox.h:773
POLY_BLOCK * poly_block() const
Definition: pdblock.h:55
TO_ROW_LIST * get_rows()
Definition: blobbox.h:700
bool IsText() const
Definition: polyblk.h:52
void compute_block_pitch(TO_BLOCK *block, FCOORD rotation, inT32 block_index, BOOL8 testing_on)
Definition: topitch.cpp:308
void fix_row_pitch(TO_ROW *bad_row, TO_BLOCK *bad_block, TO_BLOCK_LIST *blocks, inT32 row_target, inT32 block_target)
Definition: topitch.cpp:142
BOOL8 try_block_fixed(TO_BLOCK *block, inT32 block_index)
Definition: topitch.cpp:535
EXTERN ScrollView * to_win
Definition: drawtord.cpp:38
BOOL8 try_rows_fixed(TO_BLOCK *block, inT32 block_index, BOOL8 testing_on)
Definition: topitch.cpp:549
EXTERN bool textord_show_initial_words
Definition: tovars.cpp:25

◆ compute_pitch_sd()

float compute_pitch_sd ( TO_ROW row,
STATS projection,
inT16  projection_left,
inT16  projection_right,
float  space_size,
float  initial_pitch,
float &  sp_sd,
inT16 mid_cuts,
ICOORDELT_LIST *  row_cells,
BOOL8  testing_on,
inT16  start = 0,
inT16  end = 0 
)

Definition at line 1373 of file topitch.cpp.

1386  {
1387  inT16 occupation; //no of cells in word.
1388  //blobs
1389  BLOBNBOX_IT blob_it = row->blob_list ();
1390  BLOBNBOX_IT start_it; //start of word
1391  BLOBNBOX_IT plot_it; //for plotting
1392  inT16 blob_count; //no of blobs
1393  TBOX blob_box; //bounding box
1394  TBOX prev_box; //of super blob
1395  inT32 prev_right; //of word sync
1396  int scale_factor; //on scores for big words
1397  inT32 sp_count; //spaces
1398  FPSEGPT_LIST seg_list; //char cells
1399  FPSEGPT_IT seg_it; //iterator
1400  inT16 segpos; //position of segment
1401  inT16 cellpos; //previous cell boundary
1402  //iterator
1403  ICOORDELT_IT cell_it = row_cells;
1404  ICOORDELT *cell; //new cell
1405  double sqsum; //sum of squares
1406  double spsum; //of spaces
1407  double sp_var; //space error
1408  double word_sync; //result for word
1409  inT32 total_count; //total blobs
1410 
1411  if ((pitsync_linear_version & 3) > 1) {
1412  word_sync = compute_pitch_sd2 (row, projection, projection_left,
1413  projection_right, initial_pitch,
1414  occupation, mid_cuts, row_cells,
1415  testing_on, start, end);
1416  sp_sd = occupation;
1417  return word_sync;
1418  }
1419  mid_cuts = 0;
1420  cellpos = 0;
1421  total_count = 0;
1422  sqsum = 0;
1423  sp_count = 0;
1424  spsum = 0;
1425  prev_right = -1;
1426  if (blob_it.empty ())
1427  return space_size * 10;
1428 #ifndef GRAPHICS_DISABLED
1429  if (testing_on && to_win != NULL) {
1430  blob_box = blob_it.data ()->bounding_box ();
1431  projection->plot (to_win, projection_left,
1432  row->intercept (), 1.0f, -1.0f, ScrollView::CORAL);
1433  }
1434 #endif
1435  start_it = blob_it;
1436  blob_count = 0;
1437  blob_box = box_next (&blob_it);//first blob
1438  blob_it.mark_cycle_pt ();
1439  do {
1440  for (; blob_count > 0; blob_count--)
1441  box_next(&start_it);
1442  do {
1443  prev_box = blob_box;
1444  blob_count++;
1445  blob_box = box_next (&blob_it);
1446  }
1447  while (!blob_it.cycled_list ()
1448  && blob_box.left () - prev_box.right () < space_size);
1449  plot_it = start_it;
1450  if (pitsync_linear_version & 3)
1451  word_sync =
1452  check_pitch_sync2 (&start_it, blob_count, (inT16) initial_pitch, 2,
1453  projection, projection_left, projection_right,
1455  occupation, &seg_list, start, end);
1456  else
1457  word_sync =
1458  check_pitch_sync (&start_it, blob_count, (inT16) initial_pitch, 2,
1459  projection, &seg_list);
1460  if (testing_on) {
1461  tprintf ("Word ending at (%d,%d), len=%d, sync rating=%g, ",
1462  prev_box.right (), prev_box.top (),
1463  seg_list.length () - 1, word_sync);
1464  seg_it.set_to_list (&seg_list);
1465  for (seg_it.mark_cycle_pt (); !seg_it.cycled_list ();
1466  seg_it.forward ()) {
1467  if (seg_it.data ()->faked)
1468  tprintf ("(F)");
1469  tprintf ("%d, ", seg_it.data ()->position ());
1470  // tprintf("C=%g, s=%g, sq=%g\n",
1471  // seg_it.data()->cost_function(),
1472  // seg_it.data()->sum(),
1473  // seg_it.data()->squares());
1474  }
1475  tprintf ("\n");
1476  }
1477 #ifndef GRAPHICS_DISABLED
1478  if (textord_show_fixed_cuts && blob_count > 0 && to_win != NULL)
1479  plot_fp_cells2(to_win, ScrollView::GOLDENROD, row, &seg_list);
1480 #endif
1481  seg_it.set_to_list (&seg_list);
1482  if (prev_right >= 0) {
1483  sp_var = seg_it.data ()->position () - prev_right;
1484  sp_var -= floor (sp_var / initial_pitch + 0.5) * initial_pitch;
1485  sp_var *= sp_var;
1486  spsum += sp_var;
1487  sp_count++;
1488  }
1489  for (seg_it.mark_cycle_pt (); !seg_it.cycled_list (); seg_it.forward ()) {
1490  segpos = seg_it.data ()->position ();
1491  if (cell_it.empty () || segpos > cellpos + initial_pitch / 2) {
1492  //big gap
1493  while (!cell_it.empty () && segpos > cellpos + initial_pitch * 3 / 2) {
1494  cell = new ICOORDELT (cellpos + (inT16) initial_pitch, 0);
1495  cell_it.add_after_then_move (cell);
1496  cellpos += (inT16) initial_pitch;
1497  }
1498  //make new one
1499  cell = new ICOORDELT (segpos, 0);
1500  cell_it.add_after_then_move (cell);
1501  cellpos = segpos;
1502  }
1503  else if (segpos > cellpos - initial_pitch / 2) {
1504  cell = cell_it.data ();
1505  //average positions
1506  cell->set_x ((cellpos + segpos) / 2);
1507  cellpos = cell->x ();
1508  }
1509  }
1510  seg_it.move_to_last ();
1511  prev_right = seg_it.data ()->position ();
1513  scale_factor = (seg_list.length () - 2) / 2;
1514  if (scale_factor < 1)
1515  scale_factor = 1;
1516  }
1517  else
1518  scale_factor = 1;
1519  sqsum += word_sync * scale_factor;
1520  total_count += (seg_list.length () - 1) * scale_factor;
1521  seg_list.clear ();
1522  }
1523  while (!blob_it.cycled_list ());
1524  sp_sd = sp_count > 0 ? sqrt (spsum / sp_count) : 0;
1525  return total_count > 0 ? sqrt (sqsum / total_count) : space_size * 10;
1526 }
float compute_pitch_sd2(TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float initial_pitch, inT16 &occupation, inT16 &mid_cuts, ICOORDELT_LIST *row_cells, BOOL8 testing_on, inT16 start, inT16 end)
Definition: topitch.cpp:1536
EXTERN double textord_projection_scale
Definition: topitch.cpp:57
short inT16
Definition: host.h:33
double check_pitch_sync(BLOBNBOX_IT *blob_it, inT16 blob_count, inT16 pitch, inT16 pitch_error, STATS *projection, FPSEGPT_LIST *seg_list)
Definition: pitsync1.cpp:148
EXTERN bool textord_pitch_scalebigwords
Definition: tovars.cpp:69
void plot_fp_cells2(ScrollView *win, ScrollView::Color colour, TO_ROW *row, FPSEGPT_LIST *seg_list)
Definition: drawtord.cpp:364
BLOBNBOX_LIST * blob_list()
Definition: blobbox.h:595
EXTERN bool textord_show_fixed_cuts
Definition: drawtord.cpp:36
inT16 x() const
access function
Definition: points.h:52
TBOX box_next(BLOBNBOX_IT *it)
Definition: blobbox.cpp:631
inT16 left() const
Definition: rect.h:68
int inT32
Definition: host.h:35
double check_pitch_sync2(BLOBNBOX_IT *blob_it, inT16 blob_count, inT16 pitch, inT16 pitch_error, STATS *projection, inT16 projection_left, inT16 projection_right, float projection_scale, inT16 &occupation_count, FPSEGPT_LIST *seg_list, inT16 start, inT16 end)
Definition: pithsync.cpp:298
#define tprintf(...)
Definition: tprintf.h:31
float xheight
Definition: blobbox.h:653
inT16 top() const
Definition: rect.h:54
void set_x(inT16 xin)
rewrite function
Definition: points.h:61
Definition: rect.h:30
inT16 right() const
Definition: rect.h:75
EXTERN ScrollView * to_win
Definition: drawtord.cpp:38
void plot(ScrollView *window, float xorigin, float yorigin, float xscale, float yscale, ScrollView::Color colour) const
Definition: statistc.cpp:585
float intercept() const
Definition: blobbox.h:584

◆ compute_pitch_sd2()

float compute_pitch_sd2 ( TO_ROW row,
STATS projection,
inT16  projection_left,
inT16  projection_right,
float  initial_pitch,
inT16 occupation,
inT16 mid_cuts,
ICOORDELT_LIST *  row_cells,
BOOL8  testing_on,
inT16  start = 0,
inT16  end = 0 
)

Definition at line 1536 of file topitch.cpp.

1548  {
1549  //blobs
1550  BLOBNBOX_IT blob_it = row->blob_list ();
1551  BLOBNBOX_IT plot_it;
1552  inT16 blob_count; //no of blobs
1553  TBOX blob_box; //bounding box
1554  FPSEGPT_LIST seg_list; //char cells
1555  FPSEGPT_IT seg_it; //iterator
1556  inT16 segpos; //position of segment
1557  //iterator
1558  ICOORDELT_IT cell_it = row_cells;
1559  ICOORDELT *cell; //new cell
1560  double word_sync; //result for word
1561 
1562  mid_cuts = 0;
1563  if (blob_it.empty ()) {
1564  occupation = 0;
1565  return initial_pitch * 10;
1566  }
1567 #ifndef GRAPHICS_DISABLED
1568  if (testing_on && to_win != NULL) {
1569  projection->plot (to_win, projection_left,
1570  row->intercept (), 1.0f, -1.0f, ScrollView::CORAL);
1571  }
1572 #endif
1573  blob_count = 0;
1574  blob_it.mark_cycle_pt ();
1575  do {
1576  //first blob
1577  blob_box = box_next (&blob_it);
1578  blob_count++;
1579  }
1580  while (!blob_it.cycled_list ());
1581  plot_it = blob_it;
1582  word_sync = check_pitch_sync2 (&blob_it, blob_count, (inT16) initial_pitch,
1583  2, projection, projection_left,
1584  projection_right,
1586  occupation, &seg_list, start, end);
1587  if (testing_on) {
1588  tprintf ("Row ending at (%d,%d), len=%d, sync rating=%g, ",
1589  blob_box.right (), blob_box.top (),
1590  seg_list.length () - 1, word_sync);
1591  seg_it.set_to_list (&seg_list);
1592  for (seg_it.mark_cycle_pt (); !seg_it.cycled_list (); seg_it.forward ()) {
1593  if (seg_it.data ()->faked)
1594  tprintf ("(F)");
1595  tprintf ("%d, ", seg_it.data ()->position ());
1596  // tprintf("C=%g, s=%g, sq=%g\n",
1597  // seg_it.data()->cost_function(),
1598  // seg_it.data()->sum(),
1599  // seg_it.data()->squares());
1600  }
1601  tprintf ("\n");
1602  }
1603 #ifndef GRAPHICS_DISABLED
1604  if (textord_show_fixed_cuts && blob_count > 0 && to_win != NULL)
1605  plot_fp_cells2(to_win, ScrollView::GOLDENROD, row, &seg_list);
1606 #endif
1607  seg_it.set_to_list (&seg_list);
1608  for (seg_it.mark_cycle_pt (); !seg_it.cycled_list (); seg_it.forward ()) {
1609  segpos = seg_it.data ()->position ();
1610  //make new one
1611  cell = new ICOORDELT (segpos, 0);
1612  cell_it.add_after_then_move (cell);
1613  if (seg_it.at_last ())
1614  mid_cuts = seg_it.data ()->cheap_cuts ();
1615  }
1616  seg_list.clear ();
1617  return occupation > 0 ? sqrt (word_sync / occupation) : initial_pitch * 10;
1618 }
EXTERN double textord_projection_scale
Definition: topitch.cpp:57
short inT16
Definition: host.h:33
void plot_fp_cells2(ScrollView *win, ScrollView::Color colour, TO_ROW *row, FPSEGPT_LIST *seg_list)
Definition: drawtord.cpp:364
BLOBNBOX_LIST * blob_list()
Definition: blobbox.h:595
EXTERN bool textord_show_fixed_cuts
Definition: drawtord.cpp:36
TBOX box_next(BLOBNBOX_IT *it)
Definition: blobbox.cpp:631
double check_pitch_sync2(BLOBNBOX_IT *blob_it, inT16 blob_count, inT16 pitch, inT16 pitch_error, STATS *projection, inT16 projection_left, inT16 projection_right, float projection_scale, inT16 &occupation_count, FPSEGPT_LIST *seg_list, inT16 start, inT16 end)
Definition: pithsync.cpp:298
#define tprintf(...)
Definition: tprintf.h:31
float xheight
Definition: blobbox.h:653
inT16 top() const
Definition: rect.h:54
Definition: rect.h:30
inT16 right() const
Definition: rect.h:75
EXTERN ScrollView * to_win
Definition: drawtord.cpp:38
void plot(ScrollView *window, float xorigin, float yorigin, float xscale, float yscale, ScrollView::Color colour) const
Definition: statistc.cpp:585
float intercept() const
Definition: blobbox.h:584

◆ compute_rows_pitch()

BOOL8 compute_rows_pitch ( TO_BLOCK block,
inT32  block_index,
BOOL8  testing_on 
)

Definition at line 351 of file topitch.cpp.

355  {
356  inT32 maxwidth; //of spaces
357  TO_ROW *row; //current row
358  inT32 row_index; //row number.
359  float lower, upper; //cluster thresholds
360  TO_ROW_IT row_it = block->get_rows ();
361 
362  row_index = 1;
363  for (row_it.mark_cycle_pt (); !row_it.cycled_list (); row_it.forward ()) {
364  row = row_it.data ();
365  ASSERT_HOST (row->xheight > 0);
367  maxwidth = (inT32) ceil (row->xheight * textord_words_maxspace);
368  if (row_pitch_stats (row, maxwidth, testing_on)
369  && find_row_pitch (row, maxwidth,
370  textord_dotmatrix_gap + 1, block, block_index,
371  row_index, testing_on)) {
372  if (row->fixed_pitch == 0) {
373  lower = row->pr_nonsp;
374  upper = row->pr_space;
375  row->space_size = upper;
376  row->kern_size = lower;
377  }
378  }
379  else {
380  row->fixed_pitch = 0.0f; //insufficient data
382  }
383  row_index++;
384  }
385  return FALSE;
386 }
BOOL8 find_row_pitch(TO_ROW *row, inT32 maxwidth, inT32 dm_gap, TO_BLOCK *block, inT32 block_index, inT32 row_index, BOOL8 testing_on)
Definition: topitch.cpp:834
PITCH_TYPE pitch_decision
Definition: blobbox.h:646
BOOL8 row_pitch_stats(TO_ROW *row, inT32 maxwidth, BOOL8 testing_on)
Definition: topitch.cpp:701
float pr_space
Definition: blobbox.h:650
void compute_vertical_projection()
Definition: blobbox.cpp:791
#define FALSE
Definition: capi.h:46
TO_ROW_LIST * get_rows()
Definition: blobbox.h:700
int inT32
Definition: host.h:35
float space_size
Definition: blobbox.h:663
float xheight
Definition: blobbox.h:653
EXTERN double textord_words_maxspace
Definition: tovars.cpp:44
float fixed_pitch
Definition: blobbox.h:647
float pr_nonsp
Definition: blobbox.h:651
EXTERN int textord_dotmatrix_gap
Definition: tovars.cpp:35
#define ASSERT_HOST(x)
Definition: errcode.h:84
float kern_size
Definition: blobbox.h:662

◆ count_block_votes()

void count_block_votes ( TO_BLOCK block,
inT32 def_fixed,
inT32 def_prop,
inT32 maybe_fixed,
inT32 maybe_prop,
inT32 corr_fixed,
inT32 corr_prop,
inT32 dunno 
)

Definition at line 654 of file topitch.cpp.

662  {
663  TO_ROW *row; //current row
664  TO_ROW_IT row_it = block->get_rows ();
665 
666  for (row_it.mark_cycle_pt (); !row_it.cycled_list (); row_it.forward ()) {
667  row = row_it.data ();
668  switch (row->pitch_decision) {
669  case PITCH_DUNNO:
670  dunno++;
671  break;
672  case PITCH_DEF_PROP:
673  def_prop++;
674  break;
675  case PITCH_MAYBE_PROP:
676  maybe_prop++;
677  break;
678  case PITCH_DEF_FIXED:
679  def_fixed++;
680  break;
681  case PITCH_MAYBE_FIXED:
682  maybe_fixed++;
683  break;
684  case PITCH_CORR_PROP:
685  corr_prop++;
686  break;
687  case PITCH_CORR_FIXED:
688  corr_fixed++;
689  break;
690  }
691  }
692 }
PITCH_TYPE pitch_decision
Definition: blobbox.h:646
TO_ROW_LIST * get_rows()
Definition: blobbox.h:700

◆ count_pitch_stats()

BOOL8 count_pitch_stats ( TO_ROW row,
STATS gap_stats,
STATS pitch_stats,
float  initial_pitch,
float  min_space,
BOOL8  ignore_outsize,
BOOL8  split_outsize,
inT32  dm_gap 
)

Definition at line 1058 of file topitch.cpp.

1067  {
1068  BOOL8 prev_valid; //not word broken
1069  BLOBNBOX *blob; //current blob
1070  //blobs
1071  BLOBNBOX_IT blob_it = row->blob_list ();
1072  inT32 prev_right; //end of prev blob
1073  inT32 prev_centre; //centre of previous blob
1074  inT32 x_centre; //centre of this blob
1075  inT32 blob_width; //width of blob
1076  inT32 width_units; //no of widths in blob
1077  float width; //blob width
1078  TBOX blob_box; //bounding box
1079  TBOX joined_box; //of super blob
1080 
1081  gap_stats->clear ();
1082  pitch_stats->clear ();
1083  if (blob_it.empty ())
1084  return FALSE;
1085  prev_valid = FALSE;
1086  prev_centre = 0;
1087  prev_right = 0; // stop compiler warning
1088  joined_box = blob_it.data ()->bounding_box ();
1089  do {
1090  blob_it.forward ();
1091  blob = blob_it.data ();
1092  if (!blob->joined_to_prev ()) {
1093  blob_box = blob->bounding_box ();
1094  if ((blob_box.left () - joined_box.right () < dm_gap
1095  && !blob_it.at_first ())
1096  || blob->cblob() == NULL)
1097  joined_box += blob_box; //merge blobs
1098  else {
1099  blob_width = joined_box.width ();
1100  if (split_outsize) {
1101  width_units =
1102  (inT32) floor ((float) blob_width / initial_pitch + 0.5);
1103  if (width_units < 1)
1104  width_units = 1;
1105  width_units--;
1106  }
1107  else if (ignore_outsize) {
1108  width = (float) blob_width / initial_pitch;
1109  width_units = width < 1 + words_default_fixed_limit
1110  && width > 1 - words_default_fixed_limit ? 0 : -1;
1111  }
1112  else
1113  width_units = 0; //everything in
1114  x_centre = (inT32) (joined_box.left ()
1115  + (blob_width -
1116  width_units * initial_pitch) / 2);
1117  if (prev_valid && width_units >= 0) {
1118  // if (width_units>0)
1119  // {
1120  // tprintf("wu=%d, width=%d, xc=%d, adding %d\n",
1121  // width_units,blob_width,x_centre,x_centre-prev_centre);
1122  // }
1123  gap_stats->add (joined_box.left () - prev_right, 1);
1124  pitch_stats->add (x_centre - prev_centre, 1);
1125  }
1126  prev_centre = (inT32) (x_centre + width_units * initial_pitch);
1127  prev_right = joined_box.right ();
1128  prev_valid = blob_box.left () - joined_box.right () < min_space;
1129  prev_valid = prev_valid && width_units >= 0;
1130  joined_box = blob_box;
1131  }
1132  }
1133  }
1134  while (!blob_it.at_first ());
1135  return gap_stats->get_total () >= 3;
1136 }
const TBOX & bounding_box() const
Definition: blobbox.h:215
C_BLOB * cblob() const
Definition: blobbox.h:253
void clear()
Definition: statistc.cpp:81
void add(inT32 value, inT32 count)
Definition: statistc.cpp:101
inT16 width() const
Definition: rect.h:111
unsigned char BOOL8
Definition: host.h:46
BLOBNBOX_LIST * blob_list()
Definition: blobbox.h:595
inT32 get_total() const
Definition: statistc.h:86
bool joined_to_prev() const
Definition: blobbox.h:241
#define FALSE
Definition: capi.h:46
inT16 left() const
Definition: rect.h:68
int inT32
Definition: host.h:35
EXTERN double words_default_fixed_limit
Definition: tovars.cpp:74
Definition: rect.h:30
inT16 right() const
Definition: rect.h:75

◆ find_repeated_chars()

void find_repeated_chars ( TO_BLOCK block,
BOOL8  testing_on 
)

Definition at line 1756 of file topitch.cpp.

1757  { // Debug mode.
1758  POLY_BLOCK* pb = block->block->poly_block();
1759  if (pb != NULL && !pb->IsText())
1760  return; // Don't find repeated chars in non-text blocks.
1761 
1762  TO_ROW *row;
1763  BLOBNBOX_IT box_it;
1764  BLOBNBOX_IT search_it; // forward search
1765  WERD_IT word_it; // new words
1766  WERD *word; // new word
1767  TBOX word_box; // for plotting
1768  int blobcount, repeated_set;
1769 
1770  TO_ROW_IT row_it = block->get_rows();
1771  if (row_it.empty()) return; // empty block
1772  for (row_it.mark_cycle_pt(); !row_it.cycled_list(); row_it.forward()) {
1773  row = row_it.data();
1774  box_it.set_to_list(row->blob_list());
1775  if (box_it.empty()) continue; // no blobs in this row
1776  if (!row->rep_chars_marked()) {
1777  mark_repeated_chars(row);
1778  }
1779  if (row->num_repeated_sets() == 0) continue; // nothing to do for this row
1780  word_it.set_to_list(&row->rep_words);
1781  do {
1782  if (box_it.data()->repeated_set() != 0 &&
1783  !box_it.data()->joined_to_prev()) {
1784  blobcount = 1;
1785  repeated_set = box_it.data()->repeated_set();
1786  search_it = box_it;
1787  search_it.forward();
1788  while (!search_it.at_first() &&
1789  search_it.data()->repeated_set() == repeated_set) {
1790  blobcount++;
1791  search_it.forward();
1792  }
1793  // After the call to make_real_word() all the blobs from this
1794  // repeated set will be removed from the blob list. box_it will be
1795  // set to point to the blob after the end of the extracted sequence.
1796  word = make_real_word(&box_it, blobcount, box_it.at_first(), 1);
1797  if (!box_it.empty() && box_it.data()->joined_to_prev()) {
1798  tprintf("Bad box joined to prev at");
1799  box_it.data()->bounding_box().print();
1800  tprintf("After repeated word:");
1801  word->bounding_box().print();
1802  }
1803  ASSERT_HOST(box_it.empty() || !box_it.data()->joined_to_prev());
1804  word->set_flag(W_REP_CHAR, true);
1805  word->set_flag(W_DONT_CHOP, true);
1806  word_it.add_after_then_move(word);
1807  } else {
1808  box_it.forward();
1809  }
1810  } while (!box_it.at_first());
1811  }
1812 }
WERD_LIST rep_words
Definition: blobbox.h:664
int num_repeated_sets() const
Definition: blobbox.h:633
BLOCK * block
Definition: blobbox.h:773
BLOBNBOX_LIST * blob_list()
Definition: blobbox.h:595
POLY_BLOCK * poly_block() const
Definition: pdblock.h:55
Definition: werd.h:60
void mark_repeated_chars(TO_ROW *row)
Definition: makerow.cpp:2657
bool rep_chars_marked() const
Definition: blobbox.h:627
TO_ROW_LIST * get_rows()
Definition: blobbox.h:700
void print() const
Definition: rect.h:270
#define tprintf(...)
Definition: tprintf.h:31
bool IsText() const
Definition: polyblk.h:52
Definition: rect.h:30
void set_flag(WERD_FLAGS mask, BOOL8 value)
Definition: werd.h:129
#define ASSERT_HOST(x)
Definition: errcode.h:84
WERD * make_real_word(BLOBNBOX_IT *box_it, inT32 blobcount, BOOL8 bol, uinT8 blanks)
Definition: wordseg.cpp:594
TBOX bounding_box() const
Definition: werd.cpp:160

◆ find_row_pitch()

BOOL8 find_row_pitch ( TO_ROW row,
inT32  maxwidth,
inT32  dm_gap,
TO_BLOCK block,
inT32  block_index,
inT32  row_index,
BOOL8  testing_on 
)

Definition at line 834 of file topitch.cpp.

842  {
843  BOOL8 used_dm_model; //looks lik dot matrix
844  float min_space; //estimate threshold
845  float non_space; //gap size
846  float gap_iqr; //interquartile range
847  float pitch_iqr;
848  float dm_gap_iqr; //interquartile range
849  float dm_pitch_iqr;
850  float dm_pitch; //pitch with dm on
851  float pitch; //revised estimate
852  float initial_pitch; //guess at pitch
853  STATS gap_stats (0, maxwidth);
854  //centre-centre
855  STATS pitch_stats (0, maxwidth);
856 
857  row->fixed_pitch = 0.0f;
858  initial_pitch = row->fp_space;
859  if (initial_pitch > row->xheight * (1 + words_default_fixed_limit))
860  initial_pitch = row->xheight;//keep pitch decent
861  non_space = row->fp_nonsp;
862  if (non_space > initial_pitch)
863  non_space = initial_pitch;
864  min_space = (initial_pitch + non_space) / 2;
865 
866  if (!count_pitch_stats (row, &gap_stats, &pitch_stats,
867  initial_pitch, min_space, TRUE, FALSE, dm_gap)) {
868  dm_gap_iqr = 0.0001;
869  dm_pitch_iqr = maxwidth * 2.0f;
870  dm_pitch = initial_pitch;
871  }
872  else {
873  dm_gap_iqr = gap_stats.ile (0.75) - gap_stats.ile (0.25);
874  dm_pitch_iqr = pitch_stats.ile (0.75) - pitch_stats.ile (0.25);
875  dm_pitch = pitch_stats.ile (0.5);
876  }
877  gap_stats.clear ();
878  pitch_stats.clear ();
879  if (!count_pitch_stats (row, &gap_stats, &pitch_stats,
880  initial_pitch, min_space, TRUE, FALSE, 0)) {
881  gap_iqr = 0.0001;
882  pitch_iqr = maxwidth * 3.0f;
883  }
884  else {
885  gap_iqr = gap_stats.ile (0.75) - gap_stats.ile (0.25);
886  pitch_iqr = pitch_stats.ile (0.75) - pitch_stats.ile (0.25);
887  if (testing_on)
888  tprintf
889  ("First fp iteration:initial_pitch=%g, gap_iqr=%g, pitch_iqr=%g, pitch=%g\n",
890  initial_pitch, gap_iqr, pitch_iqr, pitch_stats.ile (0.5));
891  initial_pitch = pitch_stats.ile (0.5);
892  if (min_space > initial_pitch
893  && count_pitch_stats (row, &gap_stats, &pitch_stats,
894  initial_pitch, initial_pitch, TRUE, FALSE, 0)) {
895  min_space = initial_pitch;
896  gap_iqr = gap_stats.ile (0.75) - gap_stats.ile (0.25);
897  pitch_iqr = pitch_stats.ile (0.75) - pitch_stats.ile (0.25);
898  if (testing_on)
899  tprintf
900  ("Revised fp iteration:initial_pitch=%g, gap_iqr=%g, pitch_iqr=%g, pitch=%g\n",
901  initial_pitch, gap_iqr, pitch_iqr, pitch_stats.ile (0.5));
902  initial_pitch = pitch_stats.ile (0.5);
903  }
904  }
906  tprintf("Blk=%d:Row=%d:%c:p_iqr=%g:g_iqr=%g:dm_p_iqr=%g:dm_g_iqr=%g:%c:",
907  block_index, row_index, 'X',
908  pitch_iqr, gap_iqr, dm_pitch_iqr, dm_gap_iqr,
909  pitch_iqr > maxwidth && dm_pitch_iqr > maxwidth ? 'D' :
910  (pitch_iqr * dm_gap_iqr <= dm_pitch_iqr * gap_iqr ? 'S' : 'M'));
911  if (pitch_iqr > maxwidth && dm_pitch_iqr > maxwidth) {
914  tprintf ("\n");
915  return FALSE; //insufficient data
916  }
917  if (pitch_iqr * dm_gap_iqr <= dm_pitch_iqr * gap_iqr) {
918  if (testing_on)
919  tprintf
920  ("Choosing non dm version:pitch_iqr=%g, gap_iqr=%g, dm_pitch_iqr=%g, dm_gap_iqr=%g\n",
921  pitch_iqr, gap_iqr, dm_pitch_iqr, dm_gap_iqr);
922  gap_iqr = gap_stats.ile (0.75) - gap_stats.ile (0.25);
923  pitch_iqr = pitch_stats.ile (0.75) - pitch_stats.ile (0.25);
924  pitch = pitch_stats.ile (0.5);
925  used_dm_model = FALSE;
926  }
927  else {
928  if (testing_on)
929  tprintf
930  ("Choosing dm version:pitch_iqr=%g, gap_iqr=%g, dm_pitch_iqr=%g, dm_gap_iqr=%g\n",
931  pitch_iqr, gap_iqr, dm_pitch_iqr, dm_gap_iqr);
932  gap_iqr = dm_gap_iqr;
933  pitch_iqr = dm_pitch_iqr;
934  pitch = dm_pitch;
935  used_dm_model = TRUE;
936  }
938  tprintf ("rev_p_iqr=%g:rev_g_iqr=%g:pitch=%g:",
939  pitch_iqr, gap_iqr, pitch);
940  tprintf ("p_iqr/g=%g:p_iqr/x=%g:iqr_res=%c:",
941  pitch_iqr / gap_iqr, pitch_iqr / block->xheight,
942  pitch_iqr < gap_iqr * textord_fpiqr_ratio
943  && pitch_iqr < block->xheight * textord_max_pitch_iqr
944  && pitch < block->xheight * textord_words_default_maxspace
945  ? 'F' : 'P');
946  }
947  if (pitch_iqr < gap_iqr * textord_fpiqr_ratio
948  && pitch_iqr < block->xheight * textord_max_pitch_iqr
949  && pitch < block->xheight * textord_words_default_maxspace)
951  else
953  row->fixed_pitch = pitch;
954  row->kern_size = gap_stats.ile (0.5);
955  row->min_space = (inT32) (row->fixed_pitch + non_space) / 2;
956  if (row->min_space > row->fixed_pitch)
957  row->min_space = (inT32) row->fixed_pitch;
958  row->max_nonspace = row->min_space;
959  row->space_size = row->fixed_pitch;
960  row->space_threshold = (row->max_nonspace + row->min_space) / 2;
961  row->used_dm_model = used_dm_model;
962  return TRUE;
963 }
#define TRUE
Definition: capi.h:45
inT32 space_threshold
Definition: blobbox.h:661
EXTERN double textord_fpiqr_ratio
Definition: tovars.cpp:81
EXTERN double textord_max_pitch_iqr
Definition: tovars.cpp:82
PITCH_TYPE pitch_decision
Definition: blobbox.h:646
EXTERN bool textord_debug_pitch_metric
Definition: topitch.cpp:50
unsigned char BOOL8
Definition: host.h:46
inT32 min_space
Definition: blobbox.h:659
#define FALSE
Definition: capi.h:46
float xheight
Definition: blobbox.h:784
BOOL8 count_pitch_stats(TO_ROW *row, STATS *gap_stats, STATS *pitch_stats, float initial_pitch, float min_space, BOOL8 ignore_outsize, BOOL8 split_outsize, inT32 dm_gap)
Definition: topitch.cpp:1058
inT32 max_nonspace
Definition: blobbox.h:660
int inT32
Definition: host.h:35
float space_size
Definition: blobbox.h:663
#define tprintf(...)
Definition: tprintf.h:31
float xheight
Definition: blobbox.h:653
float fp_space
Definition: blobbox.h:648
float fixed_pitch
Definition: blobbox.h:647
EXTERN double words_default_fixed_limit
Definition: tovars.cpp:74
EXTERN double textord_words_default_maxspace
Definition: tovars.cpp:46
float fp_nonsp
Definition: blobbox.h:649
Definition: statistc.h:33
float kern_size
Definition: blobbox.h:662
BOOL8 used_dm_model
Definition: blobbox.h:643

◆ fix_row_pitch()

void fix_row_pitch ( TO_ROW bad_row,
TO_BLOCK bad_block,
TO_BLOCK_LIST *  blocks,
inT32  row_target,
inT32  block_target 
)

Definition at line 142 of file topitch.cpp.

146  { // number of block
147  inT16 mid_cuts;
148  int block_votes; //votes in block
149  int like_votes; //votes over page
150  int other_votes; //votes of unlike blocks
151  int block_index; //number of block
152  int row_index; //number of row
153  int maxwidth; //max pitch
154  TO_BLOCK_IT block_it = blocks; //block iterator
155  TO_ROW_IT row_it;
156  TO_BLOCK *block; //current block
157  TO_ROW *row; //current row
158  float sp_sd; //space deviation
159  STATS block_stats; //pitches in block
160  STATS like_stats; //pitches in page
161 
162  block_votes = like_votes = other_votes = 0;
163  maxwidth = (inT32) ceil (bad_row->xheight * textord_words_maxspace);
164  if (bad_row->pitch_decision != PITCH_DEF_FIXED
165  && bad_row->pitch_decision != PITCH_DEF_PROP) {
166  block_stats.set_range (0, maxwidth);
167  like_stats.set_range (0, maxwidth);
168  block_index = 1;
169  for (block_it.mark_cycle_pt(); !block_it.cycled_list();
170  block_it.forward()) {
171  block = block_it.data();
172  POLY_BLOCK* pb = block->block->poly_block();
173  if (pb != NULL && !pb->IsText()) continue; // Non text doesn't exist!
174  row_index = 1;
175  row_it.set_to_list (block->get_rows ());
176  for (row_it.mark_cycle_pt (); !row_it.cycled_list ();
177  row_it.forward ()) {
178  row = row_it.data ();
179  if ((bad_row->all_caps
180  && row->xheight + row->ascrise
181  <
182  (bad_row->xheight + bad_row->ascrise) * (1 +
184  && row->xheight + row->ascrise >
185  (bad_row->xheight + bad_row->ascrise) * (1 -
187  || (!bad_row->all_caps
188  && row->xheight <
189  bad_row->xheight * (1 + textord_pitch_rowsimilarity)
190  && row->xheight >
191  bad_row->xheight * (1 - textord_pitch_rowsimilarity))) {
192  if (block_index == block_target) {
193  if (row->pitch_decision == PITCH_DEF_FIXED) {
194  block_votes += textord_words_veto_power;
195  block_stats.add ((inT32) row->fixed_pitch,
197  }
198  else if (row->pitch_decision == PITCH_MAYBE_FIXED
199  || row->pitch_decision == PITCH_CORR_FIXED) {
200  block_votes++;
201  block_stats.add ((inT32) row->fixed_pitch, 1);
202  }
203  else if (row->pitch_decision == PITCH_DEF_PROP)
204  block_votes -= textord_words_veto_power;
205  else if (row->pitch_decision == PITCH_MAYBE_PROP
206  || row->pitch_decision == PITCH_CORR_PROP)
207  block_votes--;
208  }
209  else {
210  if (row->pitch_decision == PITCH_DEF_FIXED) {
211  like_votes += textord_words_veto_power;
212  like_stats.add ((inT32) row->fixed_pitch,
214  }
215  else if (row->pitch_decision == PITCH_MAYBE_FIXED
216  || row->pitch_decision == PITCH_CORR_FIXED) {
217  like_votes++;
218  like_stats.add ((inT32) row->fixed_pitch, 1);
219  }
220  else if (row->pitch_decision == PITCH_DEF_PROP)
221  like_votes -= textord_words_veto_power;
222  else if (row->pitch_decision == PITCH_MAYBE_PROP
223  || row->pitch_decision == PITCH_CORR_PROP)
224  like_votes--;
225  }
226  }
227  else {
228  if (row->pitch_decision == PITCH_DEF_FIXED)
229  other_votes += textord_words_veto_power;
230  else if (row->pitch_decision == PITCH_MAYBE_FIXED
231  || row->pitch_decision == PITCH_CORR_FIXED)
232  other_votes++;
233  else if (row->pitch_decision == PITCH_DEF_PROP)
234  other_votes -= textord_words_veto_power;
235  else if (row->pitch_decision == PITCH_MAYBE_PROP
236  || row->pitch_decision == PITCH_CORR_PROP)
237  other_votes--;
238  }
239  row_index++;
240  }
241  block_index++;
242  }
243  if (block_votes > textord_words_veto_power) {
244  bad_row->fixed_pitch = block_stats.ile (0.5);
245  bad_row->pitch_decision = PITCH_CORR_FIXED;
246  }
247  else if (block_votes <= textord_words_veto_power && like_votes > 0) {
248  bad_row->fixed_pitch = like_stats.ile (0.5);
249  bad_row->pitch_decision = PITCH_CORR_FIXED;
250  }
251  else {
252  bad_row->pitch_decision = PITCH_CORR_PROP;
253  if (block_votes == 0 && like_votes == 0 && other_votes > 0
255  tprintf
256  ("Warning:row %d of block %d set prop with no like rows against trend\n",
257  row_target, block_target);
258  }
259  }
261  tprintf(":b_votes=%d:l_votes=%d:o_votes=%d",
262  block_votes, like_votes, other_votes);
263  tprintf("x=%g:asc=%g\n", bad_row->xheight, bad_row->ascrise);
264  }
265  if (bad_row->pitch_decision == PITCH_CORR_FIXED) {
266  if (bad_row->fixed_pitch < textord_min_xheight) {
267  if (block_votes > 0)
268  bad_row->fixed_pitch = block_stats.ile (0.5);
269  else if (block_votes == 0 && like_votes > 0)
270  bad_row->fixed_pitch = like_stats.ile (0.5);
271  else {
272  tprintf
273  ("Warning:guessing pitch as xheight on row %d, block %d\n",
274  row_target, block_target);
275  bad_row->fixed_pitch = bad_row->xheight;
276  }
277  }
278  if (bad_row->fixed_pitch < textord_min_xheight)
279  bad_row->fixed_pitch = (float) textord_min_xheight;
280  bad_row->kern_size = bad_row->fixed_pitch / 4;
281  bad_row->min_space = (inT32) (bad_row->fixed_pitch * 0.6);
282  bad_row->max_nonspace = (inT32) (bad_row->fixed_pitch * 0.4);
283  bad_row->space_threshold =
284  (bad_row->min_space + bad_row->max_nonspace) / 2;
285  bad_row->space_size = bad_row->fixed_pitch;
286  if (bad_row->char_cells.empty() && !bad_row->blob_list()->empty()) {
287  tune_row_pitch (bad_row, &bad_row->projection,
288  bad_row->projection_left, bad_row->projection_right,
289  (bad_row->fixed_pitch +
290  bad_row->max_nonspace * 3) / 4, bad_row->fixed_pitch,
291  sp_sd, mid_cuts, &bad_row->char_cells, FALSE);
292  }
293  }
294  else if (bad_row->pitch_decision == PITCH_CORR_PROP
295  || bad_row->pitch_decision == PITCH_DEF_PROP) {
296  bad_row->fixed_pitch = 0.0f;
297  bad_row->char_cells.clear ();
298  }
299 }
inT16 projection_right
Definition: blobbox.h:645
BOOL8 all_caps
Definition: blobbox.h:642
short inT16
Definition: host.h:33
int textord_min_xheight
Definition: makerow.cpp:69
inT32 space_threshold
Definition: blobbox.h:661
EXTERN double textord_pitch_rowsimilarity
Definition: tovars.cpp:67
EXTERN int textord_words_veto_power
Definition: tovars.cpp:65
PITCH_TYPE pitch_decision
Definition: blobbox.h:646
EXTERN bool textord_debug_pitch_metric
Definition: topitch.cpp:50
BLOCK * block
Definition: blobbox.h:773
void add(inT32 value, inT32 count)
Definition: statistc.cpp:101
BLOBNBOX_LIST * blob_list()
Definition: blobbox.h:595
POLY_BLOCK * poly_block() const
Definition: pdblock.h:55
inT32 min_space
Definition: blobbox.h:659
#define FALSE
Definition: capi.h:46
float ascrise
Definition: blobbox.h:655
inT32 max_nonspace
Definition: blobbox.h:660
float tune_row_pitch(TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float &initial_pitch, float &best_sp_sd, inT16 &best_mid_cuts, ICOORDELT_LIST *best_cells, BOOL8 testing_on)
Definition: topitch.cpp:1146
TO_ROW_LIST * get_rows()
Definition: blobbox.h:700
int inT32
Definition: host.h:35
ICOORDELT_LIST char_cells
Definition: blobbox.h:665
float space_size
Definition: blobbox.h:663
#define tprintf(...)
Definition: tprintf.h:31
float xheight
Definition: blobbox.h:653
bool IsText() const
Definition: polyblk.h:52
EXTERN double textord_words_maxspace
Definition: tovars.cpp:44
EXTERN bool textord_debug_pitch_test
Definition: topitch.cpp:44
float fixed_pitch
Definition: blobbox.h:647
double ile(double frac) const
Definition: statistc.cpp:174
inT16 projection_left
Definition: blobbox.h:644
STATS projection
Definition: blobbox.h:667
Definition: statistc.h:33
float kern_size
Definition: blobbox.h:662
bool set_range(inT32 min_bucket_value, inT32 max_bucket_value_plus_1)
Definition: statistc.cpp:62

◆ fixed_pitch_row()

BOOL8 fixed_pitch_row ( TO_ROW row,
BLOCK block,
inT32  block_index 
)

Definition at line 974 of file topitch.cpp.

977  {
978  const char *res_string; // pitch result
979  inT16 mid_cuts; // no of cheap cuts
980  float non_space; // gap size
981  float pitch_sd; // error on pitch
982  float sp_sd = 0.0f; // space sd
983 
984  non_space = row->fp_nonsp;
985  if (non_space > row->fixed_pitch)
986  non_space = row->fixed_pitch;
987  POLY_BLOCK* pb = block != NULL ? block->poly_block() : NULL;
988  if (textord_all_prop || (pb != NULL && !pb->IsText())) {
989  // Set the decision to definitely proportional.
990  pitch_sd = textord_words_def_prop * row->fixed_pitch;
992  } else {
993  pitch_sd = tune_row_pitch (row, &row->projection, row->projection_left,
994  row->projection_right,
995  (row->fixed_pitch + non_space * 3) / 4,
996  row->fixed_pitch, sp_sd, mid_cuts,
997  &row->char_cells,
998  block_index == textord_debug_block);
999  if (pitch_sd < textord_words_pitchsd_threshold * row->fixed_pitch
1000  && ((pitsync_linear_version & 3) < 3
1001  || ((pitsync_linear_version & 3) >= 3 && (row->used_dm_model
1002  || sp_sd > 20
1003  || (pitch_sd == 0 && sp_sd > 10))))) {
1004  if (pitch_sd < textord_words_def_fixed * row->fixed_pitch
1005  && !row->all_caps
1006  && ((pitsync_linear_version & 3) < 3 || sp_sd > 20))
1008  else
1010  }
1011  else if ((pitsync_linear_version & 3) < 3
1012  || sp_sd > 20
1013  || mid_cuts > 0
1014  || pitch_sd >= textord_words_pitchsd_threshold * row->fixed_pitch) {
1015  if (pitch_sd < textord_words_def_prop * row->fixed_pitch)
1017  else
1019  }
1020  else
1021  row->pitch_decision = PITCH_DUNNO;
1022  }
1023 
1025  res_string = "??";
1026  switch (row->pitch_decision) {
1027  case PITCH_DEF_PROP:
1028  res_string = "DP";
1029  break;
1030  case PITCH_MAYBE_PROP:
1031  res_string = "MP";
1032  break;
1033  case PITCH_DEF_FIXED:
1034  res_string = "DF";
1035  break;
1036  case PITCH_MAYBE_FIXED:
1037  res_string = "MF";
1038  break;
1039  default:
1040  res_string = "??";
1041  }
1042  tprintf (":sd/p=%g:occ=%g:init_res=%s\n",
1043  pitch_sd / row->fixed_pitch, sp_sd, res_string);
1044  }
1045  return TRUE;
1046 }
inT16 projection_right
Definition: blobbox.h:645
BOOL8 all_caps
Definition: blobbox.h:642
#define TRUE
Definition: capi.h:45
short inT16
Definition: host.h:33
PITCH_TYPE pitch_decision
Definition: blobbox.h:646
EXTERN bool textord_debug_pitch_metric
Definition: topitch.cpp:50
EXTERN double textord_words_pitchsd_threshold
Definition: tovars.cpp:59
EXTERN int textord_debug_block
Definition: tovars.cpp:36
POLY_BLOCK * poly_block() const
Definition: pdblock.h:55
float tune_row_pitch(TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float &initial_pitch, float &best_sp_sd, inT16 &best_mid_cuts, ICOORDELT_LIST *best_cells, BOOL8 testing_on)
Definition: topitch.cpp:1146
ICOORDELT_LIST char_cells
Definition: blobbox.h:665
#define tprintf(...)
Definition: tprintf.h:31
bool IsText() const
Definition: polyblk.h:52
float fixed_pitch
Definition: blobbox.h:647
EXTERN bool textord_all_prop
Definition: topitch.cpp:42
inT16 projection_left
Definition: blobbox.h:644
STATS projection
Definition: blobbox.h:667
EXTERN double textord_words_def_prop
Definition: tovars.cpp:63
float fp_nonsp
Definition: blobbox.h:649
BOOL8 used_dm_model
Definition: blobbox.h:643

◆ plot_fp_word()

void plot_fp_word ( TO_BLOCK block,
float  pitch,
float  nonspace 
)

Definition at line 1822 of file topitch.cpp.

1826  {
1827  TO_ROW *row; //current row
1828  TO_ROW_IT row_it = block->get_rows ();
1829 
1830  for (row_it.mark_cycle_pt (); !row_it.cycled_list (); row_it.forward ()) {
1831  row = row_it.data ();
1832  row->min_space = (inT32) ((pitch + nonspace) / 2);
1833  row->max_nonspace = row->min_space;
1834  row->space_threshold = row->min_space;
1835  plot_word_decisions (to_win, (inT16) pitch, row);
1836  }
1837 }
short inT16
Definition: host.h:33
inT32 space_threshold
Definition: blobbox.h:661
void plot_word_decisions(ScrollView *win, inT16 pitch, TO_ROW *row)
Definition: drawtord.cpp:250
inT32 min_space
Definition: blobbox.h:659
inT32 max_nonspace
Definition: blobbox.h:660
TO_ROW_LIST * get_rows()
Definition: blobbox.h:700
int inT32
Definition: host.h:35
EXTERN ScrollView * to_win
Definition: drawtord.cpp:38

◆ print_block_counts()

void print_block_counts ( TO_BLOCK block,
inT32  block_index 
)

Definition at line 617 of file topitch.cpp.

620  {
621  inT32 def_fixed = 0; //counters
622  inT32 def_prop = 0;
623  inT32 maybe_fixed = 0;
624  inT32 maybe_prop = 0;
625  inT32 dunno = 0;
626  inT32 corr_fixed = 0;
627  inT32 corr_prop = 0;
628 
629  count_block_votes(block,
630  def_fixed,
631  def_prop,
632  maybe_fixed,
633  maybe_prop,
634  corr_fixed,
635  corr_prop,
636  dunno);
637  tprintf ("Block %d has (%d,%d,%d)",
638  block_index, def_fixed, maybe_fixed, corr_fixed);
639  if (textord_blocksall_prop && (def_fixed || maybe_fixed || corr_fixed))
640  tprintf (" (Wrongly)");
641  tprintf (" fixed, (%d,%d,%d)", def_prop, maybe_prop, corr_prop);
642  if (textord_blocksall_fixed && (def_prop || maybe_prop || corr_prop))
643  tprintf (" (Wrongly)");
644  tprintf (" prop, %d dunno\n", dunno);
645 }
EXTERN bool textord_blocksall_prop
Definition: tovars.cpp:31
void count_block_votes(TO_BLOCK *block, inT32 &def_fixed, inT32 &def_prop, inT32 &maybe_fixed, inT32 &maybe_prop, inT32 &corr_fixed, inT32 &corr_prop, inT32 &dunno)
Definition: topitch.cpp:654
int inT32
Definition: host.h:35
#define tprintf(...)
Definition: tprintf.h:31
EXTERN bool textord_blocksall_fixed
Definition: tovars.cpp:29

◆ print_pitch_sd()

void print_pitch_sd ( TO_ROW row,
STATS projection,
inT16  projection_left,
inT16  projection_right,
float  space_size,
float  initial_pitch 
)

Definition at line 1628 of file topitch.cpp.

1635  {
1636  const char *res2; //pitch result
1637  inT16 occupation; //used cells
1638  float sp_sd; //space sd
1639  //blobs
1640  BLOBNBOX_IT blob_it = row->blob_list ();
1641  BLOBNBOX_IT start_it; //start of word
1642  BLOBNBOX_IT row_start; //start of row
1643  inT16 blob_count; //no of blobs
1644  inT16 total_blob_count; //total blobs in line
1645  TBOX blob_box; //bounding box
1646  TBOX prev_box; //of super blob
1647  inT32 prev_right; //of word sync
1648  int scale_factor; //on scores for big words
1649  inT32 sp_count; //spaces
1650  FPSEGPT_LIST seg_list; //char cells
1651  FPSEGPT_IT seg_it; //iterator
1652  double sqsum; //sum of squares
1653  double spsum; //of spaces
1654  double sp_var; //space error
1655  double word_sync; //result for word
1656  double total_count; //total cuts
1657 
1658  if (blob_it.empty ())
1659  return;
1660  row_start = blob_it;
1661  total_blob_count = 0;
1662 
1663  total_count = 0;
1664  sqsum = 0;
1665  sp_count = 0;
1666  spsum = 0;
1667  prev_right = -1;
1668  blob_it = row_start;
1669  start_it = blob_it;
1670  blob_count = 0;
1671  blob_box = box_next (&blob_it);//first blob
1672  blob_it.mark_cycle_pt ();
1673  do {
1674  for (; blob_count > 0; blob_count--)
1675  box_next(&start_it);
1676  do {
1677  prev_box = blob_box;
1678  blob_count++;
1679  blob_box = box_next (&blob_it);
1680  }
1681  while (!blob_it.cycled_list ()
1682  && blob_box.left () - prev_box.right () < space_size);
1683  word_sync =
1684  check_pitch_sync2 (&start_it, blob_count, (inT16) initial_pitch, 2,
1685  projection, projection_left, projection_right,
1687  occupation, &seg_list, 0, 0);
1688  total_blob_count += blob_count;
1689  seg_it.set_to_list (&seg_list);
1690  if (prev_right >= 0) {
1691  sp_var = seg_it.data ()->position () - prev_right;
1692  sp_var -= floor (sp_var / initial_pitch + 0.5) * initial_pitch;
1693  sp_var *= sp_var;
1694  spsum += sp_var;
1695  sp_count++;
1696  }
1697  seg_it.move_to_last ();
1698  prev_right = seg_it.data ()->position ();
1700  scale_factor = (seg_list.length () - 2) / 2;
1701  if (scale_factor < 1)
1702  scale_factor = 1;
1703  }
1704  else
1705  scale_factor = 1;
1706  sqsum += word_sync * scale_factor;
1707  total_count += (seg_list.length () - 1) * scale_factor;
1708  seg_list.clear ();
1709  }
1710  while (!blob_it.cycled_list ());
1711  sp_sd = sp_count > 0 ? sqrt (spsum / sp_count) : 0;
1712  word_sync = total_count > 0 ? sqrt (sqsum / total_count) : space_size * 10;
1713  tprintf ("new_sd=%g:sd/p=%g:new_sp_sd=%g:res=%c:",
1714  word_sync, word_sync / initial_pitch, sp_sd,
1715  word_sync < textord_words_pitchsd_threshold * initial_pitch
1716  ? 'F' : 'P');
1717 
1718  start_it = row_start;
1719  blob_it = row_start;
1720  word_sync =
1721  check_pitch_sync2 (&blob_it, total_blob_count, (inT16) initial_pitch, 2,
1722  projection, projection_left, projection_right,
1723  row->xheight * textord_projection_scale, occupation,
1724  &seg_list, 0, 0);
1725  if (occupation > 1)
1726  word_sync /= occupation;
1727  word_sync = sqrt (word_sync);
1728 
1729 #ifndef GRAPHICS_DISABLED
1730  if (textord_show_row_cuts && to_win != NULL)
1731  plot_fp_cells2(to_win, ScrollView::CORAL, row, &seg_list);
1732 #endif
1733  seg_list.clear ();
1734  if (word_sync < textord_words_pitchsd_threshold * initial_pitch) {
1735  if (word_sync < textord_words_def_fixed * initial_pitch
1736  && !row->all_caps)
1737  res2 = "DF";
1738  else
1739  res2 = "MF";
1740  }
1741  else
1742  res2 = word_sync < textord_words_def_prop * initial_pitch ? "MP" : "DP";
1743  tprintf
1744  ("row_sd=%g:sd/p=%g:res=%c:N=%d:res2=%s,init pitch=%g, row_pitch=%g, all_caps=%d\n",
1745  word_sync, word_sync / initial_pitch,
1746  word_sync < textord_words_pitchsd_threshold * initial_pitch ? 'F' : 'P',
1747  occupation, res2, initial_pitch, row->fixed_pitch, row->all_caps);
1748 }
BOOL8 all_caps
Definition: blobbox.h:642
EXTERN double textord_projection_scale
Definition: topitch.cpp:57
short inT16
Definition: host.h:33
EXTERN bool textord_pitch_scalebigwords
Definition: tovars.cpp:69
EXTERN double textord_words_pitchsd_threshold
Definition: tovars.cpp:59
void plot_fp_cells2(ScrollView *win, ScrollView::Color colour, TO_ROW *row, FPSEGPT_LIST *seg_list)
Definition: drawtord.cpp:364
BLOBNBOX_LIST * blob_list()
Definition: blobbox.h:595
TBOX box_next(BLOBNBOX_IT *it)
Definition: blobbox.cpp:631
inT16 left() const
Definition: rect.h:68
int inT32
Definition: host.h:35
double check_pitch_sync2(BLOBNBOX_IT *blob_it, inT16 blob_count, inT16 pitch, inT16 pitch_error, STATS *projection, inT16 projection_left, inT16 projection_right, float projection_scale, inT16 &occupation_count, FPSEGPT_LIST *seg_list, inT16 start, inT16 end)
Definition: pithsync.cpp:298
#define tprintf(...)
Definition: tprintf.h:31
float xheight
Definition: blobbox.h:653
EXTERN bool textord_show_row_cuts
Definition: topitch.cpp:51
float fixed_pitch
Definition: blobbox.h:647
Definition: rect.h:30
inT16 right() const
Definition: rect.h:75
EXTERN ScrollView * to_win
Definition: drawtord.cpp:38
EXTERN double textord_words_def_prop
Definition: tovars.cpp:63
EXTERN double textord_words_def_fixed
Definition: tovars.cpp:61

◆ row_pitch_stats()

BOOL8 row_pitch_stats ( TO_ROW row,
inT32  maxwidth,
BOOL8  testing_on 
)

Definition at line 701 of file topitch.cpp.

705  {
706  BLOBNBOX *blob; //current blob
707  int gap_index; //current gap
708  inT32 prev_x; //end of prev blob
709  inT32 cluster_count; //no of clusters
710  inT32 prev_count; //of clusters
711  inT32 smooth_factor; //for smoothing stats
712  TBOX blob_box; //bounding box
713  float lower, upper; //cluster thresholds
714  //gap sizes
715  float gaps[BLOCK_STATS_CLUSTERS];
716  //blobs
717  BLOBNBOX_IT blob_it = row->blob_list ();
718  STATS gap_stats (0, maxwidth);
719  STATS cluster_stats[BLOCK_STATS_CLUSTERS + 1];
720  //clusters
721 
722  smooth_factor =
724  if (!blob_it.empty ()) {
725  prev_x = blob_it.data ()->bounding_box ().right ();
726  blob_it.forward ();
727  while (!blob_it.at_first ()) {
728  blob = blob_it.data ();
729  if (!blob->joined_to_prev ()) {
730  blob_box = blob->bounding_box ();
731  if (blob_box.left () - prev_x < maxwidth)
732  gap_stats.add (blob_box.left () - prev_x, 1);
733  prev_x = blob_box.right ();
734  }
735  blob_it.forward ();
736  }
737  }
738  if (gap_stats.get_total () == 0) {
739  return FALSE;
740  }
741  cluster_count = 0;
742  lower = row->xheight * words_initial_lower;
743  upper = row->xheight * words_initial_upper;
744  gap_stats.smooth (smooth_factor);
745  do {
746  prev_count = cluster_count;
747  cluster_count = gap_stats.cluster (lower, upper,
749  BLOCK_STATS_CLUSTERS, cluster_stats);
750  }
751  while (cluster_count > prev_count && cluster_count < BLOCK_STATS_CLUSTERS);
752  if (cluster_count < 1) {
753  return FALSE;
754  }
755  for (gap_index = 0; gap_index < cluster_count; gap_index++)
756  gaps[gap_index] = cluster_stats[gap_index + 1].ile (0.5);
757  //get medians
758  if (testing_on) {
759  tprintf ("cluster_count=%d:", cluster_count);
760  for (gap_index = 0; gap_index < cluster_count; gap_index++)
761  tprintf (" %g(%d)", gaps[gap_index],
762  cluster_stats[gap_index + 1].get_total ());
763  tprintf ("\n");
764  }
765  qsort (gaps, cluster_count, sizeof (float), sort_floats);
766 
767  //Try to find proportional non-space and space for row.
768  lower = row->xheight * words_default_prop_nonspace;
769  upper = row->xheight * textord_words_min_minspace;
770  for (gap_index = 0; gap_index < cluster_count
771  && gaps[gap_index] < lower; gap_index++);
772  if (gap_index == 0) {
773  if (testing_on)
774  tprintf ("No clusters below nonspace threshold!!\n");
775  if (cluster_count > 1) {
776  row->pr_nonsp = gaps[0];
777  row->pr_space = gaps[1];
778  }
779  else {
780  row->pr_nonsp = lower;
781  row->pr_space = gaps[0];
782  }
783  }
784  else {
785  row->pr_nonsp = gaps[gap_index - 1];
786  while (gap_index < cluster_count && gaps[gap_index] < upper)
787  gap_index++;
788  if (gap_index == cluster_count) {
789  if (testing_on)
790  tprintf ("No clusters above nonspace threshold!!\n");
791  row->pr_space = lower * textord_spacesize_ratioprop;
792  }
793  else
794  row->pr_space = gaps[gap_index];
795  }
796 
797  //Now try to find the fixed pitch space and non-space.
798  upper = row->xheight * words_default_fixed_space;
799  for (gap_index = 0; gap_index < cluster_count
800  && gaps[gap_index] < upper; gap_index++);
801  if (gap_index == 0) {
802  if (testing_on)
803  tprintf ("No clusters below space threshold!!\n");
804  row->fp_nonsp = upper;
805  row->fp_space = gaps[0];
806  }
807  else {
808  row->fp_nonsp = gaps[gap_index - 1];
809  if (gap_index == cluster_count) {
810  if (testing_on)
811  tprintf ("No clusters above space threshold!!\n");
812  row->fp_space = row->xheight;
813  }
814  else
815  row->fp_space = gaps[gap_index];
816  }
817  if (testing_on) {
818  tprintf
819  ("Initial estimates:pr_nonsp=%g, pr_space=%g, fp_nonsp=%g, fp_space=%g\n",
820  row->pr_nonsp, row->pr_space, row->fp_nonsp, row->fp_space);
821  }
822  return TRUE; //computed some stats
823 }
const TBOX & bounding_box() const
Definition: blobbox.h:215
#define TRUE
Definition: capi.h:45
EXTERN double textord_words_min_minspace
Definition: tovars.cpp:49
float pr_space
Definition: blobbox.h:650
EXTERN double textord_spacesize_ratioprop
Definition: tovars.cpp:80
EXTERN double words_initial_upper
Definition: tovars.cpp:71
BLOBNBOX_LIST * blob_list()
Definition: blobbox.h:595
EXTERN double words_default_fixed_space
Definition: tovars.cpp:73
bool joined_to_prev() const
Definition: blobbox.h:241
#define FALSE
Definition: capi.h:46
int sort_floats(const void *arg1, const void *arg2)
Definition: helpers.h:97
inT16 left() const
Definition: rect.h:68
int inT32
Definition: host.h:35
#define tprintf(...)
Definition: tprintf.h:31
float xheight
Definition: blobbox.h:653
float fp_space
Definition: blobbox.h:648
#define BLOCK_STATS_CLUSTERS
Definition: topitch.cpp:62
EXTERN double words_initial_lower
Definition: tovars.cpp:70
Definition: rect.h:30
inT16 right() const
Definition: rect.h:75
float pr_nonsp
Definition: blobbox.h:651
float fp_nonsp
Definition: blobbox.h:649
EXTERN double textord_wordstats_smooth_factor
Definition: tovars.cpp:39
Definition: statistc.h:33
EXTERN double words_default_prop_nonspace
Definition: tovars.cpp:72

◆ try_block_fixed()

BOOL8 try_block_fixed ( TO_BLOCK block,
inT32  block_index 
)

Definition at line 535 of file topitch.cpp.

538  {
539  return FALSE;
540 }
#define FALSE
Definition: capi.h:46

◆ try_doc_fixed()

BOOL8 try_doc_fixed ( ICOORD  page_tr,
TO_BLOCK_LIST *  port_blocks,
float  gradient 
)

Definition at line 395 of file topitch.cpp.

399  {
400  inT16 master_x; //uniform shifts
401  inT16 pitch; //median pitch.
402  int x; //profile coord
403  int prop_blocks; //correct counts
404  int fixed_blocks;
405  int total_row_count; //total in page
406  //iterator
407  TO_BLOCK_IT block_it = port_blocks;
408  TO_BLOCK *block; //current block;
409  TO_ROW_IT row_it; //row iterator
410  TO_ROW *row; //current row
411  inT16 projection_left; //edges
412  inT16 projection_right;
413  inT16 row_left; //edges of row
414  inT16 row_right;
415  ICOORDELT_LIST *master_cells; //cells for page
416  float master_y; //uniform shifts
417  float shift_factor; //page skew correction
418  float row_shift; //shift for row
419  float final_pitch; //output pitch
420  float row_y; //baseline
421  STATS projection; //entire page
422  STATS pitches (0, MAX_ALLOWED_PITCH);
423  //for median
424  float sp_sd; //space sd
425  inT16 mid_cuts; //no of cheap cuts
426  float pitch_sd; //sync rating
427 
428  if (block_it.empty ()
429  // || block_it.data()==block_it.data_relative(1)
431  return FALSE;
432  shift_factor = gradient / (gradient * gradient + 1);
433  row_it.set_to_list (block_it.data ()->get_rows ());
434  master_x = row_it.data ()->projection_left;
435  master_y = row_it.data ()->baseline.y (master_x);
436  projection_left = MAX_INT16;
437  projection_right = -MAX_INT16;
438  prop_blocks = 0;
439  fixed_blocks = 0;
440  total_row_count = 0;
441 
442  for (block_it.mark_cycle_pt (); !block_it.cycled_list ();
443  block_it.forward ()) {
444  block = block_it.data ();
445  row_it.set_to_list (block->get_rows ());
446  for (row_it.mark_cycle_pt (); !row_it.cycled_list (); row_it.forward ()) {
447  row = row_it.data ();
448  total_row_count++;
449  if (row->fixed_pitch > 0)
450  pitches.add ((inT32) (row->fixed_pitch), 1);
451  //find median
452  row_y = row->baseline.y (master_x);
453  row_left =
454  (inT16) (row->projection_left -
455  shift_factor * (master_y - row_y));
456  row_right =
457  (inT16) (row->projection_right -
458  shift_factor * (master_y - row_y));
459  if (row_left < projection_left)
460  projection_left = row_left;
461  if (row_right > projection_right)
462  projection_right = row_right;
463  }
464  }
465  if (pitches.get_total () == 0)
466  return FALSE;
467  projection.set_range (projection_left, projection_right);
468 
469  for (block_it.mark_cycle_pt (); !block_it.cycled_list ();
470  block_it.forward ()) {
471  block = block_it.data ();
472  row_it.set_to_list (block->get_rows ());
473  for (row_it.mark_cycle_pt (); !row_it.cycled_list (); row_it.forward ()) {
474  row = row_it.data ();
475  row_y = row->baseline.y (master_x);
476  row_left =
477  (inT16) (row->projection_left -
478  shift_factor * (master_y - row_y));
479  for (x = row->projection_left; x < row->projection_right;
480  x++, row_left++) {
481  projection.add (row_left, row->projection.pile_count (x));
482  }
483  }
484  }
485 
486  row_it.set_to_list (block_it.data ()->get_rows ());
487  row = row_it.data ();
488 #ifndef GRAPHICS_DISABLED
489  if (textord_show_page_cuts && to_win != NULL)
490  projection.plot (to_win, projection_left,
491  row->intercept (), 1.0f, -1.0f, ScrollView::CORAL);
492 #endif
493  final_pitch = pitches.ile (0.5);
494  pitch = (inT16) final_pitch;
495  pitch_sd =
496  tune_row_pitch (row, &projection, projection_left, projection_right,
497  pitch * 0.75, final_pitch, sp_sd, mid_cuts,
498  &row->char_cells, FALSE);
499 
501  tprintf
502  ("try_doc:props=%d:fixed=%d:pitch=%d:final_pitch=%g:pitch_sd=%g:sp_sd=%g:sd/trc=%g:sd/p=%g:sd/trc/p=%g\n",
503  prop_blocks, fixed_blocks, pitch, final_pitch, pitch_sd, sp_sd,
504  pitch_sd / total_row_count, pitch_sd / pitch,
505  pitch_sd / total_row_count / pitch);
506 
507 #ifndef GRAPHICS_DISABLED
508  if (textord_show_page_cuts && to_win != NULL) {
509  master_cells = &row->char_cells;
510  for (block_it.mark_cycle_pt (); !block_it.cycled_list ();
511  block_it.forward ()) {
512  block = block_it.data ();
513  row_it.set_to_list (block->get_rows ());
514  for (row_it.mark_cycle_pt (); !row_it.cycled_list ();
515  row_it.forward ()) {
516  row = row_it.data ();
517  row_y = row->baseline.y (master_x);
518  row_shift = shift_factor * (master_y - row_y);
519  plot_row_cells(to_win, ScrollView::GOLDENROD, row, row_shift, master_cells);
520  }
521  }
522  }
523 #endif
524  row->char_cells.clear ();
525  return FALSE;
526 }
inT16 projection_right
Definition: blobbox.h:645
short inT16
Definition: host.h:33
EXTERN bool textord_show_page_cuts
Definition: topitch.cpp:52
#define MAX_ALLOWED_PITCH
Definition: topitch.cpp:63
EXTERN bool textord_debug_pitch_metric
Definition: topitch.cpp:50
void plot_row_cells(ScrollView *win, ScrollView::Color colour, TO_ROW *row, float xshift, ICOORDELT_LIST *cells)
Definition: drawtord.cpp:397
void add(inT32 value, inT32 count)
Definition: statistc.cpp:101
double y(double x) const
Definition: quspline.cpp:217
#define MAX_INT16
Definition: host.h:52
#define FALSE
Definition: capi.h:46
float tune_row_pitch(TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float &initial_pitch, float &best_sp_sd, inT16 &best_mid_cuts, ICOORDELT_LIST *best_cells, BOOL8 testing_on)
Definition: topitch.cpp:1146
TO_ROW_LIST * get_rows()
Definition: blobbox.h:700
int inT32
Definition: host.h:35
QSPLINE baseline
Definition: blobbox.h:666
ICOORDELT_LIST char_cells
Definition: blobbox.h:665
#define tprintf(...)
Definition: tprintf.h:31
float fixed_pitch
Definition: blobbox.h:647
inT32 pile_count(inT32 value) const
Definition: statistc.h:78
inT16 projection_left
Definition: blobbox.h:644
EXTERN bool textord_blockndoc_fixed
Definition: topitch.cpp:56
STATS projection
Definition: blobbox.h:667
EXTERN ScrollView * to_win
Definition: drawtord.cpp:38
Definition: statistc.h:33
void plot(ScrollView *window, float xorigin, float yorigin, float xscale, float yscale, ScrollView::Color colour) const
Definition: statistc.cpp:585
bool set_range(inT32 min_bucket_value, inT32 max_bucket_value_plus_1)
Definition: statistc.cpp:62
float intercept() const
Definition: blobbox.h:584

◆ try_rows_fixed()

BOOL8 try_rows_fixed ( TO_BLOCK block,
inT32  block_index,
BOOL8  testing_on 
)

Definition at line 549 of file topitch.cpp.

553  {
554  TO_ROW *row; //current row
555  inT32 row_index; //row number.
556  inT32 def_fixed = 0; //counters
557  inT32 def_prop = 0;
558  inT32 maybe_fixed = 0;
559  inT32 maybe_prop = 0;
560  inT32 dunno = 0;
561  inT32 corr_fixed = 0;
562  inT32 corr_prop = 0;
563  float lower, upper; //cluster thresholds
564  TO_ROW_IT row_it = block->get_rows ();
565 
566  row_index = 1;
567  for (row_it.mark_cycle_pt (); !row_it.cycled_list (); row_it.forward ()) {
568  row = row_it.data ();
569  ASSERT_HOST (row->xheight > 0);
570  if (row->fixed_pitch > 0 &&
571  fixed_pitch_row(row, block->block, block_index)) {
572  if (row->fixed_pitch == 0) {
573  lower = row->pr_nonsp;
574  upper = row->pr_space;
575  row->space_size = upper;
576  row->kern_size = lower;
577  }
578  }
579  row_index++;
580  }
581  count_block_votes(block,
582  def_fixed,
583  def_prop,
584  maybe_fixed,
585  maybe_prop,
586  corr_fixed,
587  corr_prop,
588  dunno);
589  if (testing_on
592  tprintf ("Initially:");
593  print_block_counts(block, block_index);
594  }
595  if (def_fixed > def_prop * textord_words_veto_power)
597  else if (def_prop > def_fixed * textord_words_veto_power)
599  else if (def_fixed > 0 || def_prop > 0)
600  block->pitch_decision = PITCH_DUNNO;
601  else if (maybe_fixed > maybe_prop * textord_words_veto_power)
603  else if (maybe_prop > maybe_fixed * textord_words_veto_power)
605  else
606  block->pitch_decision = PITCH_DUNNO;
607  return FALSE;
608 }
EXTERN bool textord_blocksall_prop
Definition: tovars.cpp:31
void count_block_votes(TO_BLOCK *block, inT32 &def_fixed, inT32 &def_prop, inT32 &maybe_fixed, inT32 &maybe_prop, inT32 &corr_fixed, inT32 &corr_prop, inT32 &dunno)
Definition: topitch.cpp:654
EXTERN int textord_words_veto_power
Definition: tovars.cpp:65
BOOL8 fixed_pitch_row(TO_ROW *row, BLOCK *block, inT32 block_index)
Definition: topitch.cpp:974
BLOCK * block
Definition: blobbox.h:773
void print_block_counts(TO_BLOCK *block, inT32 block_index)
Definition: topitch.cpp:617
float pr_space
Definition: blobbox.h:650
PITCH_TYPE pitch_decision
Definition: blobbox.h:774
#define FALSE
Definition: capi.h:46
TO_ROW_LIST * get_rows()
Definition: blobbox.h:700
int inT32
Definition: host.h:35
float space_size
Definition: blobbox.h:663
#define tprintf(...)
Definition: tprintf.h:31
float xheight
Definition: blobbox.h:653
EXTERN bool textord_debug_pitch_test
Definition: topitch.cpp:44
float fixed_pitch
Definition: blobbox.h:647
float pr_nonsp
Definition: blobbox.h:651
#define ASSERT_HOST(x)
Definition: errcode.h:84
float kern_size
Definition: blobbox.h:662
EXTERN bool textord_blocksall_fixed
Definition: tovars.cpp:29

◆ tune_row_pitch()

float tune_row_pitch ( TO_ROW row,
STATS projection,
inT16  projection_left,
inT16  projection_right,
float  space_size,
float &  initial_pitch,
float &  best_sp_sd,
inT16 best_mid_cuts,
ICOORDELT_LIST *  best_cells,
BOOL8  testing_on 
)

Definition at line 1146 of file topitch.cpp.

1157  {
1158  int pitch_delta; //offset pitch
1159  inT16 mid_cuts; //cheap cuts
1160  float pitch_sd; //current sd
1161  float best_sd; //best result
1162  float best_pitch; //pitch for best result
1163  float initial_sd; //starting error
1164  float sp_sd; //space sd
1165  ICOORDELT_LIST test_cells; //row cells
1166  ICOORDELT_IT best_it; //start of best list
1167 
1169  return tune_row_pitch2 (row, projection, projection_left,
1170  projection_right, space_size, initial_pitch,
1171  best_sp_sd,
1172  //space sd
1173  best_mid_cuts, best_cells, testing_on);
1175  best_sp_sd = initial_pitch;
1176  return initial_pitch;
1177  }
1178  initial_sd =
1179  compute_pitch_sd(row,
1180  projection,
1181  projection_left,
1182  projection_right,
1183  space_size,
1184  initial_pitch,
1185  best_sp_sd,
1186  best_mid_cuts,
1187  best_cells,
1188  testing_on);
1189  best_sd = initial_sd;
1190  best_pitch = initial_pitch;
1191  if (testing_on)
1192  tprintf ("tune_row_pitch:start pitch=%g, sd=%g\n", best_pitch, best_sd);
1193  for (pitch_delta = 1; pitch_delta <= textord_pitch_range; pitch_delta++) {
1194  pitch_sd =
1195  compute_pitch_sd (row, projection, projection_left, projection_right,
1196  space_size, initial_pitch + pitch_delta, sp_sd,
1197  mid_cuts, &test_cells, testing_on);
1198  if (testing_on)
1199  tprintf ("testing pitch at %g, sd=%g\n", initial_pitch + pitch_delta,
1200  pitch_sd);
1201  if (pitch_sd < best_sd) {
1202  best_sd = pitch_sd;
1203  best_mid_cuts = mid_cuts;
1204  best_sp_sd = sp_sd;
1205  best_pitch = initial_pitch + pitch_delta;
1206  best_cells->clear ();
1207  best_it.set_to_list (best_cells);
1208  best_it.add_list_after (&test_cells);
1209  }
1210  else
1211  test_cells.clear ();
1212  if (pitch_sd > initial_sd)
1213  break; //getting worse
1214  }
1215  for (pitch_delta = 1; pitch_delta <= textord_pitch_range; pitch_delta++) {
1216  pitch_sd =
1217  compute_pitch_sd (row, projection, projection_left, projection_right,
1218  space_size, initial_pitch - pitch_delta, sp_sd,
1219  mid_cuts, &test_cells, testing_on);
1220  if (testing_on)
1221  tprintf ("testing pitch at %g, sd=%g\n", initial_pitch - pitch_delta,
1222  pitch_sd);
1223  if (pitch_sd < best_sd) {
1224  best_sd = pitch_sd;
1225  best_mid_cuts = mid_cuts;
1226  best_sp_sd = sp_sd;
1227  best_pitch = initial_pitch - pitch_delta;
1228  best_cells->clear ();
1229  best_it.set_to_list (best_cells);
1230  best_it.add_list_after (&test_cells);
1231  }
1232  else
1233  test_cells.clear ();
1234  if (pitch_sd > initial_sd)
1235  break;
1236  }
1237  initial_pitch = best_pitch;
1238 
1240  print_pitch_sd(row,
1241  projection,
1242  projection_left,
1243  projection_right,
1244  space_size,
1245  best_pitch);
1246 
1247  return best_sd;
1248 }
short inT16
Definition: host.h:33
EXTERN int textord_pitch_range
Definition: tovars.cpp:37
EXTERN bool textord_debug_pitch_metric
Definition: topitch.cpp:50
EXTERN bool textord_disable_pitch_test
Definition: topitch.cpp:46
void print_pitch_sd(TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float initial_pitch)
Definition: topitch.cpp:1628
#define tprintf(...)
Definition: tprintf.h:31
float tune_row_pitch2(TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float &initial_pitch, float &best_sp_sd, inT16 &best_mid_cuts, ICOORDELT_LIST *best_cells, BOOL8 testing_on)
Definition: topitch.cpp:1258
float compute_pitch_sd(TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float initial_pitch, float &sp_sd, inT16 &mid_cuts, ICOORDELT_LIST *row_cells, BOOL8 testing_on, inT16 start, inT16 end)
Definition: topitch.cpp:1373
EXTERN bool textord_fast_pitch_test
Definition: topitch.cpp:48

◆ tune_row_pitch2()

float tune_row_pitch2 ( TO_ROW row,
STATS projection,
inT16  projection_left,
inT16  projection_right,
float  space_size,
float &  initial_pitch,
float &  best_sp_sd,
inT16 best_mid_cuts,
ICOORDELT_LIST *  best_cells,
BOOL8  testing_on 
)

Definition at line 1258 of file topitch.cpp.

1269  {
1270  int pitch_delta; //offset pitch
1271  inT16 pixel; //pixel coord
1272  inT16 best_pixel; //pixel coord
1273  inT16 best_delta; //best pitch
1274  inT16 best_pitch; //best pitch
1275  inT16 start; //of good range
1276  inT16 end; //of good range
1277  inT32 best_count; //lowest sum
1278  float best_sd; //best result
1279  STATS *sum_proj; //summed projection
1280 
1281  best_sp_sd = initial_pitch;
1282 
1283  best_pitch = static_cast<int>(initial_pitch);
1284  if (textord_disable_pitch_test || best_pitch <= textord_pitch_range) {
1285  return initial_pitch;
1286  }
1287  sum_proj = new STATS[textord_pitch_range * 2 + 1];
1288 
1289  for (pitch_delta = -textord_pitch_range; pitch_delta <= textord_pitch_range;
1290  pitch_delta++)
1291  sum_proj[textord_pitch_range + pitch_delta].set_range (0,
1292  best_pitch +
1293  pitch_delta + 1);
1294  for (pixel = projection_left; pixel <= projection_right; pixel++) {
1295  for (pitch_delta = -textord_pitch_range; pitch_delta <= textord_pitch_range;
1296  pitch_delta++) {
1297  sum_proj[textord_pitch_range + pitch_delta].add(
1298  (pixel - projection_left) % (best_pitch + pitch_delta),
1299  projection->pile_count(pixel));
1300  }
1301  }
1302  best_count = sum_proj[textord_pitch_range].pile_count (0);
1303  best_delta = 0;
1304  best_pixel = 0;
1305  for (pitch_delta = -textord_pitch_range; pitch_delta <= textord_pitch_range;
1306  pitch_delta++) {
1307  for (pixel = 0; pixel < best_pitch + pitch_delta; pixel++) {
1308  if (sum_proj[textord_pitch_range + pitch_delta].pile_count (pixel)
1309  < best_count) {
1310  best_count =
1311  sum_proj[textord_pitch_range +
1312  pitch_delta].pile_count (pixel);
1313  best_delta = pitch_delta;
1314  best_pixel = pixel;
1315  }
1316  }
1317  }
1318  if (testing_on)
1319  tprintf ("tune_row_pitch:start pitch=%g, best_delta=%d, count=%d\n",
1320  initial_pitch, best_delta, best_count);
1321  best_pitch += best_delta;
1322  initial_pitch = best_pitch;
1323  best_count++;
1324  best_count += best_count;
1325  for (start = best_pixel - 2; start > best_pixel - best_pitch
1326  && sum_proj[textord_pitch_range +
1327  best_delta].pile_count (start % best_pitch) <= best_count;
1328  start--);
1329  for (end = best_pixel + 2;
1330  end < best_pixel + best_pitch
1331  && sum_proj[textord_pitch_range +
1332  best_delta].pile_count (end % best_pitch) <= best_count;
1333  end++);
1334 
1335  best_sd =
1336  compute_pitch_sd(row,
1337  projection,
1338  projection_left,
1339  projection_right,
1340  space_size,
1341  initial_pitch,
1342  best_sp_sd,
1343  best_mid_cuts,
1344  best_cells,
1345  testing_on,
1346  start,
1347  end);
1348  if (testing_on)
1349  tprintf ("tune_row_pitch:output pitch=%g, sd=%g\n", initial_pitch,
1350  best_sd);
1351 
1353  print_pitch_sd(row,
1354  projection,
1355  projection_left,
1356  projection_right,
1357  space_size,
1358  initial_pitch);
1359 
1360  delete[]sum_proj;
1361 
1362  return best_sd;
1363 }
short inT16
Definition: host.h:33
EXTERN int textord_pitch_range
Definition: tovars.cpp:37
EXTERN bool textord_debug_pitch_metric
Definition: topitch.cpp:50
void add(inT32 value, inT32 count)
Definition: statistc.cpp:101
EXTERN bool textord_disable_pitch_test
Definition: topitch.cpp:46
void print_pitch_sd(TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float initial_pitch)
Definition: topitch.cpp:1628
int inT32
Definition: host.h:35
#define tprintf(...)
Definition: tprintf.h:31
inT32 pile_count(inT32 value) const
Definition: statistc.h:78
Definition: statistc.h:33
float compute_pitch_sd(TO_ROW *row, STATS *projection, inT16 projection_left, inT16 projection_right, float space_size, float initial_pitch, float &sp_sd, inT16 &mid_cuts, ICOORDELT_LIST *row_cells, BOOL8 testing_on, inT16 start, inT16 end)
Definition: topitch.cpp:1373

Variable Documentation

◆ textord_balance_factor

double textord_balance_factor = 2.0

"Ding rate for unbalanced char cells"

Definition at line 59 of file topitch.cpp.

◆ textord_blockndoc_fixed

bool textord_blockndoc_fixed = TRUE

"Attempt whole doc/block fixed pitch"

Definition at line 56 of file topitch.cpp.

◆ textord_debug_pitch_metric

bool textord_debug_pitch_metric = FALSE

"Write full metric stuff"

Definition at line 50 of file topitch.cpp.

◆ textord_debug_pitch_test

bool textord_debug_pitch_test = FALSE

"Debug on fixed pitch test"

Definition at line 44 of file topitch.cpp.

◆ textord_fast_pitch_test

bool textord_fast_pitch_test = FALSE

"Do even faster pitch algorithm"

Definition at line 48 of file topitch.cpp.

◆ textord_pitch_cheat

bool textord_pitch_cheat = FALSE

"Use correct answer for fixed/prop"

Definition at line 54 of file topitch.cpp.

◆ textord_projection_scale

double textord_projection_scale = 0.125

"Ding rate for mid-cuts"

Definition at line 57 of file topitch.cpp.

◆ textord_show_page_cuts

bool textord_show_page_cuts = FALSE

"Draw page-level cuts"

Definition at line 52 of file topitch.cpp.

◆ textord_show_row_cuts

bool textord_show_row_cuts = FALSE

"Draw row-level cuts"

Definition at line 51 of file topitch.cpp.