Rounded corner box having holes using three.js

纵然是瞬间 提交于 2020-02-04 12:52:33

问题


I need to create a box having rounded corners on 4 sides but on top and bottom. Three.js has a very good example webgl_geometry_shapes and I'm trying to replicate it to add a hole to the green rounded box. In this example to add a hole to the circle:

        // Arc circle

            var arcShape = new THREE.Shape()
                .moveTo( 50, 10 )
                .absarc( 10, 10, 40, 0, Math.PI * 2, false );

            var holePath = new THREE.Path()
                .moveTo( 20, 10 )
                .absarc( 10, 10, 10, 0, Math.PI * 2, true )

            arcShape.holes.push( holePath );

They create a shape and a path and then they add the hole. The result is

I used the same logic to the rounded square

Original shape

 // Rounded rectangle

            var roundedRectShape = new THREE.Shape();

            ( function roundedRect( ctx, x, y, width, height, radius ) {

                ctx.moveTo( x, y + radius );
                ctx.lineTo( x, y + height - radius );
                ctx.quadraticCurveTo( x, y + height, x + radius, y + height );
                ctx.lineTo( x + width - radius, y + height );
                ctx.quadraticCurveTo( x + width, y + height, x + width, y + height - radius );
                ctx.lineTo( x + width, y + radius );
                ctx.quadraticCurveTo( x + width, y, x + width - radius, y );
                ctx.lineTo( x + radius, y );
                ctx.quadraticCurveTo( x, y, x, y + radius );

            } )( roundedRectShape, 0, 0, 50, 50, 20 );

new hole shape

            var roundedRectShape_small = new THREE.Path();

            ( function roundedRect( ctx, x, y, width, height, radius ) {

                ctx.moveTo( x, y + radius );
                ctx.lineTo( x, y + height - radius );
                ctx.quadraticCurveTo( x, y + height, x + radius, y + height );
                ctx.lineTo( x + width - radius, y + height );
                ctx.quadraticCurveTo( x + width, y + height, x + width, y + height - radius );
                ctx.lineTo( x + width, y + radius );
                ctx.quadraticCurveTo( x + width, y, x + width - radius, y );
                ctx.lineTo( x + radius, y );
                ctx.quadraticCurveTo( x, y, x, y + radius );

            } )( roundedRectShape_small, 10, 10, 30, 30, 20 );

Then I add the hole

roundedRectShape.holes.push( roundedRectShape_small );

The result is not the same...

As you can see the hole is there but the 3D effect is wrong on internal side. How to fix it?


回答1:


You have to drew the hole in the opposite winding order. Since the outer rectangle path is draw clockwise, the hole path has to be draw counter clockwise:

let f_rect = function roundedRect( ctx, x, y, width, height, radius ) {
    ctx.moveTo( x, y + radius );
    ctx.lineTo( x, y + height - radius );
    ctx.quadraticCurveTo( x, y + height, x + radius, y + height );
    ctx.lineTo( x + width - radius, y + height );
    ctx.quadraticCurveTo( x + width, y + height, x + width, y + height - radius );
    ctx.lineTo( x + width, y + radius );
    ctx.quadraticCurveTo( x + width, y, x + width - radius, y );
    ctx.lineTo( x + radius, y );
    ctx.quadraticCurveTo( x, y, x, y + radius );
}

let f_rect_reverse = function roundedRect( ctx, x, y, width, height, radius ) { 
    ctx.moveTo( x, y + height - radius );
    ctx.lineTo( x, y + radius ); 
    ctx.quadraticCurveTo( x, y, x + radius, y );
    ctx.lineTo( x + width - radius, y );
    ctx.quadraticCurveTo( x + width, y, x + width, y + radius );
    ctx.lineTo( x + width, y + height - radius );
    ctx.quadraticCurveTo( x + width, y + height, x + width - radius, y + height );
    ctx.lineTo( x + radius, y + height );
    ctx.quadraticCurveTo( x, y + height, x, y + height - radius );       
}

var roundedRectShape = new THREE.Shape();
f_rect( roundedRectShape, -25, -25, 50, 50, 20 );
var roundedRectShape_small = new THREE.Path();
f_rect_reverse( roundedRectShape_small, -15, -15, 30, 30, 10 );
roundedRectShape.holes.push( roundedRectShape_small );

See the complete example:

