国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

簡單一招搞定 three.js 屏幕適配

騫諱護 / 2104人閱讀

摘要:我們先規定好相機到平面的距離為,然后試試看能不能通過計算設置值,剛好讓平面填滿一個寬高比為的屏幕。最終代碼與效果計算相機的函數在相機前方距離想要看到最大正方形區域邊長為屏幕寬高比效果的完整代碼原文鏈接

這篇文章只討論 PerspectiveCamera 的適配方法

做過手機 H5 的同學可能會覺得屏幕適配挺麻煩。原因是設計師提供的設計稿尺寸比固定,但是前端開發者卻要適配不同大小、長寬比的目標設備。適配的終極目標無非是最大程度把主體內容優雅地呈現給用戶。開發和設計如果沒有協調好的話可能會妥協比較丑陋的方案,例如由于設計比例問題,為了照顧主體內容不被裁剪,只好設備兩邊,或者上下留黑邊這種。

不過在 3D 的世界里,我們不用擔心會有黑邊的問題,因為 3D 場景是無限延伸的,總能填滿任何比例的屏幕。

先看看 PerspectiveCamera 官方 API 說明如下:

PerspectiveCamera( fov, aspect, near, far )

fov — Camera frustum vertical field of view.
aspect — Camera frustum aspect ratio.
near — Camera frustum near plane.
far — Camera frustum far plane.

上面四個參數都會影響成像結果,fovaspect 設置 XY 平面的范圍,也就是廣度。 nearfar 影響的是縱深 Z 軸的范圍,也就是深度。縱深只要保證物體離相機距離在這個范圍就可以了,這是為了性能而設置的參數,由用戶設置,只渲染必要的東西。實際上真實的相機這兩個值對應的是 0 到 無限遠。

這些參數設置好之后,成像就相應確定了。最后 three.js 把相機拍攝到的矩形區域對應好四個頂點渲染到屏幕上。同樣比例的屏幕看到的圖像是一致的,與屏幕大小無關

下面我用一個簡單的場景來看一下這些參數對成像的影響。

場景元素

相機 (PerspectiveCamera)

一個邊長為 100 的平面(主體內容范圍),放在世界坐標中心。

var camera = new THREE.PerspectiveCamera(53, 500 / 500, 0.1, 1000);

var planeGemo = new THREE.PlaneGeometry( 100, 100, 10, 10 )
var meshMaterial = new THREE.MeshLambertMaterial();
meshMaterial.color = new THREE.Color(0x2dcaf1);
meshMaterial.side = THREE.DoubleSide;

var wireFrameMat = new THREE.MeshBasicMaterial();
wireFrameMat.color = new THREE.Color(0xffffdffffd);
wireFrameMat.wireframe = true;

var plane = THREE.SceneUtils.createMultiMaterialObject(planeGemo, [meshMaterial, wireFrameMat]);

scene.add(plane);
目標

在任何屏幕下,都能最大程度地顯示完整的立方體。最大程度,就是最少多余空間的意思。下面是要達到效果

設置 fov 參數

可以直接想到的一種適配方法是——改變 camera 到目標物體的距離以控制成像的內容,但是這樣做計算成本比較高,而且還有可能影響其他一些數值,然后需要相應一起計算修改。
我想到改變視角也可以達到控制成像內容多少的目的,于是我想可不可以只通過改變 fov 一個數值,達到我要的效果。

fov 官網的定義翻譯過來是垂直方向的視角大小。我們先規定好相機到平面的距離為 100,然后試試看能不能通過計算設置 fov 值,剛好讓平面填滿一個寬高比為 1:1 的屏幕。

plane.position.set(0,0,0);
camera.position.set(0,0,100);
camera.lookAt(new THREE.Vector3);

觀察上面的圖,可以很容易求出 fov 的值, fov = arctan((100/2)/100) * 2; fov 為 0.9272952180016122,約等于 53 度。

camera.fov = Math.atan((100/2)/100) * 2 * (180 / Math.PI);
camera.updateProjectionMatrix();

設置完剛剛求出的 fov 值,將場景渲染到 寬高比為 1:1 的畫布上。

渲染結果和預想的一樣,平面剛好填滿了 1:1 的畫布。

fov 和寬高比例的關系

下面在固定的 fov 下,使用 dat.gui 工具調整寬高比,觀察渲染區域的變化。

因為fov設置的是垂直方向的視角范圍,可以看到無論我們怎么改變寬高比例,垂直方向的渲染范圍,都是一致的。水平方向則是以裁剪的方式顯示。也就是說當我們設置好視角讓垂直方向范圍剛好等于主體內容的范圍,只要寬高比大于1,我們得到的渲染結果,已經是最佳的了。問題就只剩下當寬高比小于1的情況了。

