1
0
mirror of git://projects.qi-hardware.com/fped.git synced 2024-11-04 23:55:54 +02:00
fped/coord.c
werner 5d7ab083a3 - README: described use of semicolons
- README: added that loops can also execute zero times
- accept labels only at the beginning of a line
- rectangles and lines no longer use the bounding box for drawing (caused
  offset problems since we now correct for the line width)
- dist_rect and inside_rect no longer require their input pre-sorted
- pad instances now have their own structure and no longer abuse the bounding
  box to know the pad coordinates
- Makefile: use $(GEN) for fig2dev, to reduce chattiness
- when dragging a point, the symbol is now adjusted accordingly
- added moving of rects, pads, circles, and arcs
- added creation of pads 
- moved rotate_r from gui_inst.c to coord.c
- new function "theta" that combines most of the angle calculations
- save_pix_buf: y < 0 clipping changed width, not height



git-svn-id: http://svn.openmoko.org/trunk/eda/fped@5386 99fdad57-331a-0410-800a-d7fa5415bdb3
2009-08-04 21:45:33 +00:00

214 lines
3.7 KiB
C

/*
* coord.c - Coordinate representation and basic operations
*
* Written 2009 by Werner Almesberger
* Copyright 2009 by Werner Almesberger
*
* 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; either version 2 of the License, or
* (at your option) any later version.
*/
#include <math.h>
#include "coord.h"
/* ----- unit conversion --------------------------------------------------- */
double mm_to_mil(double mm, int exponent)
{
return mm*pow(MIL_IN_MM, -exponent);
}
double mil_to_mm(double mil, int exponent)
{
return mil*pow(MIL_IN_MM, exponent);
}
/* ----- vector operations ------------------------------------------------- */
struct coord normalize(struct coord v, unit_type len)
{
double f;
f = len/hypot(v.x, v.y);
v.x *= f;
v.y *= f;
return v;
}
struct coord rotate(struct coord v, double angle)
{
double rad = M_PI*angle/180.0;
struct coord res;
res.x = v.x*cos(rad)-v.y*sin(rad);
res.y = v.y*cos(rad)+v.x*sin(rad);
return res;
}
struct coord add_vec(struct coord a, struct coord b)
{
a.x += b.x;
a.y += b.y;
return a;
}
struct coord sub_vec(struct coord a, struct coord b)
{
a.x -= b.x;
a.y -= b.y;
return a;
}
struct coord neg_vec(struct coord v)
{
v.x = -v.x;
v.y = -v.y;
return v;
}
/* ----- point on circle --------------------------------------------------- */
struct coord rotate_r(struct coord c, unit_type r, double angle)
{
struct coord p;
angle = angle/180.0*M_PI;
p.x = c.x+r*cos(angle);
p.y = c.y+r*sin(angle);
return p;
}
double theta(struct coord c, struct coord p)
{
double a;
a = atan2(p.y-c.y, p.x-c.x)/M_PI*180.0;
if (a < 0)
a += 360.0;
return a;
}
/* ----- sorting coordinates ----------------------------------------------- */
void swap_coord(unit_type *a, unit_type *b)
{
unit_type tmp;
tmp = *a;
*a = *b;
*b = tmp;
}
void sort_coord(struct coord *min, struct coord *max)
{
if (min->x > max->x)
swap_coord(&min->x, &max->x);
if (min->y > max->y)
swap_coord(&min->y, &max->y);
}
/* ----- distance calculations --------------------------------------------- */
unit_type dist_point(struct coord a, struct coord b)
{
return hypot(a.x-b.x, a.y-b.y);
}
static unit_type dist_line_xy(unit_type px, unit_type py,
unit_type ax, unit_type ay, unit_type bx, unit_type by)
{
unit_type d_min, d;
double a, f;
d_min = hypot(ax-px, ay-py);
d = hypot(bx-px, by-py);
if (d < d_min)
d_min = d;
if (ax != bx || ay != by) {
/*
* We make a the line vector from point B and b the vector from
* B to point P. Then we calculate the projection of b on a.
*/
ax -= bx;
ay -= by;
bx = px-bx;
by = py-by;
a = hypot(ax, ay);
f = ((double) ax*bx+(double) ay*by)/a/a;
if (f >= 0 && f <= 1) {
bx -= f*ax;
by -= f*ay;
d = hypot(bx, by);
if (d < d_min)
d_min = d;
}
}
return d_min;
}
unit_type dist_line(struct coord p, struct coord a, struct coord b)
{
return dist_line_xy(p.x, p.y, a.x, a.y, b.x, b.y);
}
unit_type dist_rect(struct coord p, struct coord a, struct coord b)
{
unit_type d_min, d;
d_min = dist_line_xy(p.x, p.y, a.x, a.y, b.x, a.y);
d = dist_line_xy(p.x, p.y, a.x, a.y, a.x, b.y);
if (d < d_min)
d_min = d;
d = dist_line_xy(p.x, p.y, a.x, b.y, b.x, b.y);
if (d < d_min)
d_min = d;
d = dist_line_xy(p.x, p.y, b.x, a.y, b.x, b.y);
if (d < d_min)
d_min = d;
return d_min;
}
int inside_rect(struct coord p, struct coord a, struct coord b)
{
sort_coord(&a, &b);
if (p.x < a.x || p.x > b.x)
return 0;
if (p.y < a.y || p.y > b.y)
return 0;
return 1;
}
unit_type dist_circle(struct coord p, struct coord c, unit_type r)
{
unit_type d;
d = hypot(p.x-c.x, p.y-c.y);
return fabs(d-r);
}