three js 做小地图

image.png

做个最简单的mini map,这就是游戏中常用的小地图,

研究了一圈,原来发现居然还有几个小坑,这里分享一下,希望对碰到这个问题的朋友有帮助

image.png

最初在试用的时候一切正常

我们先引入three js和相关lib,然后创建一个dom

<div id="ThreeJS" style="position: absolute; left:0px; top:0px"></div>

然后是js代码

// MAIN

//全局变量
  var container, scene, renderer, controls, stats;
  var clock = new THREE.Clock();

  var MovingCube;
  var perspectiveCamera;

  var mapCamera, mapWidth = 240, mapHeight = 160; // w/h should match div dimensions

  var shape2

  init();
  animate();





  // FUNCTIONS
  function init() {
  
    scene = new THREE.Scene();
 
    var SCREEN_WIDTH = window.innerWidth, SCREEN_HEIGHT = window.innerHeight;
    var VIEW_ANGLE = 45, ASPECT = SCREEN_WIDTH / SCREEN_HEIGHT, NEAR = 0.1, FAR = 20000;

    // orthographic 相机
    mapCamera = new THREE.OrthographicCamera(
      window.innerWidth / -2,       // Left
      window.innerWidth / 2,        // Right
      window.innerHeight / 2,       // Top
      window.innerHeight / -2,  // Bottom
      -5000,                        // Near
      10000 );                      // Far
    mapCamera.up = new THREE.Vector3(0,0,-1);
    mapCamera.lookAt( new THREE.Vector3(0,-1,0) );
    scene.add(mapCamera);

    // perspective 相机
    perspectiveCamera = new THREE.PerspectiveCamera(VIEW_ANGLE, ASPECT, NEAR, FAR);
    perspectiveCamera.position.set(0,200,550);
    perspectiveCamera.lookAt(scene.position);
    scene.add(perspectiveCamera);



      renderer = new THREE.WebGLRenderer( {antialias:true} );
    renderer.setSize(SCREEN_WIDTH, SCREEN_HEIGHT);
    container = document.getElementById( 'ThreeJS' );
    container.appendChild( renderer.domElement );


    controls = new THREE.OrbitControls(perspectiveCamera, renderer.domElement);
    controls.enableKeys = false;
    controls.enableKeys = false;


    // fps显示stats
    stats = new Stats();
    stats.domElement.style.position = 'absolute';
    stats.domElement.style.bottom = '0px';
    stats.domElement.style.zIndex = 100;
    container.appendChild( stats.domElement );
    // 灯光
    var light = new THREE.PointLight(0xffffff);
    light.position.set(0,250,0);
    scene.add(light);
    var ambientlight = new THREE.AmbientLight(0x111111);
    scene.add( ambientlight );

    // 地板
    var floorMaterial = new THREE.MeshBasicMaterial( { color: 0x999999, side: THREE.DoubleSide } );
    var floorGeometry = new THREE.PlaneGeometry(2000, 2000, 10, 10);
    var floor = new THREE.Mesh(floorGeometry, floorMaterial);
    floor.position.y = -0.5;
    floor.rotation.x = Math.PI / 2;
    scene.add(floor);




    // 创建6面颜色
    var materialArray = [];
    materialArray.push(new THREE.MeshBasicMaterial( { color: 0xff0000 }));
    materialArray.push(new THREE.MeshBasicMaterial( { color: 0xffee00 }));
    materialArray.push(new THREE.MeshBasicMaterial( { color: 0xffeeff }));
    materialArray.push(new THREE.MeshBasicMaterial( { color: 0x00eeff }));
    materialArray.push(new THREE.MeshBasicMaterial( { color: 0x0000ff }));
    materialArray.push(new THREE.MeshBasicMaterial( { color: 0xff00ff }));

    
    var MovingCubeGeom = new THREE.CubeGeometry( 50, 50, 50, 1, 1, 1, materialArray );
    MovingCube = new THREE.Mesh( MovingCubeGeom, materialArray );
    MovingCube.position.set(0, 25.1, 0);
    scene.add( MovingCube );

  



    // 创建几个场景上的物体
    var colorMaterial = new THREE.MeshLambertMaterial( { color: 0xff3333 } );
    var shape = new THREE.Mesh( new THREE.TorusKnotGeometry( 30, 6, 160, 10, 2, 5 ), colorMaterial );
    shape.position.set(-200, 50, -200);
    scene.add( shape );
    // torus knot
    var colorMaterial = new THREE.MeshLambertMaterial( { color: 0x33ff33 } );
    var shape = new THREE.Mesh( new THREE.TorusKnotGeometry( 30, 6, 160, 10, 3, 2 ), colorMaterial );
    shape.position.set(200, 50, -200);
    scene.add( shape );
    // torus knot
    var colorMaterial = new THREE.MeshLambertMaterial( { color: 0xffff33 } );
    var shape = new THREE.Mesh( new THREE.TorusKnotGeometry( 30, 6, 160, 10, 4, 3 ), colorMaterial );
    shape.position.set(200, 50, 200);
    scene.add( shape );
    // torus knot 这个让它动起来,所以放到外面
    var colorMaterial = new THREE.MeshLambertMaterial( { color: 0x3333ff } );
     shape2 = new THREE.Mesh( new THREE.TorusKnotGeometry( 30, 6, 160, 10, 3, 4 ), colorMaterial );
    shape2.position.set(-200, 50, 200);
    scene.add( shape2 );

    renderer.setSize( window.innerWidth, window.innerHeight );
    renderer.setClearColor( 0x000000, 1 );
    renderer.autoClear = false;

  }

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

  function update()
  {
    stats.update();
    controls.update();
  }

    var w = window.innerWidth, h = window.innerHeight;
  function render()
  {


    renderer.clear();
    if(shape2){
      shape2.rotation.y += 0.01
    }


    renderer.setViewport( 10, h - mapHeight - 10, mapWidth, mapHeight );
    renderer.render( scene, mapCamera );

    renderer.setViewport( 0, 0, w, h );
    renderer.render( scene, perspectiveCamera );

  }

