00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00047 #include <map>
00048 #include <vector>
00049
00050 #include <mln/util/ord.hh>
00051
00052 #include <mln/core/image/image2d.hh>
00053 #include <mln/core/alias/point2d.hh>
00054 #include <mln/core/alias/window2d.hh>
00055 #include <mln/core/alias/neighb2d.hh>
00056
00057 #include <mln/convert/to_window.hh>
00058
00060 #include <mln/core/image/edge_image.hh>
00061 #include <mln/core/var.hh>
00062 #include <mln/fun/i2v/array.hh>
00063 #include <mln/util/graph.hh>
00064
00065 #include <mln/morpho/gradient.hh>
00066 #include <mln/morpho/closing/area.hh>
00067 #include <mln/morpho/meyer_wst.hh>
00068
00069 #include <mln/value/int_u8.hh>
00070 #include <mln/value/rgb8.hh>
00071 #include <mln/literal/black.hh>
00072 #include <mln/literal/colors.hh>
00073
00074 #include <mln/io/pgm/load.hh>
00075 #include <mln/io/ppm/save.hh>
00076
00077 #include <mln/math/max.hh>
00078 #include <mln/math/abs.hh>
00079
00080 #include <mln/util/site_pair.hh>
00081
00082 #include "tests/data.hh"
00083
00084
00085
00086 int main()
00087 {
00088 using namespace mln;
00089 using value::int_u8;
00090 using value::rgb8;
00091
00092
00093
00094
00095
00096 typedef int_u8 input_val_t;
00097 image2d<input_val_t> input;
00098 io::pgm::load(input, MLN_IMG_DIR "/tiny.pgm");
00099
00100
00101
00102 image2d<input_val_t> gradient =
00103 morpho::gradient (input, convert::to_window(c4()));
00104
00105
00106 image2d<input_val_t> work(input.domain());
00107 work = morpho::closing::area(gradient, c4(), 10);
00108
00109
00110
00111
00112
00113
00114
00115 util::graph g;
00116
00117
00118 image2d<unsigned> equiv_vertex;
00119 initialize(equiv_vertex, work);
00120
00121
00122 mln_fwd_piter_(image2d<input_val_t>) p(work.domain());
00123 for_all(p)
00124 equiv_vertex(p) = g.add_vertex();
00125
00126
00127 window2d next_c4_win;
00128 next_c4_win.insert(0, 1).insert(1, 0);
00129 typedef fun::i2v::array<int> edge_values_t;
00130 typedef fun::i2v::array< util::site_pair<point2d> > edge_sites_t;
00131 edge_values_t edge_values;
00132 edge_sites_t edge_sites;
00133 mln_fwd_qiter_(window2d) q(next_c4_win, p);
00134 for_all (p)
00135 for_all(q)
00136 if (work.domain().has(q))
00137 {
00138 g.add_edge(equiv_vertex(p), equiv_vertex(q));
00139 edge_values.append(math::max(work(p), work(q)));
00140 edge_sites.append(util::site_pair<point2d>(p, q));
00141 }
00142
00143
00144 typedef edge_image<util::site_pair<point2d>,int,util::graph> lg_ima_t;
00145 lg_ima_t lg_ima(g, edge_sites, edge_values);
00146
00147
00148
00149
00150
00151 typedef lg_ima_t::nbh_t nbh_t;
00152 nbh_t nbh;
00153
00154
00155 int_u8 nbasins;
00156 typedef edge_image<util::site_pair<point2d>,int_u8,util::graph> wshed_t;
00157 wshed_t wshed = morpho::meyer_wst(lg_ima, nbh, nbasins);
00158 mln_assertion(nbasins == 5u);
00159
00160
00161
00162
00163
00164
00165
00166
00167
00168
00169 typedef rgb8 output_val_t;
00170 typedef image2d<output_val_t> output_t;
00171 point2d output_pmin = input.domain().pmin();
00172 point2d output_pmax(input.domain().pmax()[0] * 2,
00173 input.domain().pmax()[1] * 2);
00174 output_t output(box2d(output_pmin, output_pmax));
00175 data::fill(output, literal::black);
00176
00177 for_all(p)
00178 {
00179
00180 point2d q(p[0] * 2, p[1] * 2);
00181 input_val_t v = input(p);
00182
00183
00184 output(q) = output_val_t(v, v, v);
00185 }
00186
00187 mln_piter_(output_t) p_out(output.domain());
00188 for_all(p_out)
00189 {
00190
00191 if (p_out[0] % 2 == 0 && p_out[1] % 2 == 1)
00192 output(p_out) = (output(p_out + left) + output(p_out + right)) / 2;
00193
00194 if (p_out[0] % 2 == 1 && p_out[1] % 2 == 0)
00195 output(p_out) = (output(p_out + up) + output(p_out + down)) / 2;
00196
00197 if (p_out[0] % 2 == 1 && p_out[1] % 2 == 1)
00198 output(p_out) =
00199 (output(p_out + dpoint2d(-1, -1)) +
00200 output(p_out + dpoint2d(-1, +1)) +
00201 output(p_out + dpoint2d(+1, -1)) +
00202 output(p_out + dpoint2d(+1, +1))) / 4;
00203 }
00204
00205
00206
00207
00208 mln_piter_(wshed_t) pw(wshed.domain());
00209 for_all(pw)
00210 {
00211 if (wshed(pw) == 0u)
00212 {
00213 mln_psite_(lg_ima_t) pp(pw);
00214
00215 int row1 = pp.first()[0] * 2;
00216 int col1 = pp.first()[1] * 2;
00217 int row2 = pp.second()[0] * 2;
00218 int col2 = pp.second()[1] * 2;
00219 point2d q((row1 + row2) / 2, (col1 + col2) / 2);
00220
00221 output(q) = literal::red;
00222 }
00223 }
00224
00225
00226
00227
00228
00229
00230
00231
00232
00233 for_all (p_out)
00234
00235 if (p_out[0] % 2 == 1 && p_out[1] % 2 == 1)
00236 {
00237
00238
00239
00240
00241 unsigned nwsheds =
00242 (output.has(p_out + up ) && output(p_out + up ) == literal::red) +
00243 (output.has(p_out + down ) && output(p_out + down ) == literal::red) +
00244 (output.has(p_out + left ) && output(p_out + right) == literal::red) +
00245 (output.has(p_out + right) && output(p_out + left ) == literal::red);
00246 if (nwsheds >= 2)
00247 output(p_out) = literal::red;
00248 }
00249 io::ppm::save(output, "out.ppm");
00250 }