[GRASS-SVN] r32283 - grass/trunk/lib/vector/vedit
svn_grass at osgeo.org
svn_grass at osgeo.org
Fri Jul 25 06:23:18 EDT 2008
Author: martinl
Date: 2008-07-25 06:23:18 -0400 (Fri, 25 Jul 2008)
New Revision: 32283
Modified:
grass/trunk/lib/vector/vedit/break.c
Log:
veditlib: connect_lines() rewritten
Modified: grass/trunk/lib/vector/vedit/break.c
===================================================================
--- grass/trunk/lib/vector/vedit/break.c 2008-07-25 10:15:54 UTC (rev 32282)
+++ grass/trunk/lib/vector/vedit/break.c 2008-07-25 10:23:18 UTC (rev 32283)
@@ -1,29 +1,31 @@
/**
- \file break.c
+ \file vector/vedit/break.c
- \brief Vedit library - split/break lines
+ \brief Vedit library - split / break lines
- This program is free software under the
- GNU General Public License (>=v2).
- Read the file COPYING that comes with GRASS
- for details.
+ (C) 2007-2008 by the GRASS Development Team
- \author (C) 2007-2008 by the GRASS Development Team
- Martin Landa <landa.martin gmail.com>
+ This program is free software under the GNU General Public License
+ (>=v2). Read the file COPYING that comes with GRASS for details.
+ \author Martin Landa <landa.martin gmail.com>
+
\date 2007-2008
*/
#include <math.h>
#include <grass/vedit.h>
+static int connect_lines(struct Map_info *, int, int, int,
+ double, struct ilist *);
+
/**
\brief Split selected lines on given position
- \param[in] Map vector map
- \param[in] List list of selected lines
- \param[in] coord points location
- \param[in] List_updated list of rewritten features (or NULL)
+ \param Map vector map
+ \param List list of selected lines
+ \param coord points location
+ \param[out] List_updated list of rewritten features (or NULL)
\return number of modified lines
\return -1 on error
@@ -132,206 +134,193 @@
}
/**
- \brief Connect *two* lines
+ \brief Connect lines in given threshold
- The first line from the list is connected to the second one
- if necessary the second line is broken
-
- \ \
- id1 \ -> \
+ <pre>
+ \ \
+ id1 \ -> \
\
- id2 --------- ---------
-
- \param[in] Map vector map
- \param[in] List list of selected lines
- \param[in] thresh threshold value for connect
+ id2 --------- -----+---
+ </pre>
- \return 1 lines connected
- \return 0 lines not connected
+ \param Map vector map
+ \param List list of selected lines
+ \param thresh threshold value
+
+ \return number of modified lines
\return -1 on error
*/
int Vedit_connect_lines (struct Map_info *Map, struct ilist *List,
double thresh)
{
- int nlines_modified, newline, intersection, linep, nlines_to_modify;
- int i, idx, pnt_idx[2];
- int line[2], type[2], seg[2];
- double px[2], py[2], pz[2], dist[2], spdist[2], lpdist[2];
- double nx[2], ny[2], nz[2];
- char n_on_line[2], connected;
- double angle, dseg, dist_connect;
-
- struct ilist *List_break, *List_updated;
-
- struct line_pnts *Points[2];
- struct line_cats *Cats[2];
-
+ int nlines_modified;
+ int i, j, node[2], n_nodes;
+ int line, found;
+ double x, y, z;
+
nlines_modified = 0;
- nlines_to_modify = 2; /* modify *two* lines at the time */
- Points[0] = Points[1] = NULL;
- Cats[0] = Cats[1] = NULL;
+ /* collect lines to be modified */
+ for (i = 0; i < List->n_values; i++) {
+ line = List->value[i];
- List_break = Vect_new_list ();
- List_updated = Vect_new_list();
+ if (!Vect_line_alive(Map, line))
+ continue;
- for (linep = 0; linep < List -> n_values; linep += 2) {
- connected = 0;
- /* read the lines */
- for (i = 0; i < nlines_to_modify; i++) {
- line[i] = List -> value[linep + i];
+ node[0] = node[1] = -1;
- if (!Vect_line_alive (Map, line[i])) {
- line[i] = -1;
- break;
- }
- if (!Points[i])
- Points[i] = Vect_new_line_struct();
- if (!Cats[i])
- Cats[i] = Vect_new_cats_struct();
+ Vect_get_line_nodes(Map, line, &(node[0]), &(node[1]));
+ if (node[0] < 0 || node[1] < 0)
+ continue;
- type[i] = Vect_read_line(Map, Points[i], Cats[i], line[i]);
-
- if (!(type[i] & GV_LINES)) {
- line[i] = -1;
- break;
+ n_nodes = 2;
+ for (j = 0; j < n_nodes; j++) {
+ /* for each line node find lines in threshold */
+ Vect_get_node_coor(Map, node[j], &x, &y, &z);
+ found = Vect_find_line(Map, x, y, z,
+ GV_LINES, thresh, WITHOUT_Z,
+ line);
+
+ if (found > 0 && Vect_line_alive(Map, found)) {
+ /* try to connect lines (given node) */
+ G_debug(3, "Vedit_connect_lines(): lines=%d,%d",
+ line, found);
+ if (connect_lines(Map, !j, line, found, thresh, List)) {
+ G_debug(3, " -> connected");
+ nlines_modified += 2;
+ break;
+ }
}
}
+ }
- if (line[0] == -1 || line[1] == -1)
- continue;
+ return nlines_modified;
+}
- /* check lines intersection
- if (Vect_line_check_intersection (Points[0], Points[1], WITHOUT_Z))
- return 0;
- */
+int connect_lines(struct Map_info *Map, int first, int line_from, int line_to,
+ double thresh, struct ilist *List)
+{
+ int line_new;
+ int type_from, type_to;
+ int n_points, seg, is;
+ double x, y, px, py, fx, fy, x1, y1;
+ double dist, spdist, lpdist, length, dist_p, dist_1, dist_2, dist_3;
+ double angle_p, angle_n, angle;
- /* find node to connect */
- for (i = 0; i < nlines_to_modify; i++) {
- if (i == 0)
- idx = 0;
- else
- idx = Points[0] -> n_points - 1;
+ struct line_pnts *Points_from, *Points_to, *Points_final;
+ struct line_cats *Cats_from, *Cats_to;
- seg[i] = Vect_line_distance (Points[1], Points[0]->x[idx], Points[0]->y[idx], Points[0]->z[idx],
- WITHOUT_Z,
- &px[i], &py[i], &pz[i],
- &dist[i], &spdist[i], &lpdist[i]);
- }
+ Points_from = Vect_new_line_struct();
+ Points_to = Vect_new_line_struct();
+ Points_final = Vect_new_line_struct();
+ Cats_from = Vect_new_cats_struct();
+ Cats_to = Vect_new_cats_struct();
- idx = (dist[0] <= dist[1]) ? 0 : 1;
- pnt_idx[0] = (idx == 0) ? 0 : Points[0] -> n_points - 1; /* which node to connect */
+ type_from = Vect_read_line(Map, Points_from, Cats_from, line_from);
+ type_to = Vect_read_line(Map, Points_to, Cats_to, line_to);
- angle =
- M_PI / 2 -
- acos (fabs (Points[0] -> x[0] - Points[0] -> x[Points[0] -> n_points - 1]) /
- Vect_line_length (Points[0])) -
- asin (fabs (Points[1] -> y[seg[idx] - 1] - py[idx]) / spdist[idx]);
-
- dseg = dist[idx] * tan (angle);
+ line_new = 0;
+ if (!(type_from & GV_LINES) || !(type_to & GV_LINES))
+ line_new = -1;
- /* compute connect points */
- for (i = 0; i < nlines_to_modify; i++) {
- if (0 == Vect_point_on_line (Points[1], lpdist[idx] + dseg, &nx[i], &ny[i], &nz[i], NULL, NULL)) {
- n_on_line[i] = 0;
- }
- else {
- n_on_line[i] = 1;
- }
- dseg *= -1.0;
+ if (line_new > -1) {
+ /* get node (line_from/node) */
+ if (first) {
+ x = Points_from->x[0];
+ y = Points_from->y[0];
+ x1 = Points_from->x[1];
+ y1 = Points_from->y[1];
}
-
- if (!n_on_line[0] || !n_on_line[1]) {
- G_warning (_("Unable to connect lines %d,%d"), line[0], line[1]);
- continue;
+ else {
+ n_points = Points_from->n_points - 1;
+ x = Points_from->x[n_points];
+ y = Points_from->y[n_points];
+ x1 = Points_from->x[n_points-1];
+ y1 = Points_from->y[n_points-1];
}
+ Vect_line_distance(Points_to, x, y, 0.0, WITHOUT_Z,
+ &px, &py, NULL, &dist, &spdist, &lpdist);
+ if (dist > 0.0 && dist <= thresh) {
+ /* lines in threshold */
+ if (first)
+ length = 0;
+ else
+ length = Vect_line_length(Points_from);
- /* choose connect point */
- if (idx == 0)
- pnt_idx[1] = Points[0] -> n_points - 1;
- else
- pnt_idx[1] = 0;
-
- if (Vect_points_distance (nx[0], ny[0], nz[0],
- Points[0] -> x[pnt_idx[1]], Points[0] -> y[pnt_idx[1]],
- Points[0] -> z[pnt_idx[1]],
- WITHOUT_Z) >
- Vect_points_distance (nx[1], ny[1], nz[1],
- Points[0] -> x[pnt_idx[1]], Points[0] -> y[pnt_idx[1]],
- Points[0] -> z[pnt_idx[1]],
- WITHOUT_Z))
- pnt_idx[1] = 0;
- else
- pnt_idx[1] = 1;
-
- dist_connect = Vect_points_distance (nx[pnt_idx[1]], ny[pnt_idx[1]], nz[pnt_idx[1]],
- Points[0] -> x[pnt_idx[0]], Points[0] -> y[pnt_idx[0]],
- Points[0] -> z[pnt_idx[0]],
- WITHOUT_Z);
+ if (Vect_point_on_line(Points_from, length,
+ NULL, NULL, NULL, &angle_n, NULL) > 0) {
+ if (Vect_point_on_line(Points_to, lpdist,
+ NULL, NULL, NULL, &angle_p, NULL) > 0) {
+ angle = angle_p - angle_n;
+ dist_p = dist / tan(angle);
- G_debug (3, "Vedit_connect_lines(): dist=%f/%f -> pnt_idx=%d -> "
- "x=%f, y=%f / dist_connect=%f (thresh=%f)",
- dist[0], dist[1], pnt_idx[0], nx[pnt_idx[1]], ny[pnt_idx[1]], dist_connect, thresh);
+ seg = Vect_point_on_line(Points_to, lpdist + dist_p,
+ &fx, &fy, NULL, NULL, NULL);
+ dist_1 = Vect_points_distance(x, y, 0.0,
+ fx, fy, 0.0, WITHOUT_Z);
+ dist_2 = Vect_points_distance(x1, y1, 0.0,
+ x, y, 0.0, WITHOUT_Z);
+ dist_3 = Vect_points_distance(x1, x1, 0.0,
+ fx, fy, 0.0, WITHOUT_Z);
- if (thresh >= 0.0 && dist_connect > thresh) {
- continue;
- }
-
- /* modify the first line */
- Vect_reset_list (List_updated);
- intersection = Vect_line_check_intersection (Points[0], Points[1], WITHOUT_Z);
+ if (dist_3 < dist_1 + dist_2) {
+ seg = Vect_point_on_line(Points_to, lpdist - dist_p,
+ &fx, &fy, NULL, NULL, NULL);
+ }
- if (!intersection) {
- if (pnt_idx[0] == 0)
- Vect_line_insert_point (Points[0], 0, nx[pnt_idx[1]], ny[pnt_idx[1]], 0.0);
- else
- Vect_append_point (Points[0], nx[pnt_idx[1]], ny[pnt_idx[1]], 0.0);
-
- /* rewrite the first line */
- newline = Vect_rewrite_line (Map, line[0], type[0], Points[0], Cats[0]);
- if (newline < 0) {
- return -1;
+ if(seg > 0) {
+ /* lines connected -> split line_to */
+ /* update line_from */
+ if (first) {
+ Points_from->x[0] = fx;
+ Points_from->y[0] = fy;
+ Points_from->z[0] = 0.0;
+ }
+ else {
+ n_points = Points_from->n_points - 1;
+ Points_from->x[n_points] = fx;
+ Points_from->y[n_points] = fy;
+ Points_from->z[n_points] = 0.0;
+ }
+ line_new = Vect_rewrite_line(Map, line_from, type_from,
+ Points_from, Cats_from);
+ Vect_list_append(List, line_new);
+
+ /* update line_to -- first part*/
+ Vect_reset_line(Points_final);
+ for(is = 0; is < seg; is++) {
+ Vect_append_point(Points_final, Points_to->x[is],
+ Points_to->y[is], Points_to->z[is]);
+ }
+ Vect_append_point(Points_final, fx, fy, 0.0);
+ line_new = Vect_rewrite_line(Map, line_to, type_to,
+ Points_final, Cats_to);
+ Vect_list_append(List, line_new);
+
+ /* write second part */
+ Vect_reset_line(Points_final);
+ Vect_append_point(Points_final, fx, fy, 0.0);
+ for(is = seg; is < Points_to->n_points; is++) {
+ Vect_append_point(Points_final, Points_to->x[is],
+ Points_to->y[is], Points_to->z[is]);
+ }
+
+ /* rewrite first part */
+ line_new = Vect_write_line(Map, type_to,
+ Points_final, Cats_to);
+ Vect_list_append(List, line_new);
+
+ }
+ }
}
- Vect_list_append (List_updated, newline);
- connected = 1;
}
- else
- Vect_list_append (List_updated, line[0]);
-
- /* break the second line */
- Vect_reset_list (List_break);
- if (!intersection ||
- (intersection &&
- Vect_points_distance (nx[pnt_idx[1]], ny[pnt_idx[1]], nz[pnt_idx[1]],
- Points[0] -> x[pnt_idx[0]], Points[0] -> y[pnt_idx[0]],
- Points[0] -> z[pnt_idx[0]],
- WITHOUT_Z) <= 0.0)) {
- Vect_list_append (List_break, line[1]);
- struct line_pnts *coord = Vect_new_line_struct();
- Vect_append_point(coord, nx[pnt_idx[1]], ny[pnt_idx[1]], nz[pnt_idx[1]]);
- Vedit_split_lines(Map, List_break,
- coord, 1e-1,
- List_updated);
- Vect_destroy_line_struct(coord);
- /* snap lines */
- Vect_snap_lines_list (Map, List_updated, 1e-1, NULL, NULL);
- connected = 1;
- }
-
- if (connected)
- nlines_modified += 2;
-
- } /* for each line pair */
-
- /* destroy structures */
- for (i = 0; i < nlines_to_modify; i++) {
- if (Points[i])
- Vect_destroy_line_struct(Points[i]);
- if (Cats[i])
- Vect_destroy_cats_struct(Cats[i]);
}
- Vect_destroy_list (List_break);
- Vect_destroy_list (List_updated);
- return nlines_modified == 2 ? 1 : 0;
+ Vect_destroy_line_struct(Points_from);
+ Vect_destroy_line_struct(Points_to);
+ Vect_destroy_line_struct(Points_final);
+ Vect_destroy_cats_struct(Cats_from);
+ Vect_destroy_cats_struct(Cats_to);
+
+ return line_new > 0 ? 1 : 0;
}
More information about the grass-commit
mailing list