整个显示看起来非常不错,动画可以实时反应到小地图上


image.png

然后试着移植到项目中,怎么都无法显示,要么显示地图,要么显示主体,就是不正常显示

找了半天发现项目中这句

scene.background = new THREE.Color( 0xbfd1e5 );

这个是让背景不至于全黑,结果却发现造成小地图被覆盖了
注释掉这句,然后小地图就可以正常显示了

然后发现一个奇怪的问题


image.png

shader的粒子在小地图上显示得特别的大

中间尝试换了一下相机类型,改成PerspectiveCamera,结果发现居然会被地图物体遮挡


image.png

这个最终的解决方法是给OrthographicCamera的position改变一下y的高度,变成

mapCamera.position.y = 100

就可以避免这个问题,这个值的大小并不会改变小地图中显示的内容,

要修改小地图中显示内容的多少,要去改 OrthographicCamera 的 Left, Right, Top, Bottom四个参数

目前虽然做了不少优化


image.png

可最终的结果draw calls并不理想,几乎多了一倍,同时也渲染了阴影等内容,按照unity的推荐,mini map最好还是做成2D的
不知道有没有对这块了解的朋友,一起探讨下如何优化?

谢谢~~

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270

推荐阅读更多精彩内容

  • 关于修昔底德本人的记载少有信史,关于他的生平,只能到他本人的著作中去寻找。 从《伯罗奔尼撒战争史》中的记载可以推测...
    鹿韬阅读 897评论 2 1
  • 昨天跟以前的好友同床共枕,她还是像从前那样健谈。而我依然是一个听众,末了,她说了一句:“你还是跟小时候一样,什么都...
    明兮兮阅读 466评论 0 0
  • 都说 再好 也只不过是独行者的执念和向往 可我 还是把梦留在了雪域 我想去看从布达拉宫升起的太阳 那些孩子们天籁般...
    默子江阅读 345评论 0 0