寬高比小于1的時候,垂直方向顯示的高度剛好是等于主體內容的高度。為了能讓水平方向完整顯示主體內容,我們只有將垂直方向范圍增大,也就是將 fov 設置一個更大的值,此時水平方向的范圍也會隨之增大。當將 fov調整到 水平方向剛好能顯示主體內容時,垂直方向此時顯示的范圍是超過主體內容垂直方向的范圍的。其中的關系,其實可以用很簡單的函數求出來。

已知 照相機到主題內容的距離為 d
正方形主體內容的邊長為 w

設寬高比為 r,求照相機垂直方向的視角 f

當 r >= 1 時,照相機拍攝到的垂直方向范圍等于 w

當 r > 1
d * tan(f/2) * 2 = w

當 r < 1 時,照相機拍攝到的水平方向范圍等于 w,垂直方向范圍應該是 w/r
d * tan(f/2) * 2 = w/r

這樣,任意寬高比例的屏幕應該對應多大的垂直視角就確定了。

最終代碼與效果
var controls = new function () {

    camera.position.z = CAMERA_TO_MAIN_DIS;

    this.width = 500;
    this.height = 500;
    this.planeRY = 0;

    /**
    * 計算相機 fov 的函數
    * @param d : 在相機前方 d 距離
    * @param w : 想要看到最大正方形區域邊長為 w
    * @param r : 屏幕寬高比
    */
    function calcFov(d, w, r) {
        var f;
        var vertical = w;
        if (r < 1) {
            vertical = vertical/r;
        }
        f = Math.atan(vertical/d/2)*2 * (180 / Math.PI);
        return f;
    }

    this.redraw = ()=>{
        webGLRenderer.setSize(this.width, this.height);
        plane.rotation.y = this.planeRY;
        camera.fov = calcFov(CAMERA_TO_MAIN_DIS, MAIN_CONTENT_WIDTH, this.width / this.height);
        camera.aspect = this.width / this.height;
        camera.updateProjectionMatrix();
    }
}

效果:

demo 的完整代碼:http://codepen.io/JasonTurbo/pen/ZLwJMo
原文鏈接:http://gnauhca.com/blog/2016/11/24/threejs/THREEJS%E5%B1%8F%E5%B9%95%E9%80%82%E9%85%8D/

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/82162.html

相關文章

  • SegmentFault 技術周刊 Vol.35 - WebGL:打開網頁看大片

    摘要:在文末,我會附上一個可加載的模型方便學習中文藝術字渲染用原生可以很容易地繪制文字,但是原生提供的文字效果美化功能十分有限。 showImg(https://segmentfault.com/img/bVWYnb?w=900&h=385); WebGL 可以說是 HTML5 技術生態鏈中最為令人振奮的標準之一,它把 Web 帶入了 3D 的時代。 初識 WebGL 先通過幾個使用 Web...

    objc94 評論0 收藏0
  • 【51單片機】??萬物互聯??一招搞定單片機(持續更新中......)

    目錄 第一章:單片機概述 單片機的應用領域: STC89C52單片機: 單片機命名規則: ?單片機內部結構圖: ?單片機管腳圖: ?單片機最小系統: 進制表: 新建一個工程:? 檢查單片機驅動是否安裝完成: 打開Keil uVision集成開發環境 選擇Atmel下的AT89C52 創建源文件? ?第二章:LED LED介紹: ?點亮LED:?? 第一章:單片機概述 單片機(Micro Contr...

    idisfkj 評論0 收藏0
  • 全鏈路壓測一招搞定,阿里云性能測試鉑金版發布

    摘要:摘要阿里云性能測試是卓越的性能測試平臺,具備強大的分布式壓測能力,可模擬海量用戶的真實業務場景,讓所有性能問題無所遁形。近日,宣布推出了基于阿里雙全鏈路壓測平臺的鉑金版。 摘要: 阿里云性能測試(Performance Testing Service)是卓越的SaaS性能測試平臺,具備強大的分布式壓測能力,可模擬海量用戶的真實業務場景,讓所有性能問題無所遁形。近日,PTS宣布推出了基于...

    zhaochunqi 評論0 收藏0
  • 全鏈路壓測一招搞定,阿里云性能測試鉑金版發布

    摘要:摘要阿里云性能測試是卓越的性能測試平臺,具備強大的分布式壓測能力,可模擬海量用戶的真實業務場景,讓所有性能問題無所遁形。近日,宣布推出了基于阿里雙全鏈路壓測平臺的鉑金版。 摘要: 阿里云性能測試(Performance Testing Service)是卓越的SaaS性能測試平臺,具備強大的分布式壓測能力,可模擬海量用戶的真實業務場景,讓所有性能問題無所遁形。近日,PTS宣布推出了基于...

    gotham 評論0 收藏0

發表評論

0條評論

最新活動
閱讀需要支付1元查看
<