2009-07-25 00:50:30 +04:00
|
|
|
/*
|
2011-01-21 04:21:09 +03:00
|
|
|
* OpenSCAD (www.openscad.org)
|
|
|
|
* Copyright (C) 2009-2011 Clifford Wolf <clifford@clifford.at> and
|
|
|
|
* Marius Kintel <marius@kintel.net>
|
2009-07-25 00:50:30 +04:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
2010-02-01 12:34:18 +03:00
|
|
|
* As a special exception, you have permission to link this program
|
|
|
|
* with the CGAL library and distribute executables, as long as you
|
|
|
|
* follow the requirements of the GNU GPL in regard to all of the
|
|
|
|
* software in the executable aside from CGAL.
|
|
|
|
*
|
2009-07-25 00:50:30 +04:00
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2010-01-30 07:26:05 +03:00
|
|
|
#include "dxfdim.h"
|
|
|
|
#include "value.h"
|
|
|
|
#include "function.h"
|
|
|
|
#include "dxfdata.h"
|
|
|
|
#include "builtin.h"
|
2009-12-09 17:22:50 +03:00
|
|
|
#include "printutils.h"
|
2010-02-23 23:47:54 +03:00
|
|
|
#include "context.h"
|
2009-07-25 00:50:30 +04:00
|
|
|
|
2011-01-18 02:02:56 +03:00
|
|
|
#include "mathc99.h"
|
2010-01-30 07:26:05 +03:00
|
|
|
#include <QHash>
|
2011-01-22 20:14:30 +03:00
|
|
|
#include <QDateTime>
|
2011-01-18 02:20:12 +03:00
|
|
|
#include <QFileInfo>
|
2010-01-10 16:32:22 +03:00
|
|
|
|
|
|
|
QHash<QString,Value> dxf_dim_cache;
|
|
|
|
QHash<QString,Value> dxf_cross_cache;
|
|
|
|
|
2010-02-23 23:47:54 +03:00
|
|
|
Value builtin_dxf_dim(const Context *ctx, const QVector<QString> &argnames, const QVector<Value> &args)
|
2009-07-25 00:50:30 +04:00
|
|
|
{
|
|
|
|
QString filename;
|
|
|
|
QString layername;
|
|
|
|
QString name;
|
|
|
|
double xorigin = 0;
|
|
|
|
double yorigin = 0;
|
|
|
|
double scale = 1;
|
|
|
|
|
|
|
|
for (int i = 0; i < argnames.count() && i < args.count(); i++) {
|
|
|
|
if (argnames[i] == "file")
|
2010-11-08 00:29:34 +03:00
|
|
|
filename = ctx->get_absolute_path(QString::fromStdString(args[i].text));
|
2009-07-25 00:50:30 +04:00
|
|
|
if (argnames[i] == "layer")
|
2010-11-08 00:29:34 +03:00
|
|
|
layername = QString::fromStdString(args[i].text);
|
2009-07-25 00:50:30 +04:00
|
|
|
if (argnames[i] == "origin")
|
|
|
|
args[i].getv2(xorigin, yorigin);
|
|
|
|
if (argnames[i] == "scale")
|
|
|
|
args[i].getnum(scale);
|
|
|
|
if (argnames[i] == "name")
|
2010-11-08 00:29:34 +03:00
|
|
|
name = QString::fromStdString(args[i].text);
|
2009-07-25 00:50:30 +04:00
|
|
|
}
|
|
|
|
|
2011-01-18 02:20:12 +03:00
|
|
|
QFileInfo fileInfo(filename);
|
2010-01-10 16:32:22 +03:00
|
|
|
|
|
|
|
QString key = filename + "|" + layername + "|" + name + "|" + QString::number(xorigin) + "|" + QString::number(yorigin) +
|
2011-01-18 02:20:12 +03:00
|
|
|
"|" + QString::number(scale) + "|" + QString::number(fileInfo.lastModified().toTime_t()) + "|" + QString::number(fileInfo.size());
|
2010-01-10 16:32:22 +03:00
|
|
|
|
|
|
|
if (dxf_dim_cache.contains(key))
|
|
|
|
return dxf_dim_cache[key];
|
|
|
|
|
2009-07-25 00:50:30 +04:00
|
|
|
DxfData dxf(36, 0, 0, filename, layername, xorigin, yorigin, scale);
|
|
|
|
|
|
|
|
for (int i = 0; i < dxf.dims.count(); i++)
|
|
|
|
{
|
|
|
|
if (!name.isNull() && dxf.dims[i].name != name)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
DxfData::Dim *d = &dxf.dims[i];
|
|
|
|
int type = d->type & 7;
|
|
|
|
|
|
|
|
if (type == 0) {
|
|
|
|
// Rotated, horizontal or vertical
|
|
|
|
double x = d->coords[4][0] - d->coords[3][0];
|
|
|
|
double y = d->coords[4][1] - d->coords[3][1];
|
|
|
|
double angle = d->angle;
|
2009-07-25 01:57:46 +04:00
|
|
|
double distance_projected_on_line = fabs(x * cos(angle*M_PI/180) + y * sin(angle*M_PI/180));
|
2010-01-10 16:32:22 +03:00
|
|
|
return dxf_dim_cache[key] = Value(distance_projected_on_line);
|
2009-07-25 00:50:30 +04:00
|
|
|
}
|
2010-01-21 14:25:46 +03:00
|
|
|
else if (type == 1) {
|
2009-07-25 00:50:30 +04:00
|
|
|
// Aligned
|
2010-01-21 14:25:46 +03:00
|
|
|
double x = d->coords[4][0] - d->coords[3][0];
|
|
|
|
double y = d->coords[4][1] - d->coords[3][1];
|
|
|
|
return dxf_dim_cache[key] = Value(sqrt(x*x + y*y));
|
2009-07-25 00:50:30 +04:00
|
|
|
}
|
2010-01-21 14:25:46 +03:00
|
|
|
else if (type == 2) {
|
2009-07-25 00:50:30 +04:00
|
|
|
// Angular
|
2009-07-25 01:57:46 +04:00
|
|
|
double a1 = atan2(d->coords[0][0] - d->coords[5][0], d->coords[0][1] - d->coords[5][1]);
|
|
|
|
double a2 = atan2(d->coords[4][0] - d->coords[3][0], d->coords[4][1] - d->coords[3][1]);
|
2010-01-10 16:32:22 +03:00
|
|
|
return dxf_dim_cache[key] = Value(fabs(a1 - a2) * 180 / M_PI);
|
2009-07-25 00:50:30 +04:00
|
|
|
}
|
2010-01-21 14:25:46 +03:00
|
|
|
else if (type == 3 || type == 4) {
|
|
|
|
// Diameter or Radius
|
|
|
|
double x = d->coords[5][0] - d->coords[0][0];
|
|
|
|
double y = d->coords[5][1] - d->coords[0][1];
|
|
|
|
return dxf_dim_cache[key] = Value(sqrt(x*x + y*y));
|
2009-07-25 00:50:30 +04:00
|
|
|
}
|
2010-01-21 14:25:46 +03:00
|
|
|
else if (type == 5) {
|
2009-07-25 00:50:30 +04:00
|
|
|
// Angular 3 Point
|
|
|
|
}
|
2010-01-21 14:25:46 +03:00
|
|
|
else if (type == 6) {
|
|
|
|
// Ordinate
|
|
|
|
return dxf_dim_cache[key] = Value((d->type & 64) ? d->coords[3][0] : d->coords[3][1]);
|
|
|
|
}
|
2009-07-25 00:50:30 +04:00
|
|
|
|
2010-01-11 22:18:20 +03:00
|
|
|
PRINTA("WARNING: Dimension `%1' in `%2', layer `%3' has unsupported type!", name, filename, layername);
|
2009-07-25 00:50:30 +04:00
|
|
|
return Value();
|
|
|
|
}
|
|
|
|
|
2010-01-11 22:18:20 +03:00
|
|
|
PRINTA("WARNING: Can't find dimension `%1' in `%2', layer `%3'!", name, filename, layername);
|
2009-07-25 00:50:30 +04:00
|
|
|
|
|
|
|
return Value();
|
|
|
|
}
|
|
|
|
|
2010-02-23 23:47:54 +03:00
|
|
|
Value builtin_dxf_cross(const Context *ctx, const QVector<QString> &argnames, const QVector<Value> &args)
|
2009-07-25 01:57:46 +04:00
|
|
|
{
|
|
|
|
QString filename;
|
|
|
|
QString layername;
|
|
|
|
double xorigin = 0;
|
|
|
|
double yorigin = 0;
|
|
|
|
double scale = 1;
|
|
|
|
|
|
|
|
for (int i = 0; i < argnames.count() && i < args.count(); i++) {
|
|
|
|
if (argnames[i] == "file")
|
2010-11-08 00:29:34 +03:00
|
|
|
filename = ctx->get_absolute_path(QString::fromStdString(args[i].text));
|
2009-07-25 01:57:46 +04:00
|
|
|
if (argnames[i] == "layer")
|
2010-11-08 00:29:34 +03:00
|
|
|
layername = QString::fromStdString(args[i].text);
|
2009-07-25 01:57:46 +04:00
|
|
|
if (argnames[i] == "origin")
|
|
|
|
args[i].getv2(xorigin, yorigin);
|
|
|
|
if (argnames[i] == "scale")
|
|
|
|
args[i].getnum(scale);
|
|
|
|
}
|
|
|
|
|
2011-01-18 02:20:12 +03:00
|
|
|
QFileInfo fileInfo(filename);
|
2010-01-10 16:32:22 +03:00
|
|
|
|
|
|
|
QString key = filename + "|" + layername + "|" + QString::number(xorigin) + "|" + QString::number(yorigin) +
|
2011-01-18 02:20:12 +03:00
|
|
|
"|" + QString::number(scale) + "|" + QString::number(fileInfo.lastModified().toTime_t()) + "|" + QString::number(fileInfo.size());
|
2010-01-10 16:32:22 +03:00
|
|
|
|
|
|
|
if (dxf_cross_cache.contains(key))
|
|
|
|
return dxf_cross_cache[key];
|
|
|
|
|
2009-07-25 01:57:46 +04:00
|
|
|
DxfData dxf(36, 0, 0, filename, layername, xorigin, yorigin, scale);
|
|
|
|
|
|
|
|
double coords[4][2];
|
|
|
|
|
|
|
|
for (int i = 0, j = 0; i < dxf.paths.count(); i++) {
|
|
|
|
if (dxf.paths[i].points.count() != 2)
|
|
|
|
continue;
|
2011-08-05 04:11:20 +04:00
|
|
|
coords[j][0] = (*dxf.paths[i].points[0])[0];
|
|
|
|
coords[j++][1] = (*dxf.paths[i].points[0])[1];
|
|
|
|
coords[j][0] = (*dxf.paths[i].points[1])[0];
|
|
|
|
coords[j++][1] = (*dxf.paths[i].points[1])[1];
|
2009-07-25 01:57:46 +04:00
|
|
|
|
|
|
|
if (j == 4) {
|
|
|
|
double x1 = coords[0][0], y1 = coords[0][1];
|
|
|
|
double x2 = coords[1][0], y2 = coords[1][1];
|
|
|
|
double x3 = coords[2][0], y3 = coords[2][1];
|
|
|
|
double x4 = coords[3][0], y4 = coords[3][1];
|
|
|
|
double dem = (y4 - y3)*(x2 - x1) - (x4 - x3)*(y2 - y1);
|
|
|
|
if (dem == 0)
|
|
|
|
break;
|
|
|
|
double ua = ((x4 - x3)*(y1 - y3) - (y4 - y3)*(x1 - x3)) / dem;
|
|
|
|
// double ub = ((x2 - x1)*(y1 - y3) - (y2 - y1)*(x1 - x3)) / dem;
|
|
|
|
double x = x1 + ua*(x2 - x1);
|
|
|
|
double y = y1 + ua*(y2 - y1);
|
|
|
|
Value ret;
|
|
|
|
ret.type = Value::VECTOR;
|
2010-11-08 01:12:34 +03:00
|
|
|
ret.append(new Value(x));
|
|
|
|
ret.append(new Value(y));
|
2010-01-10 16:32:22 +03:00
|
|
|
return dxf_cross_cache[key] = ret;
|
2009-07-25 01:57:46 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-11 22:18:20 +03:00
|
|
|
PRINTA("WARNING: Can't find cross in `%1', layer `%2'!", filename, layername);
|
2009-07-25 01:57:46 +04:00
|
|
|
|
|
|
|
return Value();
|
|
|
|
}
|
|
|
|
|
2009-07-25 00:50:30 +04:00
|
|
|
void initialize_builtin_dxf_dim()
|
|
|
|
{
|
|
|
|
builtin_functions["dxf_dim"] = new BuiltinFunction(&builtin_dxf_dim);
|
2009-07-25 01:57:46 +04:00
|
|
|
builtin_functions["dxf_cross"] = new BuiltinFunction(&builtin_dxf_cross);
|
2009-07-25 00:50:30 +04:00
|
|
|
}
|
|
|
|
|