mirror of https://github.com/CGAL/cgal
325 lines
8.8 KiB
C++
325 lines
8.8 KiB
C++
#include <fstream>
|
|
#include <boost/config.hpp>
|
|
#include <boost/version.hpp>
|
|
// CGAL headers
|
|
#include <CGAL/Exact_predicates_inexact_constructions_kernel.h>
|
|
#include <CGAL/Apollonius_graph_2.h>
|
|
#include <CGAL/Apollonius_graph_hierarchy_2.h>
|
|
#include <CGAL/Apollonius_graph_filtered_traits_2.h>
|
|
#include <CGAL/point_generators_2.h>
|
|
#include <CGAL/IO/WKT.h>
|
|
|
|
#include <boost/random/linear_congruential.hpp>
|
|
#include <boost/random/uniform_real.hpp>
|
|
|
|
// Qt headers
|
|
#include <QtGui>
|
|
#include <QString>
|
|
#include <QActionGroup>
|
|
#include <QFileDialog>
|
|
#include <QInputDialog>
|
|
|
|
// GraphicsView items and event filters (input classes)
|
|
#include <CGAL/Qt/ApolloniusGraphGraphicsItem.h>
|
|
#include <CGAL/Qt/GraphicsViewCircleInput.h>
|
|
|
|
// for viewportsBbox
|
|
#include <CGAL/Qt/utility.h>
|
|
|
|
// the two base classes
|
|
#include "ui_Apollonius_graph_2.h"
|
|
#include <CGAL/Qt/DemosMainWindow.h>
|
|
|
|
typedef CGAL::Simple_cartesian<double> K;
|
|
typedef K::Point_2 Point_2;
|
|
typedef K::Iso_rectangle_2 Iso_rectangle_2;
|
|
|
|
typedef CGAL::Apollonius_graph_filtered_traits_2<K,CGAL::Integral_domain_without_division_tag> Gt;
|
|
|
|
typedef Gt::Point_2 Point_2;
|
|
typedef K::Circle_2 Circle_2;
|
|
typedef Gt::Site_2 Apollonius_site_2;
|
|
typedef Gt::Site_2::Weight Weight;
|
|
|
|
typedef CGAL::Apollonius_graph_2<Gt> Apollonius;
|
|
|
|
|
|
class MainWindow :
|
|
public CGAL::Qt::DemosMainWindow,
|
|
public Ui::Apollonius_graph_2
|
|
{
|
|
Q_OBJECT
|
|
|
|
private:
|
|
Apollonius ag;
|
|
QGraphicsScene scene;
|
|
|
|
CGAL::Qt::ApolloniusGraphGraphicsItem<Apollonius,K> * agi;
|
|
CGAL::Qt::GraphicsViewCircleInput<K> * ci;
|
|
|
|
public:
|
|
MainWindow();
|
|
|
|
public Q_SLOTS:
|
|
|
|
void processInput(CGAL::Object o);
|
|
|
|
void on_actionInsertRandomPoints_triggered();
|
|
|
|
void on_actionLoadPoints_triggered();
|
|
|
|
void on_actionSavePoints_triggered();
|
|
|
|
void on_actionClear_triggered();
|
|
|
|
void on_actionRecenter_triggered();
|
|
|
|
virtual void open(QString fileName);
|
|
|
|
Q_SIGNALS:
|
|
void changed();
|
|
};
|
|
|
|
|
|
MainWindow::MainWindow()
|
|
: DemosMainWindow()
|
|
{
|
|
setupUi(this);
|
|
|
|
this->graphicsView->setAcceptDrops(false);
|
|
|
|
agi = new CGAL::Qt::ApolloniusGraphGraphicsItem<Apollonius, K>(&ag);
|
|
|
|
QObject::connect(this, SIGNAL(changed()),
|
|
agi, SLOT(modelChanged()));
|
|
|
|
agi->setSitesPen(QPen(Qt::red, 0, Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin));
|
|
agi->setEdgesPen(QPen(Qt::blue, 0, Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin));
|
|
scene.addItem(agi);
|
|
|
|
// Setup input handlers. They get events before the scene gets them
|
|
// and the input they generate is passed to the triangulation with
|
|
// the signal/slot mechanism
|
|
|
|
ci = new CGAL::Qt::GraphicsViewCircleInput<K>(this, &scene);
|
|
QObject::connect(ci, SIGNAL(generate(CGAL::Object)),
|
|
this, SLOT(processInput(CGAL::Object)));
|
|
|
|
scene.installEventFilter(ci);
|
|
|
|
//
|
|
// Manual handling of actions
|
|
//
|
|
|
|
QObject::connect(this->actionQuit, SIGNAL(triggered()),
|
|
this, SLOT(close()));
|
|
|
|
//
|
|
// Setup the scene and the view
|
|
//
|
|
scene.setItemIndexMethod(QGraphicsScene::NoIndex);
|
|
scene.setSceneRect(-100, -100, 100, 100);
|
|
this->graphicsView->setScene(&scene);
|
|
this->graphicsView->setMouseTracking(true);
|
|
|
|
// Turn the vertical axis upside down
|
|
this->graphicsView->transform().scale(1, -1);
|
|
|
|
// The navigation adds zooming and translation functionality to the
|
|
// QGraphicsView
|
|
this->addNavigation(this->graphicsView);
|
|
|
|
this->setupStatusBar();
|
|
this->setupOptionsMenu();
|
|
this->addAboutDemo(":/cgal/help/about_Apollonius_graph_2.html");
|
|
this->addAboutCGAL();
|
|
|
|
this->addRecentFiles(this->menuFile, this->actionQuit);
|
|
connect(this, SIGNAL(openRecentFile(QString)),
|
|
this, SLOT(open(QString)));
|
|
}
|
|
|
|
|
|
void
|
|
MainWindow::processInput(CGAL::Object o)
|
|
{
|
|
std::pair<Point_2, double> center_and_sr;
|
|
if(CGAL::assign(center_and_sr, o)){
|
|
ag.insert(Apollonius_site_2(center_and_sr.first, sqrt(center_and_sr.second)));
|
|
Q_EMIT( changed());
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Qt Automatic Connections
|
|
* https://doc.qt.io/qt-5/designer-using-a-ui-file.html#automatic-connections
|
|
*
|
|
* setupUi(this) generates connections to the slots named
|
|
* "on_<action_name>_<signal_name>"
|
|
*/
|
|
|
|
|
|
void
|
|
MainWindow::on_actionClear_triggered()
|
|
{
|
|
ag.clear();
|
|
Q_EMIT( changed());
|
|
}
|
|
|
|
|
|
void
|
|
MainWindow::on_actionInsertRandomPoints_triggered()
|
|
{
|
|
QRectF rect = CGAL::Qt::viewportsBbox(&scene);
|
|
CGAL::Qt::Converter<K> convert;
|
|
Iso_rectangle_2 isor = convert(rect);
|
|
double width = isor.xmax() - isor.xmin();
|
|
|
|
CGAL::Random_points_in_iso_rectangle_2<Point_2> pg((isor.min)(), (isor.max)());
|
|
bool ok = false;
|
|
|
|
const int number_of_points =
|
|
QInputDialog::getInt(this,
|
|
tr("Number of random points"),
|
|
tr("Enter number of random points"),
|
|
100,
|
|
0,
|
|
(std::numeric_limits<int>::max)(),
|
|
1,
|
|
&ok);
|
|
|
|
if(!ok) {
|
|
return;
|
|
}
|
|
|
|
// wait cursor
|
|
QApplication::setOverrideCursor(Qt::WaitCursor);
|
|
std::vector<Apollonius_site_2> points;
|
|
points.reserve(number_of_points);
|
|
boost::rand48 rng;
|
|
boost::uniform_real<> dist(0.005*width, 0.05*width);
|
|
boost::variate_generator<boost::rand48&, boost::uniform_real<> > radius(rng,dist);
|
|
|
|
for(int i = 0; i < number_of_points; ++i){
|
|
points.push_back(Apollonius_site_2(*pg++,radius()));
|
|
}
|
|
ag.insert(points.begin(), points.end());
|
|
// default cursor
|
|
QApplication::restoreOverrideCursor();
|
|
Q_EMIT( changed());
|
|
}
|
|
|
|
|
|
void
|
|
MainWindow::on_actionLoadPoints_triggered()
|
|
{
|
|
QString fileName = QFileDialog::getOpenFileName(this,
|
|
tr("Open Points file"),
|
|
".",
|
|
tr("CGAL files (*.wpts.cgal);;"
|
|
"WKT files (*.wkt *.WKT);;"
|
|
"All files (*)"));
|
|
if(! fileName.isEmpty()){
|
|
open(fileName);
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWindow::open(QString fileName)
|
|
{
|
|
|
|
QApplication::setOverrideCursor(Qt::WaitCursor);
|
|
if(! fileName.isEmpty()){
|
|
std::ifstream ifs(qPrintable(fileName));
|
|
|
|
std::vector<Apollonius_site_2> points;
|
|
if(fileName.endsWith(".wkt", Qt::CaseInsensitive))
|
|
{
|
|
std::vector<K::Point_3> point_3_s;
|
|
CGAL::read_multi_point_WKT(ifs, point_3_s);
|
|
for(const K::Point_3& point_3 : point_3_s)
|
|
{
|
|
points.push_back(Apollonius_site_2(K::Point_2(point_3.x(), point_3.y()), point_3.z()));
|
|
}
|
|
} else{
|
|
K::Weighted_point_2 p;
|
|
while(ifs >> p) {
|
|
points.push_back(Apollonius_site_2(p.point(),p.weight()));
|
|
}
|
|
}
|
|
ag.insert(points.begin(), points.end());
|
|
this->addToRecentFiles(fileName);
|
|
actionRecenter->trigger();
|
|
Q_EMIT( changed());
|
|
}
|
|
QApplication::restoreOverrideCursor();
|
|
}
|
|
|
|
void
|
|
MainWindow::on_actionSavePoints_triggered()
|
|
{
|
|
QString fileName = QFileDialog::getSaveFileName(this,
|
|
tr("Save points"),
|
|
".reg.cgal",
|
|
tr("Weighted Points (*.wpts.cgal);;"
|
|
"WKT files(*.wkt *.WKT);;"
|
|
"All (*)"));
|
|
if(! fileName.isEmpty()){
|
|
std::ofstream ofs(qPrintable(fileName));
|
|
if(fileName.endsWith(".wkt",Qt::CaseInsensitive))
|
|
{
|
|
std::vector<K::Point_3> points;
|
|
for(Apollonius::Sites_iterator
|
|
vit = ag.sites_begin(),
|
|
end = ag.sites_end();
|
|
vit!= end; ++vit)
|
|
{
|
|
points.push_back(K::Point_3(vit->point().x(),
|
|
vit->point().y(),
|
|
vit->weight()));
|
|
}
|
|
CGAL::write_multi_point_WKT(ofs, points);
|
|
}
|
|
else
|
|
for(Apollonius::Sites_iterator
|
|
vit = ag.sites_begin(),
|
|
end = ag.sites_end();
|
|
vit!= end; ++vit)
|
|
{
|
|
ofs << vit->point()<<" "<<vit->weight()<<std::endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWindow::on_actionRecenter_triggered()
|
|
{
|
|
this->graphicsView->setSceneRect(agi->boundingRect());
|
|
this->graphicsView->fitInView(agi->boundingRect(), Qt::KeepAspectRatio);
|
|
}
|
|
|
|
|
|
#include "Apollonius_graph_2.moc"
|
|
#include <CGAL/Qt/resources.h>
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
QApplication app(argc, argv);
|
|
|
|
app.setOrganizationDomain("geometryfactory.com");
|
|
app.setOrganizationName("GeometryFactory");
|
|
app.setApplicationName("Apollonius_graph_2 demo");
|
|
|
|
// Import resources from libCGAL (Qt5).
|
|
// See https://doc.qt.io/qt-5/qdir.html#Q_INIT_RESOURCE
|
|
CGAL_QT_INIT_RESOURCES;
|
|
Q_INIT_RESOURCE(Apollonius_graph_2);
|
|
|
|
MainWindow mainWindow;
|
|
mainWindow.show();
|
|
return app.exec();
|
|
}
|