mirror of
https://github.com/tesseract-ocr/tesseract.git
synced 2024-11-27 20:59:36 +08:00
4d514d5a60
git-svn-id: https://tesseract-ocr.googlecode.com/svn/trunk@878 d0cd1f9f-072b-0410-8dd7-cf729c803f20
237 lines
8.3 KiB
C++
237 lines
8.3 KiB
C++
/* -*-C-*-
|
|
********************************************************************************
|
|
*
|
|
* File: split.c (Formerly split.c)
|
|
* Description:
|
|
* Author: Mark Seaman, OCR Technology
|
|
* Created: Fri Oct 16 14:37:00 1987
|
|
* Modified: Fri May 17 16:27:49 1991 (Mark Seaman) marks@hpgrlt
|
|
* Language: C
|
|
* Package: N/A
|
|
* Status: Reusable Software Component
|
|
*
|
|
* (c) Copyright 1987, Hewlett-Packard Company.
|
|
** Licensed under the Apache License, Version 2.0 (the "License");
|
|
** you may not use this file except in compliance with the License.
|
|
** You may obtain a copy of the License at
|
|
** http://www.apache.org/licenses/LICENSE-2.0
|
|
** Unless required by applicable law or agreed to in writing, software
|
|
** distributed under the License is distributed on an "AS IS" BASIS,
|
|
** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
** See the License for the specific language governing permissions and
|
|
** limitations under the License.
|
|
*
|
|
*************************************************************************/
|
|
/*----------------------------------------------------------------------
|
|
I n c l u d e s
|
|
----------------------------------------------------------------------*/
|
|
#include "split.h"
|
|
#include "coutln.h"
|
|
#include "tprintf.h"
|
|
|
|
#ifdef __UNIX__
|
|
#include <assert.h>
|
|
#endif
|
|
|
|
/*----------------------------------------------------------------------
|
|
V a r i a b l e s
|
|
----------------------------------------------------------------------*/
|
|
BOOL_VAR(wordrec_display_splits, 0, "Display splits");
|
|
|
|
/*----------------------------------------------------------------------
|
|
F u n c t i o n s
|
|
----------------------------------------------------------------------*/
|
|
|
|
/**********************************************************************
|
|
* delete_split
|
|
*
|
|
* Remove this split from existence.
|
|
**********************************************************************/
|
|
void delete_split(SPLIT *split) {
|
|
if (split) {
|
|
delete split;
|
|
}
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* make_edgept
|
|
*
|
|
* Create an EDGEPT and hook it into an existing list of edge points.
|
|
**********************************************************************/
|
|
EDGEPT *make_edgept(int x, int y, EDGEPT *next, EDGEPT *prev) {
|
|
EDGEPT *this_edgept;
|
|
/* Create point */
|
|
this_edgept = new EDGEPT;
|
|
this_edgept->pos.x = x;
|
|
this_edgept->pos.y = y;
|
|
// Now deal with the src_outline steps.
|
|
C_OUTLINE* prev_ol = prev->src_outline;
|
|
if (prev_ol != NULL && prev->next == next) {
|
|
// Compute the fraction of the segment that is being cut.
|
|
FCOORD segment_vec(next->pos.x - prev->pos.x, next->pos.y - prev->pos.y);
|
|
FCOORD target_vec(x - prev->pos.x, y - prev->pos.y);
|
|
double cut_fraction = target_vec.length() / segment_vec.length();
|
|
// Get the start and end at the step level.
|
|
ICOORD step_start = prev_ol->position_at_index(prev->start_step);
|
|
int end_step = prev->start_step + prev->step_count;
|
|
int step_length = prev_ol->pathlength();
|
|
ICOORD step_end = prev_ol->position_at_index(end_step % step_length);
|
|
ICOORD step_vec = step_end - step_start;
|
|
double target_length = step_vec.length() * cut_fraction;
|
|
// Find the point on the segment that gives the length nearest to target.
|
|
int best_step = prev->start_step;
|
|
ICOORD total_step(0, 0);
|
|
double best_dist = target_length;
|
|
for (int s = prev->start_step; s < end_step; ++s) {
|
|
total_step += prev_ol->step(s % step_length);
|
|
double dist = fabs(target_length - total_step.length());
|
|
if (dist < best_dist) {
|
|
best_dist = dist;
|
|
best_step = s + 1;
|
|
}
|
|
}
|
|
// The new point is an intermediate point.
|
|
this_edgept->src_outline = prev_ol;
|
|
this_edgept->step_count = end_step - best_step;
|
|
this_edgept->start_step = best_step % step_length;
|
|
prev->step_count = best_step - prev->start_step;
|
|
} else {
|
|
// The new point is poly only.
|
|
this_edgept->src_outline = NULL;
|
|
this_edgept->step_count = 0;
|
|
this_edgept->start_step = 0;
|
|
}
|
|
/* Hook it up */
|
|
this_edgept->next = next;
|
|
this_edgept->prev = prev;
|
|
prev->next = this_edgept;
|
|
next->prev = this_edgept;
|
|
/* Set up vec entries */
|
|
this_edgept->vec.x = this_edgept->next->pos.x - x;
|
|
this_edgept->vec.y = this_edgept->next->pos.y - y;
|
|
this_edgept->prev->vec.x = x - this_edgept->prev->pos.x;
|
|
this_edgept->prev->vec.y = y - this_edgept->prev->pos.y;
|
|
return this_edgept;
|
|
}
|
|
|
|
/**********************************************************************
|
|
* remove_edgept
|
|
*
|
|
* Remove a given EDGEPT from its list and delete it.
|
|
**********************************************************************/
|
|
void remove_edgept(EDGEPT *point) {
|
|
EDGEPT *prev = point->prev;
|
|
EDGEPT *next = point->next;
|
|
// Add point's steps onto prev's steps if they are from the same outline.
|
|
if (prev->src_outline == point->src_outline && prev->src_outline != NULL) {
|
|
prev->step_count += point->step_count;
|
|
}
|
|
prev->next = next;
|
|
next->prev = prev;
|
|
prev->vec.x = next->pos.x - prev->pos.x;
|
|
prev->vec.y = next->pos.y - prev->pos.y;
|
|
delete point;
|
|
}
|
|
|
|
/**********************************************************************
|
|
* new_split
|
|
*
|
|
* Create a new split record and initialize it. Put it on the display
|
|
* list.
|
|
**********************************************************************/
|
|
SPLIT *new_split(EDGEPT *point1, EDGEPT *point2) {
|
|
SPLIT *s = new SPLIT;
|
|
s->point1 = point1;
|
|
s->point2 = point2;
|
|
return (s);
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* print_split
|
|
*
|
|
* Print a list of splits. Show the coordinates of both points in
|
|
* each split.
|
|
**********************************************************************/
|
|
void print_split(SPLIT *split) {
|
|
if (split) {
|
|
tprintf("(%d,%d)--(%d,%d)",
|
|
split->point1->pos.x, split->point1->pos.y,
|
|
split->point2->pos.x, split->point2->pos.y);
|
|
}
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* split_outline
|
|
*
|
|
* Split between these two edge points.
|
|
**********************************************************************/
|
|
void split_outline(EDGEPT *join_point1, EDGEPT *join_point2) {
|
|
assert(join_point1 != join_point2);
|
|
|
|
EDGEPT* temp2 = join_point2->next;
|
|
EDGEPT* temp1 = join_point1->next;
|
|
/* Create two new points */
|
|
EDGEPT* new_point1 = make_edgept(join_point1->pos.x, join_point1->pos.y,
|
|
temp1, join_point2);
|
|
EDGEPT* new_point2 = make_edgept(join_point2->pos.x, join_point2->pos.y,
|
|
temp2, join_point1);
|
|
// Join_point1 and 2 are now cross-over points, so they must have NULL
|
|
// src_outlines and give their src_outline information their new
|
|
// replacements.
|
|
new_point1->src_outline = join_point1->src_outline;
|
|
new_point1->start_step = join_point1->start_step;
|
|
new_point1->step_count = join_point1->step_count;
|
|
new_point2->src_outline = join_point2->src_outline;
|
|
new_point2->start_step = join_point2->start_step;
|
|
new_point2->step_count = join_point2->step_count;
|
|
join_point1->src_outline = NULL;
|
|
join_point1->start_step = 0;
|
|
join_point1->step_count = 0;
|
|
join_point2->src_outline = NULL;
|
|
join_point2->start_step = 0;
|
|
join_point2->step_count = 0;
|
|
join_point1->MarkChop();
|
|
join_point2->MarkChop();
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* unsplit_outlines
|
|
*
|
|
* Remove the split that was put between these two points.
|
|
**********************************************************************/
|
|
void unsplit_outlines(EDGEPT *p1, EDGEPT *p2) {
|
|
EDGEPT *tmp1 = p1->next;
|
|
EDGEPT *tmp2 = p2->next;
|
|
|
|
assert (p1 != p2);
|
|
|
|
tmp1->next->prev = p2;
|
|
tmp2->next->prev = p1;
|
|
|
|
// tmp2 is coincident with p1. p1 takes tmp2's place as tmp2 is deleted.
|
|
p1->next = tmp2->next;
|
|
p1->src_outline = tmp2->src_outline;
|
|
p1->start_step = tmp2->start_step;
|
|
p1->step_count = tmp2->step_count;
|
|
// Likewise p2 takes tmp1's place.
|
|
p2->next = tmp1->next;
|
|
p2->src_outline = tmp1->src_outline;
|
|
p2->start_step = tmp1->start_step;
|
|
p2->step_count = tmp1->step_count;
|
|
p1->UnmarkChop();
|
|
p2->UnmarkChop();
|
|
|
|
delete tmp1;
|
|
delete tmp2;
|
|
|
|
p1->vec.x = p1->next->pos.x - p1->pos.x;
|
|
p1->vec.y = p1->next->pos.y - p1->pos.y;
|
|
|
|
p2->vec.x = p2->next->pos.x - p2->pos.x;
|
|
p2->vec.y = p2->next->pos.y - p2->pos.y;
|
|
}
|