Graphite Version 3
An experimental 3D geometry processing program
Loading...
Searching...
No Matches
mesh_CSG.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 2000-2023 Inria
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * * Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * * Neither the name of the ALICE Project-Team nor the names of its
14 * contributors may be used to endorse or promote products derived from this
15 * software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
21 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27 * POSSIBILITY OF SUCH DAMAGE.
28 *
29 * Contact: Bruno Levy
30 *
31 * https://www.inria.fr/fr/bruno-levy
32 *
33 * Inria,
34 * Domaine de Voluceau,
35 * 78150 Le Chesnay - Rocquencourt
36 * FRANCE
37 *
38 */
39
40#ifndef H_GEO_MESH_ALGO_MESH_CSG_H
41#define H_GEO_MESH_ALGO_MESH_CSG_H
42
44#include <geogram/mesh/mesh.h>
47
53namespace GEO {
54
55 class ProgressTask;
56 class Image;
57
61 class GEOGRAM_API CSGMesh : public Mesh, public Counted {
62 public:
63 CSGMesh();
64 ~CSGMesh() override;
70 const Box3d& bbox() const {
71 geo_debug_assert(vertices.nb() == 0 || bbox_initialized());
72 return bbox_;
73 }
74
82 bool bbox_initialized() const;
83
90
98 void append_mesh(const CSGMesh* other, index_t operand = index_t(-1));
99
110 bool may_have_intersections_with(const CSGMesh* other) const {
111 return bboxes_overlap(bbox(), other->bbox());
112 }
113
114 private:
115 Box3d bbox_;
116 };
117
122
127 typedef std::vector< CSGMesh_var > CSGScope;
128
134 class GEOGRAM_API CSGBuilder {
135 public:
137 static constexpr double DEFAULT_FA = 12.0;
138
140 static constexpr double DEFAULT_FS = 2.0;
141
143 static constexpr double DEFAULT_FN = 0.0;
144
145 CSGBuilder();
146
147 /****** Objects **********/
148
149 CSGMesh_var square(vec2 size = vec2(1.0,1.0), bool center=true);
150 CSGMesh_var circle(double r=1.0);
151 CSGMesh_var cube(vec3 size = vec3(1.0, 1.0, 1.0), bool center=true);
152 CSGMesh_var sphere(double r=1.0);
153 CSGMesh_var cylinder(
154 double h=1.0, double r1=1.0, double r2=1.0, bool center=true
155 );
156 CSGMesh_var import(
157 const std::string& filename, const std::string& layer="",
158 index_t timestamp=0,
159 vec2 origin = vec2(0.0, 0.0), vec2 scale = vec2(1.0,1.0)
160 );
161 CSGMesh_var surface(
162 const std::string& filename, bool center, bool invert
163 );
164
165 /****** Instructions ****/
166
176 CSGMesh_var multmatrix(const mat4& M, const CSGScope& scope);
177
183
189
197
204 CSGMesh_var group(const CSGScope& scope) {
205 return union_instr(scope);
206 }
207
214 CSGMesh_var color(vec4 color, const CSGScope& scope);
215
220 CSGMesh_var hull(const CSGScope& scope);
221
234 const CSGScope& scope,
235 double height = 1.0,
236 bool center = true,
237 vec2 scale = vec2(1.0,1.0),
238 index_t slices = 0,
239 double twist = 0.0
240 );
241
242
249 CSGMesh_var rotate_extrude(const CSGScope& scope, double angle = 360.0);
250
257 CSGMesh_var projection(const CSGScope& scope, bool cut);
258
264
265 /****** Parameters ******/
266
272
281 void set_fn(double fn) {
282 fn_ = std::max(fn, 0.0);
283 }
284
291 void set_fs(double fs) {
292 fs_ = std::max(fs,0.01);
293 }
294
301 void set_fa(double fa) {
302 fa_ = std::max(fa,0.01);
303 }
304
311 void set_delaunay(bool x) {
312 delaunay_ = x;
313 }
314
321 detect_intersecting_neighbors_ = x;
322 }
323
333 void set_simplify_coplanar_facets(bool x, double angle_tolerance=0.0) {
334 simplify_coplanar_facets_ = x;
335 coplanar_angle_tolerance_ = angle_tolerance;
336 }
337
345 void set_fast_union(bool x) {
346 fast_union_ = x;
347 }
348
354 void set_verbose(bool x) {
355 verbose_ = x;
356 }
357
363 bool verbose() const {
364 return verbose_;
365 }
366
373 void add_file_path(const std::string& path) {
374 file_path_.push_back(path);
375 }
376
382 file_path_.clear();
383 file_path_.push_back(".");
384 }
385
386 protected:
387
388 bool find_file(std::string& filename);
389
390 void do_CSG(CSGMesh_var mesh, const std::string& boolean_expr);
391
401 CSGMesh_var mesh, const std::string& boolean_expr,
402 bool keep_border_only=false
403 );
404
411 const std::string& filename, const std::string& layer="",
412 index_t timestamp=0
413 );
414
422 Image* load_dat_image(const std::string& file_name);
423
432
442 index_t get_fragments_from_r(double r, double twist = 360.0);
443
444 private:
445 double fn_;
446 double fs_;
447 double fa_;
448 double STL_epsilon_;
449 bool verbose_;
450 index_t max_arity_;
451 std::vector<std::string> file_path_;
452 bool detect_intersecting_neighbors_;
453 bool delaunay_;
454 bool simplify_coplanar_facets_;
455 double coplanar_angle_tolerance_;
456 bool fast_union_;
457 };
458
459 /**************************************************************/
460
465 class GEOGRAM_API CSGCompiler {
466 public:
467
468 CSGCompiler();
469 CSGMesh_var compile_file(const std::string& input_filename);
470 CSGMesh_var compile_string(const std::string& source);
471
477 void set_verbose(bool x) {
478 builder_.set_verbose(x);
479 }
480
486 return builder_;
487 }
488
489 protected:
490
491 /****** Value, Arglist **********************************/
492
497 struct Value {
498 enum Type {NONE, NUMBER, BOOLEAN, ARRAY1D, ARRAY2D, STRING};
499
500 Value();
501 Value(double x);
502 Value(int x);
503 Value(bool x);
504 Value(const std::string& x);
505 std::string to_string() const;
506
507 Type type;
508 bool boolean_val;
509 double number_val;
510 vector<vector<double> > array_val;
511 std::string string_val;
512 };
513
518 class ArgList {
519 public:
520 typedef std::pair<std::string, Value> Arg;
521
522 index_t size() const {
523 return args_.size();
524 }
525
526 const std::string& ith_arg_name(index_t i) const {
527 geo_assert(i < size());
528 return args_[i].first;
529 }
530
531 const Value& ith_arg_val(index_t i) const {
532 geo_assert(i < size());
533 return args_[i].second;
534 }
535
536 void add_arg(const std::string& name, const Value& value);
537 bool has_arg(const std::string& name) const;
538 const Value& get_arg(const std::string& name) const;
539 double get_arg(const std::string& name,double default_value) const;
540 int get_arg(const std::string& name, int default_value) const;
541 bool get_arg(const std::string& name, bool default_value) const;
542 vec2 get_arg(const std::string& name, vec2 default_value) const;
543 vec3 get_arg(const std::string& name, vec3 default_value) const;
544 vec4 get_arg(const std::string& name, vec4 default_value) const;
545 mat4 get_arg(
546 const std::string& name, const mat4& default_value
547 ) const;
548 std::string get_arg(
549 const std::string& name, const std::string& default_value
550 ) const;
551
552 private:
553 vector<Arg> args_;
554 };
555
556
557 /****** Objects *****************************************/
558
559 CSGMesh_var square(const ArgList& args);
560 CSGMesh_var circle(const ArgList& args);
561 CSGMesh_var cube(const ArgList& args);
562 CSGMesh_var sphere(const ArgList& args);
563 CSGMesh_var cylinder(const ArgList& args);
564 CSGMesh_var polyhedron(const ArgList& args);
565 CSGMesh_var polygon(const ArgList& args);
566 CSGMesh_var import(const ArgList& args);
567 CSGMesh_var surface(const ArgList& args);
568
569 /****** Instructions ************************************/
570
571 CSGMesh_var multmatrix(const ArgList& args, const CSGScope& scope);
572 CSGMesh_var resize(const ArgList& args, const CSGScope& scope);
573 CSGMesh_var union_instr(const ArgList& args, const CSGScope& scope);
574 CSGMesh_var intersection(const ArgList& args, const CSGScope& scope);
575 CSGMesh_var difference(const ArgList& args, const CSGScope& scope);
576 CSGMesh_var group(const ArgList& args, const CSGScope& scope);
577 CSGMesh_var color(const ArgList& args, const CSGScope& scope);
578 CSGMesh_var hull(const ArgList& args, const CSGScope& scope);
579 CSGMesh_var linear_extrude(const ArgList& args, const CSGScope& scope);
580 CSGMesh_var rotate_extrude(const ArgList& args, const CSGScope& scope);
581 CSGMesh_var projection(const ArgList& args, const CSGScope& scope);
582
583 /***** Parser *******************************************/
584
585 CSGMesh_var parse_instruction_or_object();
586 CSGMesh_var parse_object();
587 CSGMesh_var parse_instruction();
588 ArgList parse_arg_list();
589 Value parse_value();
590 Value parse_array();
591 bool is_object(const std::string& id) const;
592 bool is_instruction(const std::string& id) const;
593
603 bool is_modifier(int toktype) const;
604
605 /***** Parser internals ********************************/
606
607 struct Token {
608 Token();
609 std::string to_string() const;
610 int type;
611 std::string str_val;
612 int int_val;
613 double double_val;
614 bool boolean_val;
615 };
616
623 void next_token_check(char c);
624
630
636
644
648 int lines() const;
649
653 int line() const;
654
655
660 [[noreturn]] void syntax_error(const char* msg);
661
668 [[noreturn]] void syntax_error(const char* msg, const Token& tok);
669
670 private:
671 std::string filename_;
672 void* lex_;
673 Token lookahead_token_;
674 CSGBuilder builder_;
675
676 typedef CSGMesh_var (CSGCompiler::*object_funptr)(const ArgList& args);
677 typedef CSGMesh_var (CSGCompiler::*instruction_funptr)(
678 const ArgList& args, const CSGScope& scope
679 );
680 std::map<std::string, object_funptr> object_funcs_;
681 std::map<std::string, instruction_funptr> instruction_funcs_;
682 ProgressTask* progress_;
683 index_t lines_;
684 };
685}
686
687#endif
#define geo_assert(x)
Verifies that a condition is met.
Definition assert.h:149
#define geo_debug_assert(x)
Verifies that a condition is met.
Definition assert.h:196
Axis-aligned bounding box.
Definition geometry.h:689
Implements CSG objects and instructions.
Definition mesh_CSG.h:134
index_t get_fragments_from_r(double r, double twist=360.0)
Computes the number of fragments, that is, edges in a polygonal approximation of a circle.
CSGMesh_var difference(const CSGScope &scope)
Computes the intersection between two meshes.
void set_delaunay(bool x)
If set, compute constrained Delaunay triangulation in the intersected triangles. If there are interse...
Definition mesh_CSG.h:311
CSGMesh_var intersection(const CSGScope &scope)
Computes the intersection between two or more meshes.
CSGMesh_var color(vec4 color, const CSGScope &scope)
Groups several meshes into a single one and sets their color.
void set_fn(double fn)
Sets the number of fragments.
Definition mesh_CSG.h:281
void post_process(CSGMesh_var mesh)
Post-processes the result of a previous intersection.
bool verbose() const
Tests wheter verbose mode is set.
Definition mesh_CSG.h:363
CSGMesh_var group(const CSGScope &scope)
synonym for union.
Definition mesh_CSG.h:204
void set_fast_union(bool x)
Sets fast union mode.
Definition mesh_CSG.h:345
void set_fa(double fa)
Sets the minimum angle for a fragment.
Definition mesh_CSG.h:301
void add_file_path(const std::string &path)
Adds a path to the file path.
Definition mesh_CSG.h:373
void set_detect_intersecting_neighbors(bool x)
detect and compute intersections between facets that share a facet or an edge. Set to false if input ...
Definition mesh_CSG.h:320
void set_simplify_coplanar_facets(bool x, double angle_tolerance=0.0)
Specifies whether coplanar facets should be simplified.
Definition mesh_CSG.h:333
CSGMesh_var linear_extrude(const CSGScope &scope, double height=1.0, bool center=true, vec2 scale=vec2(1.0, 1.0), index_t slices=0, double twist=0.0)
Computes a 3D extrusion from a 2D shape.
Image * load_dat_image(const std::string &file_name)
Loads an ascii data file as an image.
void triangulate(CSGMesh_var mesh, const std::string &boolean_expr, bool keep_border_only=false)
Triangulates a 2D mesh.
void set_fs(double fs)
Sets the minimum size for a fragment.
Definition mesh_CSG.h:291
CSGMesh_var multmatrix(const mat4 &M, const CSGScope &scope)
Groups several meshes into a single one and transforms them.
void set_verbose(bool x)
Displays (lots of) additional information.
Definition mesh_CSG.h:354
CSGMesh_var rotate_extrude(const CSGScope &scope, double angle=360.0)
Computes a 3D extrusion from a 2D shape.
void reset_file_path()
Resets the file path to its default value, with only the current directory ".".
Definition mesh_CSG.h:381
CSGMesh_var append(const CSGScope &scope)
Appends all meshes in scope into a unique mesh, without testing for intersections.
CSGMesh_var projection(const CSGScope &scope, bool cut)
Creates a 2D mesh from 3D mesh.
CSGMesh_var union_instr(const CSGScope &scope)
Computes the union of two or more meshes.
CSGMesh_var hull(const CSGScope &scope)
Computes the convex hull of several meshes.
CSGMesh_var import_with_openSCAD(const std::string &filename, const std::string &layer="", index_t timestamp=0)
For the file formats that are not supported by geogram, get help from OpenSCAD to convert them.
void reset_defaults()
Resets defaults value for fn, fs, fa.
A parsed argument list in a .csg file.
Definition mesh_CSG.h:518
Creates meshes from OpenSCAD .csg files.
Definition mesh_CSG.h:465
void syntax_error(const char *msg, const Token &tok)
Throws an exception with an error message.
Token next_token_internal()
Function to actually get the next token from the stream.
bool is_modifier(int toktype) const
Checks if a token corresponds to an instruction or object modifier.
void next_token_check(char c)
Checks that the next token is a given character.
int line() const
Gets the currently parsed line source.
Token lookahead_token()
Gets the next token without any side effect.
void set_verbose(bool x)
Displays (lots of) additional information.
Definition mesh_CSG.h:477
CSGBuilder & builder()
Gets the CSGbuilder.
Definition mesh_CSG.h:485
void syntax_error(const char *msg)
Throws an exception with an error message.
Token next_token()
Gets the next token.
int lines() const
Gets the total number of lines of the currently parsed source.
A Mesh with reference counting and bounding box.
Definition mesh_CSG.h:61
bool bbox_initialized() const
Tests whether the bounding box was initialized.
void append_mesh(const CSGMesh *other, index_t operand=index_t(-1))
Appends a mesh to this mesh.
const Box3d & bbox() const
Gets the bounding box.
Definition mesh_CSG.h:70
bool may_have_intersections_with(const CSGMesh *other) const
Tests whether this mesh may have an intersection with another mesh.
Definition mesh_CSG.h:110
void update_bbox()
Computes the bounding box.
Base class for reference-counted objects.
Definition counted.h:71
An image.
Definition image.h:59
Represents a mesh.
Definition mesh.h:2701
Tracks the progress of a task.
Definition progress.h:240
A smart pointer with reference-counted copy semantics.
Vector with aligned memory allocation.
Definition memory.h:660
Common include file, providing basic definitions. Should be included before anything else by all head...
Types and functions for memory manipulation.
The class that represents a mesh.
Global Vorpaline namespace.
SmartPointer< CSGMesh > CSGMesh_var
A smart pointer to a CSGMesh.
Definition mesh_CSG.h:121
std::vector< CSGMesh_var > CSGScope
A list of CSGMesh.
Definition mesh_CSG.h:127
bool bboxes_overlap(const Box &B1, const Box &B2)
Tests whether two Boxes have a non-empty intersection.
Definition geometry.h:721
geo_index_t index_t
The type for storing and manipulating indices.
Definition numeric.h:329
Pointers with automatic reference counting.
A parsed value in a .csg file.
Definition mesh_CSG.h:497