File: segmentation.cc

package info (click to toggle)
exactimage 1.2.1-3
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 3,048 kB
  • sloc: cpp: 35,940; ansic: 1,952; xml: 1,447; makefile: 338; perl: 138; sh: 110; python: 45; php: 37; ruby: 12
file content (116 lines) | stat: -rw-r--r-- 3,155 bytes parent folder | download | duplicates (11)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
/*
 * Image Segmentation
 * Copyright (C) 2007 Valentin Ziegler and René Rebe
 * 
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; version 2. A copy of the GNU General
 * Public License can be found in the file LICENSE.
 * 
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANT-
 * ABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
 * Public License for more details.
 * 
 */

#include "vectorial.hh"
#include "segmentation.hh"

Segment::Segment(unsigned int ix, unsigned int iy, unsigned int iw, unsigned int ih, Segment* iparent)
{
  x=ix;
  y=iy;
  w=iw;
  h=ih;
  parent=iparent;
}


Segment::~Segment()
{
  for (unsigned int i=0; i<children.size(); i++)
    delete children[i];
}


bool Segment::Subdivide(const FGMatrix& img, double tolerance, unsigned int min_length, bool horizontal)
{
  unsigned int* counts=Count(img, horizontal);
  unsigned int end=horizontal ? h : w;
  unsigned int max_pixels=(unsigned int) (tolerance*(double)(horizontal ? w : h));

  unsigned int length=0;
  unsigned int last_start=0;
  for (unsigned int n=0; n<end; n++) {
    if (counts[n] <= max_pixels) {
      length++;
    }
    else {
      if (length >= min_length || length==n) {
	if (length < n)
	  InsertChild(last_start, n-length, horizontal);
	last_start=n;
      }
      length=0;
    }

  }
  if (last_start > 0)
    InsertChild(last_start, end-length, horizontal);
  
  delete[] counts;
  return (children.size() > 0);
}


void Segment::Draw(Image& output, uint16_t r, uint16_t g, uint16_t b)
{
  Path path;
  path.setFillColor ((double)r/255, (double)g/255, (double)b/255);
  path.addRect (x, y, x+w-1, y+h-1);
  path.draw (output);
}


void Segment::InsertChild(unsigned int start, unsigned int end, bool horizontal)
{
  if (horizontal)
    children.push_back(new Segment(x, y+start, w, end-start, this));
  else
    children.push_back(new Segment(x+start, y, end-start, h, this));
}

 
// count foreground pixels in horizontal/vertical lines
unsigned int* Segment::Count(const FGMatrix& img, bool horizontal)
{
  FGMatrix subimg(img, x, y, w, h);
  unsigned int* counts=new unsigned int[horizontal ? h : w];
  for (unsigned int n=0; n<(horizontal ? h : w) ; n++)
    counts[n]=0;

  for (unsigned int px=0; px<w; px++)
    for (unsigned int py=0; py<h; py++)
      if (subimg(px,py))
	counts[horizontal ? py : px]++;

  return counts;
}




void segment_recursion(Segment* s, const FGMatrix& img, double tolerance, unsigned int min_w, unsigned int min_h, bool horizontal)
{
  if (s->Subdivide(img, tolerance, horizontal ? min_h : min_w, horizontal))
    for (unsigned int i=0; i<s->children.size(); i++)
      segment_recursion(s->children[i], img, tolerance, min_w, min_h, !horizontal);
}

Segment* segment_image(const FGMatrix& img, double tolerance, unsigned int min_w, unsigned int min_h)
{
  Segment* top=new Segment(0, 0, img.w, img.h);
  segment_recursion(top, img, tolerance, min_w, min_h, true);
  return top;
}