]> git.xonotic.org Git - xonotic/netradiant.git/blob - radiant/brushmanip.cpp
Merge commit 'b7e5662d0c9342cc42501de697386ef5a96277e6' into master-merge
[xonotic/netradiant.git] / radiant / brushmanip.cpp
1 /*
2    Copyright (C) 1999-2006 Id Software, Inc. and contributors.
3    For a list of contributors, see the accompanying CONTRIBUTORS file.
4
5    This file is part of GtkRadiant.
6
7    GtkRadiant is free software; you can redistribute it and/or modify
8    it under the terms of the GNU General Public License as published by
9    the Free Software Foundation; either version 2 of the License, or
10    (at your option) any later version.
11
12    GtkRadiant is distributed in the hope that it will be useful,
13    but WITHOUT ANY WARRANTY; without even the implied warranty of
14    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15    GNU General Public License for more details.
16
17    You should have received a copy of the GNU General Public License
18    along with GtkRadiant; if not, write to the Free Software
19    Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
20  */
21
22 #include "brushmanip.h"
23
24
25 #include "gtkutil/widget.h"
26 #include "gtkutil/menu.h"
27 #include "gtkmisc.h"
28 #include "brushnode.h"
29 #include "map.h"
30 #include "texwindow.h"
31 #include "gtkdlgs.h"
32 #include "commands.h"
33 #include "mainframe.h"
34 #include "dialog.h"
35 #include "xywindow.h"
36 #include "preferences.h"
37
38 #include <list>
39 #include <gdk/gdkkeysyms.h>
40
41 void Brush_ConstructCuboid( Brush& brush, const AABB& bounds, const char* shader, const TextureProjection& projection ){
42         const unsigned char box[3][2] = { { 0, 1 }, { 2, 0 }, { 1, 2 } };
43         Vector3 mins( vector3_subtracted( bounds.origin, bounds.extents ) );
44         Vector3 maxs( vector3_added( bounds.origin, bounds.extents ) );
45
46         brush.clear();
47         brush.reserve( 6 );
48
49         {
50                 for ( int i = 0; i < 3; ++i )
51                 {
52                         Vector3 planepts1( maxs );
53                         Vector3 planepts2( maxs );
54                         planepts2[box[i][0]] = mins[box[i][0]];
55                         planepts1[box[i][1]] = mins[box[i][1]];
56
57                         brush.addPlane( maxs, planepts1, planepts2, shader, projection );
58                 }
59         }
60         {
61                 for ( int i = 0; i < 3; ++i )
62                 {
63                         Vector3 planepts1( mins );
64                         Vector3 planepts2( mins );
65                         planepts1[box[i][0]] = maxs[box[i][0]];
66                         planepts2[box[i][1]] = maxs[box[i][1]];
67
68                         brush.addPlane( mins, planepts1, planepts2, shader, projection );
69                 }
70         }
71 }
72
73 inline float max_extent( const Vector3& extents ){
74         return std::max( std::max( extents[0], extents[1] ), extents[2] );
75 }
76
77 inline float max_extent_2d( const Vector3& extents, int axis ){
78         switch ( axis )
79         {
80         case 0:
81                 return std::max( extents[1], extents[2] );
82         case 1:
83                 return std::max( extents[0], extents[2] );
84         default:
85                 return std::max( extents[0], extents[1] );
86         }
87 }
88
89 const std::size_t c_brushPrism_minSides = 3;
90 const std::size_t c_brushPrism_maxSides = c_brush_maxFaces - 2;
91 const char* const c_brushPrism_name = "brushPrism";
92
93 void Brush_ConstructPrism( Brush& brush, const AABB& bounds, std::size_t sides, int axis, const char* shader, const TextureProjection& projection ){
94         if ( sides < c_brushPrism_minSides ) {
95                 globalErrorStream() << c_brushPrism_name << ": sides " << Unsigned( sides ) << ": too few sides, minimum is " << Unsigned( c_brushPrism_minSides ) << "\n";
96                 return;
97         }
98         if ( sides > c_brushPrism_maxSides ) {
99                 globalErrorStream() << c_brushPrism_name << ": sides " << Unsigned( sides ) << ": too many sides, maximum is " << Unsigned( c_brushPrism_maxSides ) << "\n";
100                 return;
101         }
102
103         brush.clear();
104         brush.reserve( sides + 2 );
105
106         Vector3 mins( vector3_subtracted( bounds.origin, bounds.extents ) );
107         Vector3 maxs( vector3_added( bounds.origin, bounds.extents ) );
108
109         float radius = max_extent_2d( bounds.extents, axis );
110         const Vector3& mid = bounds.origin;
111         Vector3 planepts[3];
112
113         planepts[2][( axis + 1 ) % 3] = mins[( axis + 1 ) % 3];
114         planepts[2][( axis + 2 ) % 3] = mins[( axis + 2 ) % 3];
115         planepts[2][axis] = maxs[axis];
116         planepts[1][( axis + 1 ) % 3] = maxs[( axis + 1 ) % 3];
117         planepts[1][( axis + 2 ) % 3] = mins[( axis + 2 ) % 3];
118         planepts[1][axis] = maxs[axis];
119         planepts[0][( axis + 1 ) % 3] = maxs[( axis + 1 ) % 3];
120         planepts[0][( axis + 2 ) % 3] = maxs[( axis + 2 ) % 3];
121         planepts[0][axis] = maxs[axis];
122
123         brush.addPlane( planepts[0], planepts[1], planepts[2], shader, projection );
124
125         planepts[0][( axis + 1 ) % 3] = mins[( axis + 1 ) % 3];
126         planepts[0][( axis + 2 ) % 3] = mins[( axis + 2 ) % 3];
127         planepts[0][axis] = mins[axis];
128         planepts[1][( axis + 1 ) % 3] = maxs[( axis + 1 ) % 3];
129         planepts[1][( axis + 2 ) % 3] = mins[( axis + 2 ) % 3];
130         planepts[1][axis] = mins[axis];
131         planepts[2][( axis + 1 ) % 3] = maxs[( axis + 1 ) % 3];
132         planepts[2][( axis + 2 ) % 3] = maxs[( axis + 2 ) % 3];
133         planepts[2][axis] = mins[axis];
134
135         brush.addPlane( planepts[0], planepts[1], planepts[2], shader, projection );
136
137         for ( std::size_t i = 0 ; i < sides ; ++i )
138         {
139                 double sv = sin( i * 3.14159265 * 2 / sides );
140                 double cv = cos( i * 3.14159265 * 2 / sides );
141
142                 planepts[0][( axis + 1 ) % 3] = static_cast<float>( floor( mid[( axis + 1 ) % 3] + radius * cv + 0.5 ) );
143                 planepts[0][( axis + 2 ) % 3] = static_cast<float>( floor( mid[( axis + 2 ) % 3] + radius * sv + 0.5 ) );
144                 planepts[0][axis] = mins[axis];
145
146                 planepts[1][( axis + 1 ) % 3] = planepts[0][( axis + 1 ) % 3];
147                 planepts[1][( axis + 2 ) % 3] = planepts[0][( axis + 2 ) % 3];
148                 planepts[1][axis] = maxs[axis];
149
150                 planepts[2][( axis + 1 ) % 3] = static_cast<float>( floor( planepts[0][( axis + 1 ) % 3] - radius * sv + 0.5 ) );
151                 planepts[2][( axis + 2 ) % 3] = static_cast<float>( floor( planepts[0][( axis + 2 ) % 3] + radius * cv + 0.5 ) );
152                 planepts[2][axis] = maxs[axis];
153
154                 brush.addPlane( planepts[0], planepts[1], planepts[2], shader, projection );
155         }
156 }
157
158 const std::size_t c_brushCone_minSides = 3;
159 const std::size_t c_brushCone_maxSides = 32;
160 const char* const c_brushCone_name = "brushCone";
161
162 void Brush_ConstructCone( Brush& brush, const AABB& bounds, std::size_t sides, const char* shader, const TextureProjection& projection ){
163         if ( sides < c_brushCone_minSides ) {
164                 globalErrorStream() << c_brushCone_name << ": sides " << Unsigned( sides ) << ": too few sides, minimum is " << Unsigned( c_brushCone_minSides ) << "\n";
165                 return;
166         }
167         if ( sides > c_brushCone_maxSides ) {
168                 globalErrorStream() << c_brushCone_name << ": sides " << Unsigned( sides ) << ": too many sides, maximum is " << Unsigned( c_brushCone_maxSides ) << "\n";
169                 return;
170         }
171
172         brush.clear();
173         brush.reserve( sides + 1 );
174
175         Vector3 mins( vector3_subtracted( bounds.origin, bounds.extents ) );
176         Vector3 maxs( vector3_added( bounds.origin, bounds.extents ) );
177
178         float radius = max_extent( bounds.extents );
179         const Vector3& mid = bounds.origin;
180         Vector3 planepts[3];
181
182         planepts[0][0] = mins[0]; planepts[0][1] = mins[1]; planepts[0][2] = mins[2];
183         planepts[1][0] = maxs[0]; planepts[1][1] = mins[1]; planepts[1][2] = mins[2];
184         planepts[2][0] = maxs[0]; planepts[2][1] = maxs[1]; planepts[2][2] = mins[2];
185
186         brush.addPlane( planepts[0], planepts[1], planepts[2], shader, projection );
187
188         for ( std::size_t i = 0 ; i < sides ; ++i )
189         {
190                 double sv = sin( i * 3.14159265 * 2 / sides );
191                 double cv = cos( i * 3.14159265 * 2 / sides );
192
193                 planepts[0][0] = static_cast<float>( floor( mid[0] + radius * cv + 0.5 ) );
194                 planepts[0][1] = static_cast<float>( floor( mid[1] + radius * sv + 0.5 ) );
195                 planepts[0][2] = mins[2];
196
197                 planepts[1][0] = mid[0];
198                 planepts[1][1] = mid[1];
199                 planepts[1][2] = maxs[2];
200
201                 planepts[2][0] = static_cast<float>( floor( planepts[0][0] - radius * sv + 0.5 ) );
202                 planepts[2][1] = static_cast<float>( floor( planepts[0][1] + radius * cv + 0.5 ) );
203                 planepts[2][2] = maxs[2];
204
205                 brush.addPlane( planepts[0], planepts[1], planepts[2], shader, projection );
206         }
207 }
208
209 const std::size_t c_brushSphere_minSides = 3;
210 const std::size_t c_brushSphere_maxSides = 31;
211 const char* const c_brushSphere_name = "brushSphere";
212
213 void Brush_ConstructSphere( Brush& brush, const AABB& bounds, std::size_t sides, const char* shader, const TextureProjection& projection ){
214         if ( sides < c_brushSphere_minSides ) {
215                 globalErrorStream() << c_brushSphere_name << ": sides " << Unsigned( sides ) << ": too few sides, minimum is " << Unsigned( c_brushSphere_minSides ) << "\n";
216                 return;
217         }
218         if ( sides > c_brushSphere_maxSides ) {
219                 globalErrorStream() << c_brushSphere_name << ": sides " << Unsigned( sides ) << ": too many sides, maximum is " << Unsigned( c_brushSphere_maxSides ) << "\n";
220                 return;
221         }
222
223         brush.clear();
224         brush.reserve( sides * sides );
225
226         float radius = max_extent( bounds.extents );
227         const Vector3& mid = bounds.origin;
228         Vector3 planepts[3];
229
230         double dt = 2 * c_pi / sides;
231         double dp = c_pi / sides;
232         for ( std::size_t i = 0; i < sides; i++ )
233         {
234                 for ( std::size_t j = 0; j < sides - 1; j++ )
235                 {
236                         double t = i * dt;
237                         double p = float(j * dp - c_pi / 2);
238
239                         planepts[0] = vector3_added( mid, vector3_scaled( vector3_for_spherical( t, p ), radius ) );
240                         planepts[1] = vector3_added( mid, vector3_scaled( vector3_for_spherical( t, p + dp ), radius ) );
241                         planepts[2] = vector3_added( mid, vector3_scaled( vector3_for_spherical( t + dt, p + dp ), radius ) );
242
243                         brush.addPlane( planepts[0], planepts[1], planepts[2], shader, projection );
244                 }
245         }
246
247         {
248                 double p = ( sides - 1 ) * dp - c_pi / 2;
249                 for ( std::size_t i = 0; i < sides; i++ )
250                 {
251                         double t = i * dt;
252
253                         planepts[0] = vector3_added( mid, vector3_scaled( vector3_for_spherical( t, p ), radius ) );
254                         planepts[1] = vector3_added( mid, vector3_scaled( vector3_for_spherical( t + dt, p + dp ), radius ) );
255                         planepts[2] = vector3_added( mid, vector3_scaled( vector3_for_spherical( t + dt, p ), radius ) );
256
257                         brush.addPlane( planepts[0], planepts[1], planepts[2], shader, projection );
258                 }
259         }
260 }
261
262 const std::size_t c_brushRock_minSides = 10;
263 const std::size_t c_brushRock_maxSides = 1000;
264 const char* const c_brushRock_name = "brushRock";
265
266 void Brush_ConstructRock( Brush& brush, const AABB& bounds, std::size_t sides, const char* shader, const TextureProjection& projection ){
267         if ( sides < c_brushRock_minSides ) {
268                 globalErrorStream() << c_brushRock_name << ": sides " << Unsigned( sides ) << ": too few sides, minimum is " << Unsigned( c_brushRock_minSides ) << "\n";
269                 return;
270         }
271         if ( sides > c_brushRock_maxSides ) {
272                 globalErrorStream() << c_brushRock_name << ": sides " << Unsigned( sides ) << ": too many sides, maximum is " << Unsigned( c_brushRock_maxSides ) << "\n";
273                 return;
274         }
275
276         brush.clear();
277         brush.reserve( sides * sides );
278
279         float radius = max_extent( bounds.extents );
280         const Vector3& mid = bounds.origin;
281         Vector3 planepts[3];
282
283         for ( std::size_t j = 0; j < sides; j++ )
284         {
285                 planepts[0][0] = rand() - ( RAND_MAX / 2 );
286                 planepts[0][1] = rand() - ( RAND_MAX / 2 );
287                 planepts[0][2] = rand() - ( RAND_MAX / 2 );
288                 vector3_normalise( planepts[0] );
289
290                 // find two vectors that are perpendicular to planepts[0]
291                 ComputeAxisBase( planepts[0], planepts[1], planepts[2] );
292
293                 planepts[0] = vector3_added( mid, vector3_scaled( planepts[0], radius ) );
294                 planepts[1] = vector3_added( planepts[0], vector3_scaled( planepts[1], radius ) );
295                 planepts[2] = vector3_added( planepts[0], vector3_scaled( planepts[2], radius ) );
296
297 #if 0
298                 // make sure the orientation is right
299                 if ( vector3_dot( vector3_subtracted( planepts[0], mid ), vector3_cross( vector3_subtracted( planepts[1], mid ), vector3_subtracted( planepts[2], mid ) ) ) > 0 ) {
300                         Vector3 h;
301                         h = planepts[1];
302                         planepts[1] = planepts[2];
303                         planepts[2] = h;
304                         globalOutputStream() << "flip\n";
305                 }
306                 else{
307                         globalOutputStream() << "no flip\n";
308                 }
309 #endif
310
311                 brush.addPlane( planepts[0], planepts[1], planepts[2], shader, projection );
312         }
313 }
314
315 int GetViewAxis(){
316         switch ( GlobalXYWnd_getCurrentViewType() )
317         {
318         case XY:
319                 return 2;
320         case XZ:
321                 return 1;
322         case YZ:
323                 return 0;
324         }
325         return 2;
326 }
327
328 void Brush_ConstructPrefab( Brush& brush, EBrushPrefab type, const AABB& bounds, std::size_t sides, const char* shader, const TextureProjection& projection ){
329         switch ( type )
330         {
331         case eBrushCuboid:
332         {
333                 UndoableCommand undo( "brushCuboid" );
334
335                 Brush_ConstructCuboid( brush, bounds, shader, projection );
336         }
337         break;
338         case eBrushPrism:
339         {
340                 int axis = GetViewAxis();
341                 StringOutputStream command;
342                 command << c_brushPrism_name << " -sides " << Unsigned( sides ) << " -axis " << axis;
343                 UndoableCommand undo( command.c_str() );
344
345                 Brush_ConstructPrism( brush, bounds, sides, axis, shader, projection );
346         }
347         break;
348         case eBrushCone:
349         {
350                 StringOutputStream command;
351                 command << c_brushCone_name << " -sides " << Unsigned( sides );
352                 UndoableCommand undo( command.c_str() );
353
354                 Brush_ConstructCone( brush, bounds, sides, shader, projection );
355         }
356         break;
357         case eBrushSphere:
358         {
359                 StringOutputStream command;
360                 command << c_brushSphere_name << " -sides " << Unsigned( sides );
361                 UndoableCommand undo( command.c_str() );
362
363                 Brush_ConstructSphere( brush, bounds, sides, shader, projection );
364         }
365         break;
366         case eBrushRock:
367         {
368                 StringOutputStream command;
369                 command << c_brushRock_name << " -sides " << Unsigned( sides );
370                 UndoableCommand undo( command.c_str() );
371
372                 Brush_ConstructRock( brush, bounds, sides, shader, projection );
373         }
374         break;
375         }
376 }
377
378
379 void ConstructRegionBrushes( scene::Node* brushes[6], const Vector3& region_mins, const Vector3& region_maxs ){
380         {
381                 // set mins
382                 Vector3 mins( region_mins[0] - 32, region_mins[1] - 32, region_mins[2] - 32 );
383
384                 // vary maxs
385                 for ( std::size_t i = 0; i < 3; i++ )
386                 {
387                         Vector3 maxs( region_maxs[0] + 32, region_maxs[1] + 32, region_maxs[2] + 32 );
388                         maxs[i] = region_mins[i];
389                         Brush_ConstructCuboid( *Node_getBrush( *brushes[i] ), aabb_for_minmax( mins, maxs ), texdef_name_default(), TextureProjection() );
390                 }
391         }
392
393         {
394                 // set maxs
395                 Vector3 maxs( region_maxs[0] + 32, region_maxs[1] + 32, region_maxs[2] + 32 );
396
397                 // vary mins
398                 for ( std::size_t i = 0; i < 3; i++ )
399                 {
400                         Vector3 mins( region_mins[0] - 32, region_mins[1] - 32, region_mins[2] - 32 );
401                         mins[i] = region_maxs[i];
402                         Brush_ConstructCuboid( *Node_getBrush( *brushes[i + 3] ), aabb_for_minmax( mins, maxs ), texdef_name_default(), TextureProjection() );
403                 }
404         }
405 }
406
407
408 void Scene_BrushSetTexdef_Selected( scene::Graph& graph, const TextureProjection& projection ){
409         Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
410                 face.SetTexdef(projection);
411         });
412         SceneChangeNotify();
413 }
414
415 void Scene_BrushSetTexdef_Component_Selected( scene::Graph& graph, const TextureProjection& projection ){
416         Scene_ForEachSelectedBrushFace(graph, [&](Face &face) {
417                 face.SetTexdef(projection);
418         });
419         SceneChangeNotify();
420 }
421
422
423 void Scene_BrushSetFlags_Selected( scene::Graph& graph, const ContentsFlagsValue& flags ){
424         Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
425                 face.SetFlags(flags);
426         });
427         SceneChangeNotify();
428 }
429
430 void Scene_BrushSetFlags_Component_Selected( scene::Graph& graph, const ContentsFlagsValue& flags ){
431         Scene_ForEachSelectedBrushFace(graph, [&](Face &face) {
432                 face.SetFlags(flags);
433         });
434         SceneChangeNotify();
435 }
436
437 void Scene_BrushShiftTexdef_Selected( scene::Graph& graph, float s, float t ){
438         Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
439                 face.ShiftTexdef(s, t);
440         });
441         SceneChangeNotify();
442 }
443
444 void Scene_BrushShiftTexdef_Component_Selected( scene::Graph& graph, float s, float t ){
445         Scene_ForEachSelectedBrushFace(graph, [&](Face &face) {
446                 face.ShiftTexdef(s, t);
447         });
448         SceneChangeNotify();
449 }
450
451 void Scene_BrushScaleTexdef_Selected( scene::Graph& graph, float s, float t ){
452         Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
453                 face.ScaleTexdef(s, t);
454         });
455         SceneChangeNotify();
456 }
457
458 void Scene_BrushScaleTexdef_Component_Selected( scene::Graph& graph, float s, float t ){
459         Scene_ForEachSelectedBrushFace(graph, [&](Face &face) {
460                 face.ScaleTexdef(s, t);
461         });
462         SceneChangeNotify();
463 }
464
465 void Scene_BrushRotateTexdef_Selected( scene::Graph& graph, float angle ){
466         Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
467                 face.RotateTexdef(angle);
468         });
469         SceneChangeNotify();
470 }
471
472 void Scene_BrushRotateTexdef_Component_Selected( scene::Graph& graph, float angle ){
473         Scene_ForEachSelectedBrushFace(graph, [&](Face &face) {
474                 face.RotateTexdef(angle);
475         });
476         SceneChangeNotify();
477 }
478
479
480 void Scene_BrushSetShader_Selected( scene::Graph& graph, const char* name ){
481         Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
482                 face.SetShader(name);
483         });
484         SceneChangeNotify();
485 }
486
487 void Scene_BrushSetShader_Component_Selected( scene::Graph& graph, const char* name ){
488         Scene_ForEachSelectedBrushFace(graph, [&](Face &face) {
489                 face.SetShader(name);
490         });
491         SceneChangeNotify();
492 }
493
494 void Scene_BrushSetDetail_Selected( scene::Graph& graph, bool detail ){
495         Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
496                 face.setDetail(detail);
497         });
498         SceneChangeNotify();
499 }
500
501 bool Face_FindReplaceShader( Face& face, const char* find, const char* replace ){
502         if ( shader_equal( face.GetShader(), find ) ) {
503                 face.SetShader( replace );
504                 return true;
505         }
506         return false;
507 }
508
509 bool DoingSearch( const char *repl ){
510         return ( repl == NULL || ( strcmp( "textures/", repl ) == 0 ) );
511 }
512
513 void Scene_BrushFindReplaceShader( scene::Graph& graph, const char* find, const char* replace ){
514         if ( DoingSearch( replace ) ) {
515                 Scene_ForEachBrush_ForEachFaceInstance(graph, [&](FaceInstance &faceinst) {
516                         if (shader_equal(faceinst.getFace().GetShader(), find)) {
517                                 faceinst.setSelected(SelectionSystem::eFace, true);
518                         }
519                 });
520         }
521         else
522         {
523                 Scene_ForEachBrush_ForEachFace(graph, [&](Face &face) { Face_FindReplaceShader(face, find, replace); });
524         }
525 }
526
527 void Scene_BrushFindReplaceShader_Selected( scene::Graph& graph, const char* find, const char* replace ){
528         if ( DoingSearch( replace ) ) {
529                 Scene_ForEachSelectedBrush_ForEachFaceInstance(graph, [&](FaceInstance &faceinst) {
530                         if (shader_equal(faceinst.getFace().GetShader(), find)) {
531                                 faceinst.setSelected(SelectionSystem::eFace, true);
532                         }
533                 });
534         }
535         else
536         {
537                 Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
538                         Face_FindReplaceShader(face, find, replace);
539                 });
540         }
541 }
542
543 // TODO: find for components
544 // d1223m: dont even know what they are...
545 void Scene_BrushFindReplaceShader_Component_Selected( scene::Graph& graph, const char* find, const char* replace ){
546         if ( DoingSearch( replace ) ) {
547
548         }
549         else
550         {
551                 Scene_ForEachSelectedBrushFace(graph, [&](Face &face) {
552                         Face_FindReplaceShader(face, find, replace);
553                 });
554         }
555 }
556
557
558 void Scene_BrushFitTexture_Selected( scene::Graph& graph, float s_repeat, float t_repeat ){
559         Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
560                 face.FitTexture(s_repeat, t_repeat);
561         });
562         SceneChangeNotify();
563 }
564
565 void Scene_BrushFitTexture_Component_Selected( scene::Graph& graph, float s_repeat, float t_repeat ){
566         Scene_ForEachSelectedBrushFace(graph, [&](Face &face) {
567                 face.FitTexture(s_repeat, t_repeat);
568         });
569         SceneChangeNotify();
570 }
571
572 TextureProjection g_defaultTextureProjection;
573
574 const TextureProjection& TextureTransform_getDefault(){
575         TexDef_Construct_Default( g_defaultTextureProjection );
576         return g_defaultTextureProjection;
577 }
578
579 void Scene_BrushConstructPrefab( scene::Graph& graph, EBrushPrefab type, std::size_t sides, const char* shader ){
580         if ( GlobalSelectionSystem().countSelected() != 0 ) {
581                 const scene::Path& path = GlobalSelectionSystem().ultimateSelected().path();
582
583                 Brush* brush = Node_getBrush( path.top() );
584                 if ( brush != 0 ) {
585                         AABB bounds = brush->localAABB(); // copy bounds because the brush will be modified
586                         Brush_ConstructPrefab( *brush, type, bounds, sides, shader, TextureTransform_getDefault() );
587                         SceneChangeNotify();
588                 }
589         }
590 }
591
592 void Scene_BrushResize_Selected( scene::Graph& graph, const AABB& bounds, const char* shader ){
593         if ( GlobalSelectionSystem().countSelected() != 0 ) {
594                 const scene::Path& path = GlobalSelectionSystem().ultimateSelected().path();
595
596                 Brush* brush = Node_getBrush( path.top() );
597                 if ( brush != 0 ) {
598                         Brush_ConstructCuboid( *brush, bounds, shader, TextureTransform_getDefault() );
599                         SceneChangeNotify();
600                 }
601         }
602 }
603
604 bool Brush_hasShader( const Brush& brush, const char* name ){
605         for ( Brush::const_iterator i = brush.begin(); i != brush.end(); ++i )
606         {
607                 if ( shader_equal( ( *i )->GetShader(), name ) ) {
608                         return true;
609                 }
610         }
611         return false;
612 }
613
614 class BrushSelectByShaderWalker : public scene::Graph::Walker
615 {
616 const char* m_name;
617 public:
618 BrushSelectByShaderWalker( const char* name )
619         : m_name( name ){
620 }
621
622 bool pre( const scene::Path& path, scene::Instance& instance ) const {
623         if ( path.top().get().visible() ) {
624                 Brush* brush = Node_getBrush( path.top() );
625                 if ( brush != 0 && Brush_hasShader( *brush, m_name ) ) {
626                         Instance_getSelectable( instance )->setSelected( true );
627                 }
628         }
629         else{
630                 return false;
631         }
632         return true;
633 }
634 };
635
636 void Scene_BrushSelectByShader( scene::Graph& graph, const char* name ){
637         graph.traverse( BrushSelectByShaderWalker( name ) );
638 }
639
640 void Scene_BrushSelectByShader_Component( scene::Graph& graph, const char* name ){
641         Scene_ForEachSelectedBrush_ForEachFaceInstance(graph, [&](FaceInstance &face) {
642                 printf("checking %s = %s\n", face.getFace().GetShader(), name);
643                 if (shader_equal(face.getFace().GetShader(), name)) {
644                 face.setSelected( SelectionSystem::eFace, true );
645         }
646         });
647 }
648
649 void Scene_BrushGetTexdef_Selected( scene::Graph& graph, TextureProjection& projection ){
650         bool done = false;
651         Scene_ForEachSelectedBrush_ForEachFace(graph, [&](Face &face) {
652                 if (!done) {
653                         done = true;
654                         face.GetTexdef(projection);
655 }
656         });
657 }
658
659 void Scene_BrushGetTexdef_Component_Selected( scene::Graph& graph, TextureProjection& projection ){
660 #if 1
661         if ( !g_SelectedFaceInstances.empty() ) {
662                 FaceInstance& faceInstance = g_SelectedFaceInstances.last();
663                 faceInstance.getFace().GetTexdef( projection );
664         }
665 #else
666         FaceGetTexdef visitor( projection );
667         Scene_ForEachSelectedBrushFace( graph, visitor );
668 #endif
669 }
670
671 void Scene_BrushGetShaderSize_Component_Selected( scene::Graph& graph, size_t& width, size_t& height ){
672         if ( !g_SelectedFaceInstances.empty() ) {
673                 FaceInstance& faceInstance = g_SelectedFaceInstances.last();
674                 width = faceInstance.getFace().getShader().width();
675                 height = faceInstance.getFace().getShader().height();
676         }
677 }
678
679
680 void Scene_BrushGetFlags_Selected( scene::Graph& graph, ContentsFlagsValue& flags ){
681 #if 1
682         if ( GlobalSelectionSystem().countSelected() != 0 ) {
683                 BrushInstance* brush = Instance_getBrush( GlobalSelectionSystem().ultimateSelected() );
684                 if ( brush != 0 ) {
685                         bool done = false;
686                         Brush_forEachFace(*brush, [&](Face &face) {
687                                 if (!done) {
688                                         done = true;
689                                         face.GetFlags(flags);
690                                 }
691                         });
692                 }
693         }
694 #else
695         Scene_ForEachSelectedBrush_ForEachFace( graph, FaceGetFlags( flags ) );
696 #endif
697 }
698
699 void Scene_BrushGetFlags_Component_Selected( scene::Graph& graph, ContentsFlagsValue& flags ){
700 #if 1
701         if ( !g_SelectedFaceInstances.empty() ) {
702                 FaceInstance& faceInstance = g_SelectedFaceInstances.last();
703                 faceInstance.getFace().GetFlags( flags );
704         }
705 #else
706         Scene_ForEachSelectedBrushFace( graph, FaceGetFlags( flags ) );
707 #endif
708 }
709
710
711 void Scene_BrushGetShader_Selected( scene::Graph& graph, CopiedString& shader ){
712 #if 1
713         if ( GlobalSelectionSystem().countSelected() != 0 ) {
714                 BrushInstance* brush = Instance_getBrush( GlobalSelectionSystem().ultimateSelected() );
715                 if ( brush != 0 ) {
716                         bool done = false;
717                         Brush_forEachFace(*brush, [&](Face &face) {
718                                 if (!done) {
719                                         done = true;
720                                         shader = face.GetShader();
721                                 }
722                         });
723                 }
724         }
725 #else
726         Scene_ForEachSelectedBrush_ForEachFace( graph, FaceGetShader( shader ) );
727 #endif
728 }
729
730 void Scene_BrushGetShader_Component_Selected( scene::Graph& graph, CopiedString& shader ){
731 #if 1
732         if ( !g_SelectedFaceInstances.empty() ) {
733                 FaceInstance& faceInstance = g_SelectedFaceInstances.last();
734                 shader = faceInstance.getFace().GetShader();
735         }
736 #else
737         FaceGetShader visitor( shader );
738         Scene_ForEachSelectedBrushFace( graph, visitor );
739 #endif
740 }
741
742
743 class filter_face_shader : public FaceFilter
744 {
745 const char* m_shader;
746 public:
747 filter_face_shader( const char* shader ) : m_shader( shader ){
748 }
749
750 bool filter( const Face& face ) const {
751         return shader_equal( face.GetShader(), m_shader );
752 }
753 };
754
755 class filter_face_shader_prefix : public FaceFilter
756 {
757 const char* m_prefix;
758 public:
759 filter_face_shader_prefix( const char* prefix ) : m_prefix( prefix ){
760 }
761
762 bool filter( const Face& face ) const {
763         return shader_equal_n( face.GetShader(), m_prefix, strlen( m_prefix ) );
764 }
765 };
766
767 class filter_face_flags : public FaceFilter
768 {
769 int m_flags;
770 public:
771 filter_face_flags( int flags ) : m_flags( flags ){
772 }
773
774 bool filter( const Face& face ) const {
775         return ( face.getShader().shaderFlags() & m_flags ) != 0;
776 }
777 };
778
779 class filter_face_contents : public FaceFilter
780 {
781 int m_contents;
782 public:
783 filter_face_contents( int contents ) : m_contents( contents ){
784 }
785
786 bool filter( const Face& face ) const {
787         return ( face.getShader().m_flags.m_contentFlags & m_contents ) != 0;
788 }
789 };
790
791
792
793 class filter_brush_any_face : public BrushFilter
794 {
795 FaceFilter* m_filter;
796 public:
797 filter_brush_any_face( FaceFilter* filter ) : m_filter( filter ){
798 }
799
800 bool filter( const Brush& brush ) const {
801         bool filtered = false;
802         Brush_forEachFace(brush, [&](Face &face) {
803                 if (m_filter->filter(face)) {
804                         filtered = true;
805                 }
806         });
807         return filtered;
808 }
809 };
810
811 class filter_brush_all_faces : public BrushFilter
812 {
813 FaceFilter* m_filter;
814 public:
815 filter_brush_all_faces( FaceFilter* filter ) : m_filter( filter ){
816 }
817 bool filter( const Brush& brush ) const {
818         bool filtered = true;
819         Brush_forEachFace(brush, [&](Face &face) {
820                 if (!m_filter->filter(face)) {
821                         filtered = false;
822                 }
823         });
824         return filtered;
825 }
826 };
827
828
829 filter_face_flags g_filter_face_clip( QER_CLIP );
830 filter_brush_all_faces g_filter_brush_clip( &g_filter_face_clip );
831
832 filter_face_shader g_filter_face_clip_q2( "textures/clip" );
833 filter_brush_all_faces g_filter_brush_clip_q2( &g_filter_face_clip_q2 );
834
835 filter_face_shader g_filter_face_weapclip( "textures/common/weapclip" );
836 filter_brush_all_faces g_filter_brush_weapclip( &g_filter_face_weapclip );
837
838 filter_face_shader g_filter_face_commonclip( "textures/common/clip" );
839 filter_brush_all_faces g_filter_brush_commonclip( &g_filter_face_commonclip );
840
841 filter_face_shader g_filter_face_fullclip( "textures/common/fullclip" );
842 filter_brush_all_faces g_filter_brush_fullclip( &g_filter_face_fullclip );
843
844 filter_face_shader g_filter_face_botclip( "textures/common/botclip" );
845 filter_brush_all_faces g_filter_brush_botclip( &g_filter_face_botclip );
846
847 filter_face_shader_prefix g_filter_face_caulk( "textures/common/caulk" );
848 filter_brush_all_faces g_filter_brush_caulk( &g_filter_face_caulk );
849
850 filter_face_shader_prefix g_filter_face_caulk_ja( "textures/system/caulk" );
851 filter_brush_all_faces g_filter_brush_caulk_ja( &g_filter_face_caulk_ja );
852
853 filter_face_flags g_filter_face_liquids( QER_LIQUID );
854 filter_brush_any_face g_filter_brush_liquids( &g_filter_face_liquids );
855
856 filter_face_shader_prefix g_filter_face_liquidsdir( "textures/liquids/" );
857 filter_brush_any_face g_filter_brush_liquidsdir( &g_filter_face_liquidsdir );
858
859 filter_face_shader g_filter_face_hint( "textures/common/hint" );
860 filter_brush_any_face g_filter_brush_hint( &g_filter_face_hint );
861
862 filter_face_shader g_filter_face_hintlocal( "textures/common/hintlocal" );
863 filter_brush_any_face g_filter_brush_hintlocal( &g_filter_face_hintlocal );
864
865 filter_face_shader g_filter_face_hint_q2( "textures/hint" );
866 filter_brush_any_face g_filter_brush_hint_q2( &g_filter_face_hint_q2 );
867
868 filter_face_shader g_filter_face_hint_ja( "textures/system/hint" );
869 filter_brush_any_face g_filter_brush_hint_ja( &g_filter_face_hint_ja );
870
871 filter_face_shader g_filter_face_subtlehint( "textures/common/subtlehint" );
872 filter_brush_any_face g_filter_brush_subtlehint( &g_filter_face_subtlehint );
873
874 filter_face_shader g_filter_face_areaportal( "textures/common/areaportal" );
875 filter_brush_any_face g_filter_brush_areaportal( &g_filter_face_areaportal );
876
877 filter_face_shader g_filter_face_visportal( "textures/editor/visportal" );
878 filter_brush_any_face g_filter_brush_visportal( &g_filter_face_visportal );
879
880 filter_face_shader g_filter_face_clusterportal( "textures/common/clusterportal" );
881 filter_brush_all_faces g_filter_brush_clusterportal( &g_filter_face_clusterportal );
882
883 filter_face_shader g_filter_face_lightgrid( "textures/common/lightgrid" );
884 filter_brush_all_faces g_filter_brush_lightgrid( &g_filter_face_lightgrid );
885
886 filter_face_flags g_filter_face_translucent( QER_TRANS | QER_ALPHATEST );
887 filter_brush_any_face g_filter_brush_translucent( &g_filter_face_translucent );
888
889 filter_face_contents g_filter_face_detail( BRUSH_DETAIL_MASK );
890 filter_brush_all_faces g_filter_brush_detail( &g_filter_face_detail );
891
892 filter_face_shader_prefix g_filter_face_decals( "textures/decals/" );
893 filter_brush_any_face g_filter_brush_decals( &g_filter_face_decals );
894
895
896 void BrushFilters_construct(){
897         add_brush_filter( g_filter_brush_clip, EXCLUDE_CLIP );
898         add_brush_filter( g_filter_brush_clip_q2, EXCLUDE_CLIP );
899         add_brush_filter( g_filter_brush_weapclip, EXCLUDE_CLIP );
900         add_brush_filter( g_filter_brush_fullclip, EXCLUDE_CLIP );
901         add_brush_filter( g_filter_brush_commonclip, EXCLUDE_CLIP );
902         add_brush_filter( g_filter_brush_botclip, EXCLUDE_BOTCLIP );
903         add_brush_filter( g_filter_brush_caulk, EXCLUDE_CAULK );
904         add_brush_filter( g_filter_brush_caulk_ja, EXCLUDE_CAULK );
905         add_face_filter( g_filter_face_caulk, EXCLUDE_CAULK );
906         add_face_filter( g_filter_face_caulk_ja, EXCLUDE_CAULK );
907         add_brush_filter( g_filter_brush_liquids, EXCLUDE_LIQUIDS );
908         add_brush_filter( g_filter_brush_liquidsdir, EXCLUDE_LIQUIDS );
909         add_brush_filter( g_filter_brush_hint, EXCLUDE_HINTSSKIPS );
910         add_brush_filter( g_filter_brush_hintlocal, EXCLUDE_HINTSSKIPS );
911         add_brush_filter( g_filter_brush_hint_q2, EXCLUDE_HINTSSKIPS );
912         add_brush_filter( g_filter_brush_hint_ja, EXCLUDE_HINTSSKIPS );
913         add_brush_filter( g_filter_brush_subtlehint, EXCLUDE_HINTSSKIPS );
914         add_brush_filter( g_filter_brush_clusterportal, EXCLUDE_CLUSTERPORTALS );
915         add_brush_filter( g_filter_brush_visportal, EXCLUDE_VISPORTALS );
916         add_brush_filter( g_filter_brush_areaportal, EXCLUDE_AREAPORTALS );
917         add_brush_filter( g_filter_brush_translucent, EXCLUDE_TRANSLUCENT );
918         add_brush_filter( g_filter_brush_detail, EXCLUDE_DETAILS );
919         add_brush_filter( g_filter_brush_detail, EXCLUDE_STRUCTURAL, true );
920         add_brush_filter( g_filter_brush_lightgrid, EXCLUDE_LIGHTGRID );
921         add_brush_filter( g_filter_brush_decals, EXCLUDE_DECALS );
922 }
923
924 #if 0
925
926 void normalquantisation_draw(){
927         glPointSize( 1 );
928         glBegin( GL_POINTS );
929         for ( std::size_t i = 0; i <= c_quantise_normal; ++i )
930         {
931                 for ( std::size_t j = 0; j <= c_quantise_normal; ++j )
932                 {
933                         Normal3f vertex( normal3f_normalised( Normal3f(
934                                                                                                           static_cast<float>( c_quantise_normal - j - i ),
935                                                                                                           static_cast<float>( i ),
936                                                                                                           static_cast<float>( j )
937                                                                                                           ) ) );
938                         VectorScale( normal3f_to_array( vertex ), 64.f, normal3f_to_array( vertex ) );
939                         glVertex3fv( normal3f_to_array( vertex ) );
940                         vertex.x = -vertex.x;
941                         glVertex3fv( normal3f_to_array( vertex ) );
942                 }
943         }
944         glEnd();
945 }
946
947 class RenderableNormalQuantisation : public OpenGLRenderable
948 {
949 public:
950 void render( RenderStateFlags state ) const {
951         normalquantisation_draw();
952 }
953 };
954
955 const float g_test_quantise_normal = 1.f / static_cast<float>( 1 << 3 );
956
957 class TestNormalQuantisation
958 {
959 void check_normal( const Normal3f& normal, const Normal3f& other ){
960         spherical_t spherical = spherical_from_normal3f( normal );
961         double longditude = RAD2DEG( spherical.longditude );
962         double latitude = RAD2DEG( spherical.latitude );
963         double x = cos( spherical.longditude ) * sin( spherical.latitude );
964         double y = sin( spherical.longditude ) * sin( spherical.latitude );
965         double z = cos( spherical.latitude );
966
967         ASSERT_MESSAGE( normal3f_dot( normal, other ) > 0.99, "bleh" );
968 }
969
970 void test_normal( const Normal3f& normal ){
971         Normal3f test = normal3f_from_spherical( spherical_from_normal3f( normal ) );
972         check_normal( normal, test );
973
974         EOctant octant = normal3f_classify_octant( normal );
975         Normal3f folded = normal3f_fold_octant( normal, octant );
976         ESextant sextant = normal3f_classify_sextant( folded );
977         folded = normal3f_fold_sextant( folded, sextant );
978
979         double scale = static_cast<float>( c_quantise_normal ) / ( folded.x + folded.y + folded.z );
980
981         double zbits = folded.z * scale;
982         double ybits = folded.y * scale;
983
984         std::size_t zbits_q = static_cast<std::size_t>( zbits );
985         std::size_t ybits_q = static_cast<std::size_t>( ybits );
986
987         ASSERT_MESSAGE( zbits_q <= ( c_quantise_normal / 8 ) * 3, "bleh" );
988         ASSERT_MESSAGE( ybits_q <= ( c_quantise_normal / 2 ), "bleh" );
989         ASSERT_MESSAGE( zbits_q + ( ( c_quantise_normal / 2 ) - ybits_q ) <= ( c_quantise_normal / 2 ), "bleh" );
990
991         std::size_t y_t = ( zbits_q < ( c_quantise_normal / 4 ) ) ? ybits_q : ( c_quantise_normal / 2 ) - ybits_q;
992         std::size_t z_t = ( zbits_q < ( c_quantise_normal / 4 ) ) ? zbits_q : ( c_quantise_normal / 2 ) - zbits_q;
993         std::size_t index = ( c_quantise_normal / 4 ) * y_t + z_t;
994         ASSERT_MESSAGE( index <= ( c_quantise_normal / 4 ) * ( c_quantise_normal / 2 ), "bleh" );
995
996         Normal3f tmp( c_quantise_normal - zbits_q - ybits_q, ybits_q, zbits_q );
997         tmp = normal3f_normalised( tmp );
998
999         Normal3f unfolded = normal3f_unfold_octant( normal3f_unfold_sextant( tmp, sextant ), octant );
1000
1001         check_normal( normal, unfolded );
1002
1003         double dot = normal3f_dot( normal, unfolded );
1004         float length = VectorLength( normal3f_to_array( unfolded ) );
1005         float inv_length = 1 / length;
1006
1007         Normal3f quantised = normal3f_quantised( normal );
1008         check_normal( normal, quantised );
1009 }
1010 void test2( const Normal3f& normal, const Normal3f& other ){
1011         if ( normal3f_quantised( normal ) != normal3f_quantised( other ) ) {
1012                 int bleh = 0;
1013         }
1014 }
1015
1016 static Normal3f normalise( float x, float y, float z ){
1017         return normal3f_normalised( Normal3f( x, y, z ) );
1018 }
1019
1020 float vec_rand(){
1021         return static_cast<float>( rand() - ( RAND_MAX / 2 ) );
1022 }
1023
1024 Normal3f normal3f_rand(){
1025         return normalise( vec_rand(), vec_rand(), vec_rand() );
1026 }
1027
1028 public:
1029 TestNormalQuantisation(){
1030         for ( int i = 4096; i > 0; --i )
1031                 test_normal( normal3f_rand() );
1032
1033         test_normal( normalise( 1, 0, 0 ) );
1034         test_normal( normalise( 0, 1, 0 ) );
1035         test_normal( normalise( 0, 0, 1 ) );
1036         test_normal( normalise( 1, 1, 0 ) );
1037         test_normal( normalise( 1, 0, 1 ) );
1038         test_normal( normalise( 0, 1, 1 ) );
1039
1040         test_normal( normalise( 10000, 10000, 10000 ) );
1041         test_normal( normalise( 10000, 10000, 10001 ) );
1042         test_normal( normalise( 10000, 10000, 10002 ) );
1043         test_normal( normalise( 10000, 10000, 10010 ) );
1044         test_normal( normalise( 10000, 10000, 10020 ) );
1045         test_normal( normalise( 10000, 10000, 10030 ) );
1046         test_normal( normalise( 10000, 10000, 10100 ) );
1047         test_normal( normalise( 10000, 10000, 10101 ) );
1048         test_normal( normalise( 10000, 10000, 10102 ) );
1049         test_normal( normalise( 10000, 10000, 10200 ) );
1050         test_normal( normalise( 10000, 10000, 10201 ) );
1051         test_normal( normalise( 10000, 10000, 10202 ) );
1052         test_normal( normalise( 10000, 10000, 10203 ) );
1053         test_normal( normalise( 10000, 10000, 10300 ) );
1054
1055
1056         test2( normalise( 10000, 10000, 10000 ), normalise( 10000, 10000, 10001 ) );
1057         test2( normalise( 10000, 10000, 10001 ), normalise( 10000, 10001, 10000 ) );
1058 }
1059 };
1060
1061 TestNormalQuantisation g_testNormalQuantisation;
1062
1063
1064 #endif
1065
1066 #if 0
1067 class TestSelectableObserver : public observer_template<const Selectable&>
1068 {
1069 public:
1070 void notify( const Selectable& arguments ){
1071         bool bleh = arguments.isSelected();
1072 }
1073 };
1074
1075 inline void test_bleh(){
1076         TestSelectableObserver test;
1077         ObservableSelectableInstance< SingleObservable< SelectionChangeCallback > > bleh;
1078         bleh.attach( test );
1079         bleh.setSelected( true );
1080         bleh.detach( test );
1081 }
1082
1083 class TestBleh
1084 {
1085 public:
1086 TestBleh(){
1087         test_bleh();
1088 }
1089 };
1090
1091 const TestBleh testbleh;
1092 #endif
1093
1094
1095 #if 0
1096 class TestRefcountedString
1097 {
1098 public:
1099 TestRefcountedString(){
1100         {
1101                 // copy construct
1102                 SmartString string1( "string1" );
1103                 SmartString string2( string1 );
1104                 SmartString string3( string2 );
1105         }
1106         {
1107                 // refcounted assignment
1108                 SmartString string1( "string1" );
1109                 SmartString string2( "string2" );
1110                 string1 = string2;
1111         }
1112         {
1113                 // copy assignment
1114                 SmartString string1( "string1" );
1115                 SmartString string2( "string2" );
1116                 string1 = string2.c_str();
1117         }
1118         {
1119                 // self-assignment
1120                 SmartString string1( "string1" );
1121                 string1 = string1;
1122         }
1123         {
1124                 // self-assignment via another reference
1125                 SmartString string1( "string1" );
1126                 SmartString string2( string1 );
1127                 string1 = string2;
1128         }
1129 }
1130 };
1131
1132 const TestRefcountedString g_testRefcountedString;
1133
1134 #endif
1135
1136 void Select_MakeDetail(){
1137         UndoableCommand undo( "brushSetDetail" );
1138         Scene_BrushSetDetail_Selected( GlobalSceneGraph(), true );
1139 }
1140
1141 void Select_MakeStructural(){
1142         UndoableCommand undo( "brushClearDetail" );
1143         Scene_BrushSetDetail_Selected( GlobalSceneGraph(), false );
1144 }
1145
1146 class BrushMakeSided
1147 {
1148 std::size_t m_count;
1149 public:
1150 BrushMakeSided( std::size_t count )
1151         : m_count( count ){
1152 }
1153
1154 void set(){
1155         Scene_BrushConstructPrefab( GlobalSceneGraph(), eBrushPrism, m_count, TextureBrowser_GetSelectedShader( GlobalTextureBrowser() ) );
1156 }
1157
1158 typedef MemberCaller<BrushMakeSided, void(), &BrushMakeSided::set> SetCaller;
1159 };
1160
1161
1162 BrushMakeSided g_brushmakesided3( 3 );
1163 BrushMakeSided g_brushmakesided4( 4 );
1164 BrushMakeSided g_brushmakesided5( 5 );
1165 BrushMakeSided g_brushmakesided6( 6 );
1166 BrushMakeSided g_brushmakesided7( 7 );
1167 BrushMakeSided g_brushmakesided8( 8 );
1168 BrushMakeSided g_brushmakesided9( 9 );
1169
1170 inline int axis_for_viewtype( int viewtype ){
1171         switch ( viewtype )
1172         {
1173         case XY:
1174                 return 2;
1175         case XZ:
1176                 return 1;
1177         case YZ:
1178                 return 0;
1179         }
1180         return 2;
1181 }
1182
1183 class BrushPrefab
1184 {
1185 EBrushPrefab m_type;
1186 public:
1187 BrushPrefab( EBrushPrefab type )
1188         : m_type( type ){
1189 }
1190
1191 void set(){
1192         DoSides( m_type, axis_for_viewtype( GetViewAxis() ) );
1193 }
1194
1195 typedef MemberCaller<BrushPrefab, void(), &BrushPrefab::set> SetCaller;
1196 };
1197
1198 BrushPrefab g_brushprism( eBrushPrism );
1199 BrushPrefab g_brushcone( eBrushCone );
1200 BrushPrefab g_brushsphere( eBrushSphere );
1201 BrushPrefab g_brushrock( eBrushRock );
1202
1203
1204 void FlipClip();
1205
1206 void SplitClip();
1207
1208 void Clip();
1209
1210 void OnClipMode( bool enable );
1211
1212 bool ClipMode();
1213
1214
1215 void ClipSelected(){
1216         if ( ClipMode() ) {
1217                 UndoableCommand undo( "clipperClip" );
1218                 Clip();
1219         }
1220 }
1221
1222 void SplitSelected(){
1223         if ( ClipMode() ) {
1224                 UndoableCommand undo( "clipperSplit" );
1225                 SplitClip();
1226         }
1227 }
1228
1229 void FlipClipper(){
1230         FlipClip();
1231 }
1232
1233
1234 Callback<void()> g_texture_lock_status_changed;
1235 ConstReferenceCaller<bool, void(const Callback<void(bool)> &), PropertyImpl<bool>::Export> g_texdef_movelock_caller( g_brush_texturelock_enabled );
1236 ToggleItem g_texdef_movelock_item( g_texdef_movelock_caller );
1237
1238 void Texdef_ToggleMoveLock(){
1239         g_brush_texturelock_enabled = !g_brush_texturelock_enabled;
1240         g_texdef_movelock_item.update();
1241         g_texture_lock_status_changed();
1242 }
1243
1244
1245 void Brush_registerCommands(){
1246         GlobalToggles_insert( "TogTexLock", makeCallbackF(Texdef_ToggleMoveLock), ToggleItem::AddCallbackCaller( g_texdef_movelock_item ), Accelerator( 'T', (GdkModifierType)GDK_SHIFT_MASK ) );
1247
1248         GlobalCommands_insert( "BrushPrism", BrushPrefab::SetCaller( g_brushprism ) );
1249         GlobalCommands_insert( "BrushCone", BrushPrefab::SetCaller( g_brushcone ) );
1250         GlobalCommands_insert( "BrushSphere", BrushPrefab::SetCaller( g_brushsphere ) );
1251         GlobalCommands_insert( "BrushRock", BrushPrefab::SetCaller( g_brushrock ) );
1252
1253         GlobalCommands_insert( "Brush3Sided", BrushMakeSided::SetCaller( g_brushmakesided3 ), Accelerator( '3', (GdkModifierType)GDK_CONTROL_MASK ) );
1254         GlobalCommands_insert( "Brush4Sided", BrushMakeSided::SetCaller( g_brushmakesided4 ), Accelerator( '4', (GdkModifierType)GDK_CONTROL_MASK ) );
1255         GlobalCommands_insert( "Brush5Sided", BrushMakeSided::SetCaller( g_brushmakesided5 ), Accelerator( '5', (GdkModifierType)GDK_CONTROL_MASK ) );
1256         GlobalCommands_insert( "Brush6Sided", BrushMakeSided::SetCaller( g_brushmakesided6 ), Accelerator( '6', (GdkModifierType)GDK_CONTROL_MASK ) );
1257         GlobalCommands_insert( "Brush7Sided", BrushMakeSided::SetCaller( g_brushmakesided7 ), Accelerator( '7', (GdkModifierType)GDK_CONTROL_MASK ) );
1258         GlobalCommands_insert( "Brush8Sided", BrushMakeSided::SetCaller( g_brushmakesided8 ), Accelerator( '8', (GdkModifierType)GDK_CONTROL_MASK ) );
1259         GlobalCommands_insert( "Brush9Sided", BrushMakeSided::SetCaller( g_brushmakesided9 ), Accelerator( '9', (GdkModifierType)GDK_CONTROL_MASK ) );
1260
1261         GlobalCommands_insert( "ClipSelected", makeCallbackF(ClipSelected), Accelerator( GDK_KEY_Return ) );
1262         GlobalCommands_insert( "SplitSelected", makeCallbackF(SplitSelected), Accelerator( GDK_KEY_Return, (GdkModifierType)GDK_SHIFT_MASK ) );
1263         GlobalCommands_insert( "FlipClip", makeCallbackF(FlipClipper), Accelerator( GDK_KEY_Return, (GdkModifierType)GDK_CONTROL_MASK ) );
1264
1265         GlobalCommands_insert( "MakeDetail", makeCallbackF(Select_MakeDetail), Accelerator( 'M', (GdkModifierType)GDK_CONTROL_MASK ) );
1266         GlobalCommands_insert( "MakeStructural", makeCallbackF(Select_MakeStructural), Accelerator( 'S', (GdkModifierType)( GDK_SHIFT_MASK | GDK_CONTROL_MASK ) ) );
1267 }
1268
1269 void Brush_constructMenu( ui::Menu menu ){
1270         create_menu_item_with_mnemonic( menu, "Prism...", "BrushPrism" );
1271         create_menu_item_with_mnemonic( menu, "Cone...", "BrushCone" );
1272         create_menu_item_with_mnemonic( menu, "Sphere...", "BrushSphere" );
1273         create_menu_item_with_mnemonic( menu, "Rock...", "BrushRock" );
1274         menu_separator( menu );
1275         {
1276                 auto menu_in_menu = create_sub_menu_with_mnemonic( menu, "CSG" );
1277                 if ( g_Layout_enableDetachableMenus.m_value ) {
1278                         menu_tearoff( menu_in_menu );
1279                 }
1280                 create_menu_item_with_mnemonic( menu_in_menu, "CSG _Subtract", "CSGSubtract" );
1281                 create_menu_item_with_mnemonic( menu_in_menu, "CSG _Merge", "CSGMerge" );
1282                 create_menu_item_with_mnemonic( menu_in_menu, "Make _Room", "CSGRoom" );
1283                 create_menu_item_with_mnemonic( menu_in_menu, "CSG _Tool", "CSGTool" );
1284         }
1285         menu_separator( menu );
1286         {
1287                 auto menu_in_menu = create_sub_menu_with_mnemonic( menu, "Clipper" );
1288                 if ( g_Layout_enableDetachableMenus.m_value ) {
1289                         menu_tearoff( menu_in_menu );
1290                 }
1291
1292                 create_menu_item_with_mnemonic( menu_in_menu, "Clip selection", "ClipSelected" );
1293                 create_menu_item_with_mnemonic( menu_in_menu, "Split selection", "SplitSelected" );
1294                 create_menu_item_with_mnemonic( menu_in_menu, "Flip Clip orientation", "FlipClip" );
1295         }
1296         menu_separator( menu );
1297         create_menu_item_with_mnemonic( menu, "Make detail", "MakeDetail" );
1298         create_menu_item_with_mnemonic( menu, "Make structural", "MakeStructural" );
1299 //      create_menu_item_with_mnemonic( menu, "Snap selection to _grid", "SnapToGrid" );
1300
1301         create_check_menu_item_with_mnemonic( menu, "Texture Lock", "TogTexLock" );
1302         menu_separator( menu );
1303         create_menu_item_with_mnemonic( menu, "Copy Face Texture", "FaceCopyTexture" );
1304         create_menu_item_with_mnemonic( menu, "Paste Face Texture", "FacePasteTexture" );
1305
1306         command_connect_accelerator( "Brush3Sided" );
1307         command_connect_accelerator( "Brush4Sided" );
1308         command_connect_accelerator( "Brush5Sided" );
1309         command_connect_accelerator( "Brush6Sided" );
1310         command_connect_accelerator( "Brush7Sided" );
1311         command_connect_accelerator( "Brush8Sided" );
1312         command_connect_accelerator( "Brush9Sided" );
1313 }