mirror of https://github.com/CGAL/cgal
354 lines
14 KiB
C++
354 lines
14 KiB
C++
#include "Polyhedron_demo_plugin_helper.h"
|
|
#include "Polyhedron_demo_plugin_interface.h"
|
|
|
|
#include "ui_Mesh_segmentation_widget.h"
|
|
#include "Scene_polyhedron_item.h"
|
|
#include "Polyhedron_type.h"
|
|
#include "Scene.h"
|
|
#include "Color_map.h"
|
|
|
|
#include <CGAL/mesh_segmentation.h>
|
|
#include <QApplication>
|
|
#include <QMainWindow>
|
|
#include <QInputDialog>
|
|
#include <QTime>
|
|
#include <QAction>
|
|
#include <QDebug>
|
|
#include <QObject>
|
|
#include <QDockWidget>
|
|
//#include <QtConcurrentRun>
|
|
#include <map>
|
|
#include <algorithm>
|
|
#include <vector>
|
|
#include <CGAL/property_map.h>
|
|
|
|
|
|
template<class PolyhedronWithId, class ValueType>
|
|
struct Polyhedron_with_id_to_vector_property_map
|
|
: public boost::put_get_helper<ValueType&,
|
|
Polyhedron_with_id_to_vector_property_map<PolyhedronWithId, ValueType> >
|
|
{
|
|
public:
|
|
typedef typename PolyhedronWithId::Facet_const_handle key_type;
|
|
typedef ValueType value_type;
|
|
typedef value_type& reference;
|
|
typedef boost::lvalue_property_map_tag category;
|
|
|
|
Polyhedron_with_id_to_vector_property_map() : internal_vector(NULL) { }
|
|
Polyhedron_with_id_to_vector_property_map(std::vector<ValueType>* internal_vector)
|
|
: internal_vector(internal_vector) { }
|
|
|
|
reference operator[](key_type key) const { return (*internal_vector)[key->id()]; }
|
|
private:
|
|
std::vector<ValueType>* internal_vector;
|
|
};
|
|
|
|
class Polyhedron_demo_mesh_segmentation_plugin :
|
|
public QObject,
|
|
public Polyhedron_demo_plugin_helper
|
|
{
|
|
Q_OBJECT
|
|
Q_INTERFACES(Polyhedron_demo_plugin_interface)
|
|
private:
|
|
typedef std::map<Scene_polyhedron_item*, std::vector<double> > Item_sdf_map;
|
|
public:
|
|
|
|
QList<QAction*> actions() const {
|
|
return QList<QAction*>() << actionSegmentation;
|
|
}
|
|
|
|
bool applicable() const {
|
|
return
|
|
qobject_cast<Scene_polyhedron_item*>(scene->item(scene->mainSelectionIndex()));
|
|
}
|
|
|
|
void init(QMainWindow* mainWindow, Scene_interface* scene_interface) {
|
|
this->scene = scene_interface;
|
|
this->mw = mainWindow;
|
|
actionSegmentation = new QAction("Mesh Segmentation", mw);
|
|
connect(actionSegmentation, SIGNAL(triggered()),this, SLOT(on_actionSegmentation_triggered()));
|
|
|
|
// adding slot for itemAboutToBeDestroyed signal, aim is removing item from item-functor map.
|
|
|
|
if( Scene* scene = dynamic_cast<Scene*>(scene_interface) ) {
|
|
connect(scene, SIGNAL(itemAboutToBeDestroyed(Scene_item*)), this, SLOT(itemAboutToBeDestroyed(Scene_item*)));
|
|
}
|
|
|
|
init_color_map_sdf();
|
|
init_color_map_segmentation();
|
|
|
|
dock_widget = new QDockWidget("Mesh segmentation parameters", mw);
|
|
dock_widget->setVisible(false); // do not show at the beginning
|
|
ui_widget.setupUi(dock_widget);
|
|
mw->addDockWidget(Qt::LeftDockWidgetArea, dock_widget);
|
|
|
|
connect(ui_widget.Partition_button, SIGNAL(clicked()), this, SLOT(on_Partition_button_clicked()));
|
|
connect(ui_widget.SDF_button, SIGNAL(clicked()), this, SLOT(on_SDF_button_clicked()));
|
|
}
|
|
|
|
template<class SDFPropertyMap>
|
|
void colorize_sdf(Scene_polyhedron_item* item, SDFPropertyMap sdf_values, std::vector<QColor>& color_vector);
|
|
template<class SegmentPropertyMap>
|
|
void colorize_segmentation(Scene_polyhedron_item* item, SegmentPropertyMap segment_ids, std::vector<QColor>& color_vector);
|
|
void check_and_set_ids(Polyhedron* polyhedron);
|
|
void init_color_map_sdf();
|
|
void init_color_map_segmentation();
|
|
|
|
public slots:
|
|
void on_actionSegmentation_triggered();
|
|
void on_Partition_button_clicked();
|
|
void on_SDF_button_clicked();
|
|
void itemAboutToBeDestroyed(Scene_item*);
|
|
private:
|
|
QAction* actionSegmentation;
|
|
QDockWidget* dock_widget;
|
|
Ui::Mesh_segmentation ui_widget;
|
|
|
|
std::vector<QColor> color_map_sdf;
|
|
std::vector<QColor> color_map_segmentation;
|
|
Item_sdf_map item_sdf_map;
|
|
};
|
|
|
|
void Polyhedron_demo_mesh_segmentation_plugin::init_color_map_sdf()
|
|
{
|
|
color_map_sdf = std::vector<QColor>(256);
|
|
int r = 0, g = 0, b = 255;
|
|
for(int i = 0; i <= 255; ++i)
|
|
{
|
|
if(i > 128 && i <= 192) { r = static_cast<int>( ((i - 128) / (192.0 - 128)) * 255 ); }
|
|
if(i > 0 && i <= 98) { g = static_cast<int>( ((i) / (98.0)) * 255 ); }
|
|
if(i > 191 && i <=255) { g = 255 - static_cast<int>( ((i - 191) / (255.0 - 191)) * 255 ); }
|
|
if(i > 64 && i <= 127) { b = 255 - static_cast<int>( ((i - 64) / (127.0 - 64)) * 255 ); }
|
|
color_map_sdf[i] = QColor(r, g, b);
|
|
}
|
|
}
|
|
|
|
void Polyhedron_demo_mesh_segmentation_plugin::init_color_map_segmentation()
|
|
{
|
|
/*
|
|
color_map_segmentation.push_back(QColor("#fce94f"));
|
|
color_map_segmentation.push_back(QColor("#edd400"));
|
|
color_map_segmentation.push_back(QColor("#c4a000"));
|
|
color_map_segmentation.push_back(QColor("#fcaf3e"));
|
|
color_map_segmentation.push_back(QColor("#f57900"));
|
|
color_map_segmentation.push_back(QColor("#ce5c00"));
|
|
color_map_segmentation.push_back(QColor("#e9b96e"));
|
|
color_map_segmentation.push_back(QColor("#c17d11"));
|
|
color_map_segmentation.push_back(QColor("#8f5902"));
|
|
color_map_segmentation.push_back(QColor("#729fcf"));
|
|
color_map_segmentation.push_back(QColor("#3465a4"));
|
|
color_map_segmentation.push_back(QColor("#204a87"));
|
|
color_map_segmentation.push_back(QColor("#ad7fa8"));
|
|
color_map_segmentation.push_back(QColor("#75507b"));
|
|
color_map_segmentation.push_back(QColor("#5c3566"));
|
|
*/
|
|
|
|
color_map_segmentation.push_back(QColor( 173, 35, 35));
|
|
color_map_segmentation.push_back(QColor( 87, 87, 87));
|
|
color_map_segmentation.push_back(QColor( 42, 75, 215));
|
|
color_map_segmentation.push_back(QColor( 29, 105, 20));
|
|
color_map_segmentation.push_back(QColor( 129, 74, 25));
|
|
color_map_segmentation.push_back(QColor( 129, 38, 192));
|
|
color_map_segmentation.push_back(QColor( 160, 160, 160));
|
|
color_map_segmentation.push_back(QColor( 129, 197, 122));
|
|
color_map_segmentation.push_back(QColor( 157, 175, 255));
|
|
color_map_segmentation.push_back(QColor( 41, 208, 208));
|
|
color_map_segmentation.push_back(QColor( 255, 146, 51));
|
|
color_map_segmentation.push_back(QColor( 255, 238, 51));
|
|
color_map_segmentation.push_back(QColor( 233, 222, 187));
|
|
color_map_segmentation.push_back(QColor( 255, 205, 243));
|
|
|
|
}
|
|
|
|
void Polyhedron_demo_mesh_segmentation_plugin::itemAboutToBeDestroyed(Scene_item* scene_item)
|
|
{
|
|
if(Scene_polyhedron_item* item = qobject_cast<Scene_polyhedron_item*>(scene_item)) {
|
|
item_sdf_map.erase(item);
|
|
}
|
|
}
|
|
|
|
void Polyhedron_demo_mesh_segmentation_plugin::on_actionSegmentation_triggered()
|
|
{ dock_widget->show(); }
|
|
|
|
void Polyhedron_demo_mesh_segmentation_plugin::on_SDF_button_clicked()
|
|
{
|
|
Scene_interface::Item_id index = scene->mainSelectionIndex();
|
|
Scene_polyhedron_item* item = qobject_cast<Scene_polyhedron_item*>(scene->item(index));
|
|
if(!item) { return; }
|
|
QApplication::setOverrideCursor(Qt::WaitCursor);
|
|
|
|
std::size_t number_of_rays = ui_widget.Number_of_rays_spin_box->value();
|
|
double cone_angle = (ui_widget.Cone_angle_spin_box->value() / 180.0) * CGAL_PI;
|
|
bool create_new_item = ui_widget.New_item_check_box->isChecked();
|
|
|
|
Item_sdf_map::iterator pair;
|
|
Scene_polyhedron_item* active_item = item;
|
|
|
|
if(create_new_item) {
|
|
active_item = new Scene_polyhedron_item(*item->polyhedron());
|
|
active_item->setGouraudMode();
|
|
}
|
|
|
|
pair = item_sdf_map.insert(
|
|
std::make_pair(active_item, std::vector<double>()) ).first;
|
|
|
|
check_and_set_ids(pair->first->polyhedron());
|
|
pair->second.resize(item->polyhedron()->size_of_facets(), 0.0);
|
|
Polyhedron_with_id_to_vector_property_map<Polyhedron, double> sdf_pmap(&pair->second);
|
|
QTime time;
|
|
time.start();
|
|
std::pair<double, double> min_max_sdf = sdf_values(*(pair->first->polyhedron()), sdf_pmap, cone_angle, number_of_rays);
|
|
std::cout << "ok (" << time.elapsed() << " ms)" << std::endl;
|
|
|
|
std::cout << "SDF computation is completed. Min-SDF : " << min_max_sdf.first << " " "Max-SDF : " << min_max_sdf.second << std::endl;
|
|
|
|
pair->first->set_color_vector_read_only(true);
|
|
colorize_sdf(pair->first, sdf_pmap, pair->first->color_vector());
|
|
|
|
pair->first->setName(tr("(SDF-%1-%2)").arg(number_of_rays).arg(ui_widget.Cone_angle_spin_box->value()));
|
|
|
|
if(create_new_item) {
|
|
index = scene->addItem(pair->first);
|
|
item->setVisible(false);
|
|
scene->setSelectedItem(index);
|
|
}
|
|
else {
|
|
scene->itemChanged(index);
|
|
}
|
|
|
|
QApplication::restoreOverrideCursor();
|
|
}
|
|
|
|
void Polyhedron_demo_mesh_segmentation_plugin::on_Partition_button_clicked()
|
|
{
|
|
Scene_interface::Item_id index = scene->mainSelectionIndex();
|
|
Scene_polyhedron_item* item = qobject_cast<Scene_polyhedron_item*>(scene->item(index));
|
|
if(!item) { return; }
|
|
|
|
QApplication::setOverrideCursor(Qt::WaitCursor);
|
|
|
|
std::size_t number_of_clusters = ui_widget.Number_of_clusters_spin_box->value();
|
|
double smoothness = ui_widget.Smoothness_spin_box->value();
|
|
std::size_t number_of_rays = ui_widget.Number_of_rays_spin_box->value();
|
|
double cone_angle = (ui_widget.Cone_angle_spin_box->value() / 180.0) * CGAL_PI;
|
|
bool create_new_item = ui_widget.New_item_check_box->isChecked();
|
|
bool extract_segments = ui_widget.Extract_segments_check_box->isChecked();
|
|
|
|
Item_sdf_map::iterator pair;
|
|
if(create_new_item)
|
|
{
|
|
// create new item
|
|
Scene_polyhedron_item* new_item = new Scene_polyhedron_item(*item->polyhedron());
|
|
new_item->setGouraudMode();
|
|
|
|
// copy SDF values of existing poly to new poly
|
|
Item_sdf_map::iterator it = item_sdf_map.find(item);
|
|
const std::vector<double>& sdf_data = it == item_sdf_map.end() ?
|
|
std::vector<double>() : it->second;
|
|
pair = item_sdf_map.insert(std::make_pair(new_item, sdf_data) ).first;
|
|
}
|
|
else
|
|
{
|
|
std::pair<Item_sdf_map::iterator, bool> res =
|
|
item_sdf_map.insert(std::make_pair(item, std::vector<double>()) );
|
|
pair = res.first;
|
|
}
|
|
|
|
check_and_set_ids(pair->first->polyhedron());
|
|
QTime time;
|
|
time.start();
|
|
if(pair->second.empty()) { // SDF values are empty, calculate
|
|
pair->second.resize(pair->first->polyhedron()->size_of_facets(), 0.0);
|
|
Polyhedron_with_id_to_vector_property_map<Polyhedron, double> sdf_pmap(&pair->second);
|
|
sdf_values(*(pair->first->polyhedron()), sdf_pmap, cone_angle, number_of_rays);
|
|
}
|
|
|
|
std::vector<std::size_t> internal_segment_map(pair->first->polyhedron()->size_of_facets());
|
|
Polyhedron_with_id_to_vector_property_map<Polyhedron, std::size_t> segment_pmap(&internal_segment_map);
|
|
Polyhedron_with_id_to_vector_property_map<Polyhedron, double> sdf_pmap(&pair->second);
|
|
|
|
std::size_t nb_segments = segmentation_from_sdf_values(*(pair->first->polyhedron())
|
|
,sdf_pmap, segment_pmap, number_of_clusters, smoothness, extract_segments);
|
|
std::cout << "ok (" << time.elapsed() << " ms)" << std::endl;
|
|
std::cout << "Segmentation is completed. Number of segments : " << nb_segments << std::endl;
|
|
pair->first->set_color_vector_read_only(true);
|
|
|
|
colorize_segmentation(pair->first, segment_pmap, pair->first->color_vector());
|
|
pair->first->setName(tr("(Segmentation-%1-%2)").arg(number_of_clusters).arg(smoothness));
|
|
|
|
if(create_new_item) {
|
|
index = scene->addItem(pair->first);
|
|
item->setVisible(false);
|
|
scene->setSelectedItem(index);
|
|
}
|
|
else {
|
|
scene->itemChanged(index);
|
|
}
|
|
|
|
QApplication::restoreOverrideCursor();
|
|
}
|
|
|
|
void Polyhedron_demo_mesh_segmentation_plugin::check_and_set_ids(Polyhedron* polyhedron)
|
|
{
|
|
Polyhedron::Facet_iterator a_facet = polyhedron->facets_begin();
|
|
Polyhedron::Facet_iterator another_facet = ++polyhedron->facets_begin();
|
|
if(a_facet->id() != another_facet->id()) { return; } // ids are OK
|
|
std::size_t facet_id = 0;
|
|
for(Polyhedron::Facet_iterator facet_it = polyhedron->facets_begin();
|
|
facet_it != polyhedron->facets_end(); ++facet_it, ++facet_id)
|
|
{
|
|
facet_it->id() = facet_id;
|
|
}
|
|
}
|
|
|
|
template<class SDFPropertyMap>
|
|
void Polyhedron_demo_mesh_segmentation_plugin::colorize_sdf(
|
|
Scene_polyhedron_item* item,
|
|
SDFPropertyMap sdf_values,
|
|
std::vector<QColor>& color_vector)
|
|
{
|
|
Polyhedron* polyhedron = item->polyhedron();
|
|
color_vector.clear();
|
|
std::size_t patch_id = 0;
|
|
for(Polyhedron::Facet_iterator facet_it = polyhedron->facets_begin();
|
|
facet_it != polyhedron->facets_end(); ++facet_it, ++patch_id)
|
|
{
|
|
double sdf_value = sdf_values[facet_it];
|
|
int gray_color = static_cast<int>(255 * sdf_value);
|
|
if(gray_color < 0 || gray_color >= 256) {
|
|
color_vector.push_back(QColor::fromRgb(0,0,0));
|
|
}
|
|
else {
|
|
color_vector.push_back(color_map_sdf[gray_color]);
|
|
}
|
|
facet_it->set_patch_id(static_cast<int>(patch_id));
|
|
}
|
|
}
|
|
|
|
template<class SegmentPropertyMap>
|
|
void Polyhedron_demo_mesh_segmentation_plugin::colorize_segmentation(
|
|
Scene_polyhedron_item* item,
|
|
SegmentPropertyMap segment_ids,
|
|
std::vector<QColor>& color_vector)
|
|
{
|
|
Polyhedron* polyhedron = item->polyhedron();
|
|
color_vector.clear();
|
|
std::size_t max_segment = 0;
|
|
for(Polyhedron::Facet_iterator facet_it = polyhedron->facets_begin();
|
|
facet_it != polyhedron->facets_end(); ++facet_it)
|
|
{
|
|
std::size_t segment_id = segment_ids[facet_it];
|
|
facet_it->set_patch_id(static_cast<int>(segment_id));
|
|
max_segment = (std::max)(max_segment, segment_id);
|
|
}
|
|
for(std::size_t i = 0; i <= max_segment; ++i)
|
|
{
|
|
QColor aColor = color_map_segmentation[(max_segment - i) % color_map_segmentation.size()];
|
|
color_vector.push_back(aColor);
|
|
}
|
|
}
|
|
|
|
Q_EXPORT_PLUGIN2(Polyhedron_demo_mesh_segmentation_plugin, Polyhedron_demo_mesh_segmentation_plugin)
|
|
|
|
#include "Polyhedron_demo_mesh_segmentation_plugin.moc"
|