klayout/src/lib/libBasicRoundPath.cc

157 lines
5.2 KiB
C++
Raw Normal View History

/*
KLayout Layout Viewer
2017-02-12 15:28:14 +01:00
Copyright (C) 2006-2017 Matthias Koefferlein
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.
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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "libBasicRoundPath.h"
#include <cmath>
namespace lib
{
// --------------------------------------------------------------------------
// Implementation
static const size_t p_layer = 0;
static const size_t p_radius = 1;
static const size_t p_path = 2;
static const size_t p_npoints = 3;
static const size_t p_total = 4;
BasicRoundPath::BasicRoundPath ()
{
// .. nothing yet ..
}
bool
BasicRoundPath::can_create_from_shape (const db::Layout & /*layout*/, const db::Shape &shape, unsigned int /*layer*/) const
{
return shape.is_path ();
}
db::pcell_parameters_type
BasicRoundPath::parameters_from_shape (const db::Layout &layout, const db::Shape &shape, unsigned int layer) const
{
db::Path path;
shape.path (path);
// use map_parameters to create defaults for the other parameters
std::map<size_t, tl::Variant> nm;
nm.insert (std::make_pair (p_layer, tl::Variant (layout.get_properties (layer))));
nm.insert (std::make_pair (p_path, tl::Variant (db::CplxTrans (layout.dbu ()) * path)));
// use 1/10 of the minimum bbox dimension as a rough initialisation of the radius
nm.insert (std::make_pair (p_radius, tl::Variant (layout.dbu () * (std::min (path.box ().width (), path.box ().height ()) / 10))));
return map_parameters (nm);
}
std::vector<db::PCellLayerDeclaration>
BasicRoundPath::get_layer_declarations (const db::pcell_parameters_type &parameters) const
{
std::vector<db::PCellLayerDeclaration> layers;
if (parameters.size () > p_layer && parameters [p_layer].is_user<db::LayerProperties> ()) {
db::LayerProperties lp = parameters [p_layer].to_user<db::LayerProperties> ();
if (lp != db::LayerProperties ()) {
layers.push_back (lp);
}
}
return layers;
}
void
BasicRoundPath::produce (const db::Layout &layout, const std::vector<unsigned int> &layer_ids, const db::pcell_parameters_type &parameters, db::Cell &cell) const
{
if (parameters.size () < p_total || layer_ids.size () < 1) {
return;
}
double r = std::max (0.0, parameters [p_radius].to_double () / layout.dbu ());
int n = std::max (3, parameters [p_npoints].to_int ());
if (! parameters [p_path].is_user<db::DPath> ()) {
return;
}
db::DPath path = db::DCplxTrans (1.0 / layout.dbu ()) * parameters [p_path].to_user<db::DPath> ();
// Actually perform the rounding
db::DPath new_path = db::round_path_corners (path, r, n, 0.5);
// Create the hull and deliver it
std::vector<db::DPoint> hull;
new_path.hull (hull, n);
db::Polygon poly;
poly.assign_hull (hull.begin (), hull.end (), db::cast_op<db::Point, db::DPoint> ());
// Produce the shape
cell.shapes (layer_ids [p_layer]).insert (poly);
}
std::string
BasicRoundPath::get_display_name (const db::pcell_parameters_type &parameters) const
{
return std::string("ROUND_PATH(r=") + tl::micron_to_string (parameters [p_radius].to_double ()) + ")";
}
std::vector<db::PCellParameterDeclaration>
BasicRoundPath::get_parameter_declarations () const
{
std::vector<db::PCellParameterDeclaration> parameters;
// parameter #0: layer
tl_assert (parameters.size () == p_layer);
parameters.push_back (db::PCellParameterDeclaration ("layer"));
parameters.back ().set_type (db::PCellParameterDeclaration::t_layer);
parameters.back ().set_description (tl::to_string (QObject::tr ("Layer")));
// parameter #1: radius
tl_assert (parameters.size () == p_radius);
parameters.push_back (db::PCellParameterDeclaration ("radius"));
parameters.back ().set_type (db::PCellParameterDeclaration::t_double);
parameters.back ().set_description (tl::to_string (QObject::tr ("Radius")));
parameters.back ().set_default (0.1);
parameters.back ().set_unit (tl::to_string (QObject::tr ("micron")));
// parameter #2: handle
tl_assert (parameters.size () == p_path);
parameters.push_back (db::PCellParameterDeclaration ("path"));
parameters.back ().set_type (db::PCellParameterDeclaration::t_shape);
db::DPath p;
p.width (0.1);
db::DPoint pts[] = { db::DPoint(0, 0), db::DPoint(0.2, 0), db::DPoint (0.2, 0.2) };
p.assign (pts, pts + sizeof (pts) / sizeof (pts[0]));
parameters.back ().set_default (p);
// parameter #3: number of points
tl_assert (parameters.size () == p_npoints);
parameters.push_back (db::PCellParameterDeclaration ("npoints"));
parameters.back ().set_type (db::PCellParameterDeclaration::t_int);
parameters.back ().set_description (tl::to_string (QObject::tr ("Number of points / full circle.")));
parameters.back ().set_default (64);
return parameters;
}
}