(function onLoad() {
  var container, camera, scene, renderer, controls;
  
  init();
  animate();

  function init() {
    container = document.getElementById('container');
    
    renderer = new THREE.WebGLRenderer({
      antialias: true
    });
    renderer.setPixelRatio(window.devicePixelRatio);
    renderer.setSize(window.innerWidth, window.innerHeight);
    renderer.shadowMap.enabled = true;
    container.appendChild(renderer.domElement);

    scene = new THREE.Scene();
    scene.background = new THREE.Color(0xffffff);
    
    camera = new THREE.PerspectiveCamera(70, window.innerWidth / window.innerHeight, 1, 10000);
    camera.position.set(3, 2, 7);
    scene.add(camera);
    window.onresize = function() {
      renderer.setSize(window.innerWidth, window.innerHeight);
      camera.aspect = window.innerWidth / window.innerHeight;
      camera.updateProjectionMatrix();
    }
    
    var ambientLight = new THREE.AmbientLight(0x404040);
    scene.add(ambientLight);

    var directionalLight = new THREE.DirectionalLight( 0xffffff, 0.5 );
    directionalLight.position.x = 4;
    directionalLight.position.y = 1;
    directionalLight.position.z = 2;
    scene.add( directionalLight );
    
    controls = new THREE.OrbitControls(camera, renderer.domElement);
		
    addGridHelper();
    createModel();
  }

  function createModel() {

      let f_rect = function roundedRect( ctx, x, y, width, height, radius ) {
          ctx.moveTo( x, y + radius );
          ctx.lineTo( x, y + height - radius );
          ctx.quadraticCurveTo( x, y + height, x + radius, y + height );
          ctx.lineTo( x + width - radius, y + height );
          ctx.quadraticCurveTo( x + width, y + height, x + width, y + height - radius );
          ctx.lineTo( x + width, y + radius );
          ctx.quadraticCurveTo( x + width, y, x + width - radius, y );
          ctx.lineTo( x + radius, y );
          ctx.quadraticCurveTo( x, y, x, y + radius );
      }

      let f_rect_reverse = function roundedRect( ctx, x, y, width, height, radius ) { 
          ctx.moveTo( x, y + height - radius );
          ctx.lineTo( x, y + radius ); 
          ctx.quadraticCurveTo( x, y, x + radius, y );
          ctx.lineTo( x + width - radius, y );
          ctx.quadraticCurveTo( x + width, y, x + width, y + radius );
          ctx.lineTo( x + width, y + height - radius );
          ctx.quadraticCurveTo( x + width, y + height, x + width - radius, y + height );
          ctx.lineTo( x + radius, y + height );
          ctx.quadraticCurveTo( x, y + height, x, y + height - radius );       
      }

      var roundedRectShape = new THREE.Shape();
      f_rect( roundedRectShape, -25, -25, 50, 50, 20 );
      var roundedRectShape_small = new THREE.Path();
      f_rect_reverse( roundedRectShape_small, -15, -15, 30, 30, 10 );
      roundedRectShape.holes.push( roundedRectShape_small );

      var material = new THREE.MeshLambertMaterial( { color: 0x00ff00, side: THREE.DoubleSide } );
      var extrudeSettings = { depth: 8, bevelEnabled: true, bevelSegments: 2, steps: 3, bevelSize: 5, bevelThickness: 5 };
      var geometry = new THREE.ExtrudeBufferGeometry( roundedRectShape, extrudeSettings );
      
      let s = 0.1;
      var mesh = new THREE.Mesh( geometry, material );
      mesh.scale.set( s, s, s );

      scene.add( mesh );
  }

  function addGridHelper() {

    var helper = new THREE.GridHelper(10, 10);
    helper.material.opacity = 0.25;
    helper.material.transparent = true;
    scene.add(helper);

    var axis = new THREE.AxesHelper(100);
    scene.add(axis);
  }

  function animate() {
    requestAnimationFrame(animate);
    render();
  }

  function render() {
    renderer.render(scene, camera);
  }
})();
<!--script src="https://threejs.org/build/three.js"></!--script-->
<script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/110/three.min.js"></script>
<script src="https://threejs.org/examples/js/controls/OrbitControls.js"></script>
<div id="container"></div>


来源:https://stackoverflow.com/questions/59828669/rounded-corner-box-having-holes-using-three-js

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!