home *** CD-ROM | disk | FTP | other *** search
/ Hackers Magazine 57 / CdHackersMagazineNr57.iso / Software / Multimedia / k3d-setup-0.7.11.0.exe / include / k3d / k3dsdk / mesh_reader.h < prev    next >
Encoding:
C/C++ Source or Header  |  2009-03-12  |  6.7 KB  |  173 lines

  1. #ifndef K3DSDK_MESH_READER_H
  2. #define K3DSDK_MESH_READER_H
  3.  
  4. // K-3D
  5. // Copyright (c) 1995-2008, Timothy M. Shead
  6. //
  7. // Contact: tshead@k-3d.com
  8. //
  9. // This program is free software; you can redistribute it and/or
  10. // modify it under the terms of the GNU General Public
  11. // License as published by the Free Software Foundation; either
  12. // version 2 of the License, or (at your option) any later version.
  13. //
  14. // This program is distributed in the hope that it will be useful,
  15. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  17. // General Public License for more details.
  18. //
  19. // You should have received a copy of the GNU General Public
  20. // License along with this program; if not, write to the Free Software
  21. // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
  22.  
  23. /** \file
  24.     \author Tim Shead (tshead@k3d.com)
  25. */
  26.  
  27. #include "data.h"
  28. #include "hints.h"
  29. #include "k3d-i18n-config.h"
  30. #include "ipipeline_profiler.h"
  31. #include "imesh_source.h"
  32. #include "imesh_storage.h"
  33. #include "measurement.h"
  34. #include "mesh.h"
  35. #include "node.h"
  36. #include "pointer_demand_storage.h"
  37. #include "user_interface.h"
  38. #include "watched_path_property.h"
  39.  
  40. namespace k3d
  41. {
  42.  
  43. /////////////////////////////////////////////////////////////////////////////
  44. // mesh_reader
  45.  
  46. template<typename base_t>
  47. class mesh_reader :
  48.     public base_t,
  49.     public imesh_source,
  50.     public imesh_storage
  51. {
  52. public:
  53.     iproperty& mesh_source_output()
  54.     {
  55.         return m_output_mesh;
  56.     }
  57.  
  58.     void reset_mesh(mesh* const Mesh)
  59.     {
  60.         m_output_mesh.reset(Mesh);
  61.     }
  62.  
  63.     /// Returns a slot that should be connected to input properties to signal that the output mesh has changed
  64.     sigc::slot<void, ihint*> make_reload_mesh_slot()
  65.     {
  66.         return m_output_mesh.make_slot();
  67.     }
  68.  
  69. protected:
  70.     mesh_reader(iplugin_factory& Factory, idocument& Document) :
  71.         base_t(Factory, Document),
  72.         m_file(init_owner(*this) + init_name("file") + init_label(_("File")) + init_description(_("Input file path.")) + init_value(filesystem::path()) + init_path_mode(ipath_property::READ) + init_path_type("")),
  73.         m_center(init_owner(*this) + init_name("center") + init_label(_("Center on Origin")) + init_description(_("Center the output mesh around the origin.")) + init_value(true)),
  74.         m_scale_to_size(init_owner(*this) + init_name("scale_to_size") + init_label(_("Scale to Size")) + init_description(_("Scale the output mesh to fit within a fixed-size bounding-box.")) + init_value(true)),
  75.         m_size(init_owner(*this) + init_name("size") + init_label(_("Size")) + init_description(_("Output mesh size when \"Scale to Size\" is enabled.")) + init_value(10.0) + init_step_increment(0.1) + init_units(typeid(measurement::distance))),
  76.         m_output_mesh(init_owner(*this) + init_name("output_mesh") + init_label(_("Output Mesh")) + init_description("Output mesh"))
  77.     {
  78.         m_file.changed_signal().connect(hint::converter<
  79.             hint::convert<hint::any, hint::none> >(make_reload_mesh_slot()));
  80.         m_center.changed_signal().connect(hint::converter<
  81.             hint::convert<hint::any, hint::none> >(make_reload_mesh_slot()));
  82.         m_scale_to_size.changed_signal().connect(hint::converter<
  83.             hint::convert<hint::any, hint::none> >(make_reload_mesh_slot()));
  84.         m_size.changed_signal().connect(hint::converter<
  85.             hint::convert<hint::any, hint::none> >(make_reload_mesh_slot()));
  86.  
  87.         m_output_mesh.set_update_slot(sigc::mem_fun(*this, &mesh_reader<base_t>::execute));
  88.     }
  89.  
  90.     /// Stores the input file path.
  91.     k3d_data(filesystem::path, immutable_name, change_signal, with_undo, local_storage, no_constraint, watched_path_property, watched_path_serialization) m_file;
  92.     /// Center the output mesh on the origin.
  93.     k3d_data(bool_t, immutable_name, change_signal, with_undo, local_storage, no_constraint, writable_property, with_serialization) m_center;
  94.     /// Scale the output mesh to fit within a fixed-size bounding-box.
  95.     k3d_data(bool_t, immutable_name, change_signal, with_undo, local_storage, no_constraint, writable_property, with_serialization) m_scale_to_size;
  96.     /// Stores the size of the bounding-box used with m_scale_to_size.
  97.     k3d_data(double_t, immutable_name, change_signal, with_undo, local_storage, no_constraint, measurement_property, with_serialization) m_size;
  98.     /// Stores the output mesh, which is created / updated on-demand.
  99.     k3d_data(mesh*, immutable_name, change_signal, no_undo, pointer_demand_storage, no_constraint, read_only_property, no_serialization) m_output_mesh;
  100.  
  101. private:
  102.     /// Called whenever the output mesh has been modified and needs to be updated.
  103.     /// Note that execution is unaffected by the types of hints we've received.
  104.     void execute(const std::vector<ihint*>& Hints, mesh& Mesh)
  105.     {
  106.         // Start with an empty mesh ...
  107.         Mesh = mesh();
  108.  
  109.         const filesystem::path path = m_file.pipeline_value();
  110.         if(path.empty())
  111.             return;
  112.     
  113.         log() << info << "Loading " << path.native_console_string() << " using " << base_t::factory().name() << std::endl;
  114.  
  115.         if(!filesystem::exists(path))
  116.         {
  117.             log() << error << "filesystem path " << path.native_console_string() << " doesn't exist" << std::endl;
  118.             return;
  119.         }
  120.  
  121.         base_t::document().pipeline_profiler().start_execution(*this, "Load Mesh");
  122.         on_load_mesh(path, Mesh);
  123.         base_t::document().pipeline_profiler().finish_execution(*this, "Load Mesh");
  124.  
  125.         base_t::document().pipeline_profiler().start_execution(*this, "Transform Mesh");
  126.  
  127.         if(Mesh.points)
  128.         {
  129.             const bool_t center = m_center.pipeline_value();
  130.             const bool_t scale_to_size = m_scale_to_size.pipeline_value();
  131.  
  132.             bounding_box3 bounding_box;
  133.             matrix4 transformation = identity3();
  134.  
  135.             mesh::points_t& output_points = Mesh.points.writable();
  136.             if(center || scale_to_size)
  137.                 bounding_box = mesh::bounds(output_points);
  138.  
  139.             if(center)
  140.             {
  141.                 transformation = transformation * translate3(vector3(
  142.                     -0.5 * (bounding_box.px + bounding_box.nx),
  143.                     -0.5 * (bounding_box.py + bounding_box.ny),
  144.                     -0.5 * (bounding_box.pz + bounding_box.nz)));
  145.             }
  146.  
  147.             if(scale_to_size)
  148.             {
  149.                 const double_t current_size = std::max(bounding_box.width(), std::max(bounding_box.height(), bounding_box.depth()));
  150.                 if(current_size)
  151.                     transformation = scale3(m_size.pipeline_value() / current_size) * transformation;
  152.             }
  153.  
  154.             const uint_t point_begin = 0;
  155.             const uint_t point_end = output_points.size();
  156.             for(uint_t point = point_begin; point != point_end; ++point)
  157.                 output_points[point] = transformation * output_points[point];
  158.         }
  159.  
  160.         base_t::document().pipeline_profiler().finish_execution(*this, "Transform Mesh");
  161.     }
  162.  
  163.  
  164.     /// Implement this in derived classes to read the input file and produce an output mesh.  Note that the
  165.     /// output mesh will be empty every time this method is called.
  166.     virtual void on_load_mesh(const filesystem::path& Path, mesh& Output) = 0;
  167. };
  168.  
  169. } // namespace k3d
  170.  
  171. #endif // !K3DSDK_MESH_READER_H
  172.  
  173.