mirror of
https://github.com/tesseract-ocr/tesseract.git
synced 2025-01-08 19:27:50 +08:00
177 lines
5.2 KiB
C++
177 lines
5.2 KiB
C++
|
/**********************************************************************
|
||
|
* File: normalis.cpp (Formerly denorm.c)
|
||
|
* Description: Code for the DENORM class.
|
||
|
* Author: Ray Smith
|
||
|
* Created: Thu Apr 23 09:22:43 BST 1992
|
||
|
*
|
||
|
* (C) Copyright 1992, Hewlett-Packard Ltd.
|
||
|
** 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.
|
||
|
*
|
||
|
**********************************************************************/
|
||
|
|
||
|
#include "mfcpch.h"
|
||
|
#include "werd.h"
|
||
|
#include "normalis.h"
|
||
|
|
||
|
/**********************************************************************
|
||
|
* DENORM::binary_search_segment
|
||
|
*
|
||
|
* Find the segment to use for the given x.
|
||
|
**********************************************************************/
|
||
|
|
||
|
const DENORM_SEG *DENORM::binary_search_segment(float src_x) const {
|
||
|
int bottom, top, middle; //binary search
|
||
|
|
||
|
bottom = 0;
|
||
|
top = segments;
|
||
|
do {
|
||
|
middle = (bottom + top) / 2;
|
||
|
if (segs[middle].xstart > src_x)
|
||
|
top = middle;
|
||
|
else
|
||
|
bottom = middle;
|
||
|
}
|
||
|
while (top - bottom > 1);
|
||
|
return &segs[bottom];
|
||
|
}
|
||
|
|
||
|
/**********************************************************************
|
||
|
* DENORM::scale_at_x
|
||
|
*
|
||
|
* Return scaling at a given (normalized) x coord.
|
||
|
**********************************************************************/
|
||
|
|
||
|
float DENORM::scale_at_x(float src_x) const { // In normalized coords.
|
||
|
if (segments != 0) {
|
||
|
const DENORM_SEG* seg = binary_search_segment(src_x);
|
||
|
if (seg->scale_factor > 0.0)
|
||
|
return seg->scale_factor;
|
||
|
}
|
||
|
return scale_factor;
|
||
|
}
|
||
|
|
||
|
/**********************************************************************
|
||
|
* DENORM::yshift_at_x
|
||
|
*
|
||
|
* Return yshift at a given (normalized) x coord.
|
||
|
**********************************************************************/
|
||
|
|
||
|
float DENORM::yshift_at_x(float src_x) const { // In normalized coords.
|
||
|
if (segments != 0) {
|
||
|
const DENORM_SEG* seg = binary_search_segment(src_x);
|
||
|
if (seg->ycoord == -MAX_INT32) {
|
||
|
if (base_is_row)
|
||
|
return source_row->base_line(x(src_x)/scale_at_x(src_x) + x_centre);
|
||
|
else
|
||
|
return m * x(src_x) + c;
|
||
|
} else {
|
||
|
return seg->ycoord;
|
||
|
}
|
||
|
}
|
||
|
return source_row->base_line (x(src_x)/scale_at_x(src_x) + x_centre);
|
||
|
}
|
||
|
|
||
|
/**********************************************************************
|
||
|
* DENORM::x
|
||
|
*
|
||
|
* Denormalise an x coordinate.
|
||
|
**********************************************************************/
|
||
|
|
||
|
float DENORM::x( //convert x coord
|
||
|
float src_x //coord to convert
|
||
|
) const {
|
||
|
return src_x / scale_at_x(src_x) + x_centre;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**********************************************************************
|
||
|
* DENORM::y
|
||
|
*
|
||
|
* Denormalise a y coordinate.
|
||
|
**********************************************************************/
|
||
|
|
||
|
float DENORM::y( //convert y coord
|
||
|
float src_y, //coord to convert
|
||
|
float src_centre //x location for base
|
||
|
) const {
|
||
|
return (src_y - bln_baseline_offset) / scale_at_x(src_centre)
|
||
|
+ yshift_at_x(src_centre);
|
||
|
}
|
||
|
|
||
|
|
||
|
DENORM::DENORM(float x, //from same pieces
|
||
|
float scaling,
|
||
|
double line_m, //default line
|
||
|
double line_c,
|
||
|
INT16 seg_count, //no of segments
|
||
|
DENORM_SEG *seg_pts, //actual segments
|
||
|
BOOL8 using_row, //as baseline
|
||
|
ROW *src) {
|
||
|
x_centre = x; //just copy
|
||
|
scale_factor = scaling;
|
||
|
source_row = src;
|
||
|
if (seg_count > 0) {
|
||
|
segs = new DENORM_SEG[seg_count];
|
||
|
for (segments = 0; segments < seg_count; segments++) {
|
||
|
// It is possible, if infrequent that the segments may be out of order.
|
||
|
// since we are searching with a binary search, keep them in order.
|
||
|
if (segments == 0 || segs[segments - 1].xstart <=
|
||
|
seg_pts[segments].xstart) {
|
||
|
segs[segments] = seg_pts[segments];
|
||
|
} else {
|
||
|
int i;
|
||
|
for (i = 0; i < segments
|
||
|
&& segs[segments - 1 - i].xstart > seg_pts[segments].xstart;
|
||
|
++i) {
|
||
|
segs[segments - i ] = segs[segments - 1 - i];
|
||
|
}
|
||
|
segs[segments - i] = seg_pts[segments];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
segments = 0;
|
||
|
segs = NULL;
|
||
|
}
|
||
|
base_is_row = using_row;
|
||
|
m = line_m;
|
||
|
c = line_c;
|
||
|
}
|
||
|
|
||
|
|
||
|
DENORM::DENORM(const DENORM &src) {
|
||
|
segments = 0;
|
||
|
segs = NULL;
|
||
|
*this = src;
|
||
|
}
|
||
|
|
||
|
|
||
|
DENORM & DENORM::operator= (const DENORM & src) {
|
||
|
x_centre = src.x_centre;
|
||
|
scale_factor = src.scale_factor;
|
||
|
source_row = src.source_row;
|
||
|
if (segments > 0)
|
||
|
delete[]segs;
|
||
|
if (src.segments > 0) {
|
||
|
segs = new DENORM_SEG[src.segments];
|
||
|
for (segments = 0; segments < src.segments; segments++)
|
||
|
segs[segments] = src.segs[segments];
|
||
|
}
|
||
|
else {
|
||
|
segments = 0;
|
||
|
segs = NULL;
|
||
|
}
|
||
|
base_is_row = src.base_is_row;
|
||
|
m = src.m;
|
||
|
c = src.c;
|
||
|
return *this;
|
||
|
}
|