Shadertoy转传统技术实现程序化星球纹理

去年,一位游戏开发人员在 Shadertoy 上连续发布了多个行星渲染器:

(作者 ArchaicVirus 是一位夜晚的游戏开发者,白天上班。有兴趣的可以加他discord,在某个shader的评论区。)

代码十分简洁,却能打造天体效果,比如地球或者木星,美轮美奂。

首先,生成球面上的高度图,然后根据高度值,赋予色彩。所以高山、海洋呈现出色带效果,富有层次。

高度用叠加多个噪声计算。sphere函数计算当前像素视线与球体交点的距离(两次勾股定理),继而计算交点相对于球心的位置(向量)。基础的noise函数传入此向量。但并非计算3D噪声,而是2D噪声通过不断旋转向量、mix变成3D噪声。

有关2D/3D noise:

Noises - shadertoy 站长合集

FBM - shadertoy 站长博客

这种卡通或者 pixel art 的效果,也算是一个不错的起点。最畅销游戏我的世界、星露谷、王国不都是像素艺术么!

但性能是个问题(多层噪声)。小窗预览看不出来,全屏就知道了,gpu占用有点大(核显模式24%,比空白画面上升10%,一个星球尚且如此,多个星球不得卡爆)。

而且纯算法是没有业务逻辑的,只有数学,美术无法介入。

所以,想着使用原始贴图的技术,问了问 deepseek。普通回答太敷衍了,切换深度思考,结果...... deepseek 思考了很长时间(300秒)。

问:this shadertoy generates nice planet sphere. now I want to use traditional tech, i.e. render colors as textures and use that texture on a cubic-sphere using threejs. please answer js code.

js 复制代码
float hash(float n) {
    return fract(sin(n) * 53738.14534123);
}

float noise(vec2 p) {
    vec2 ip = floor(p);
    vec2 u = fract(p);
    u = u * u * (3.0 - 2.0 * u);
    return mix(
        mix(hash(ip.x + ip.y * 57.0), hash(ip.x + 1.0 + ip.y * 57.0), u.x),
        mix(hash(ip.x + (ip.y + 1.0) * 57.0), hash(ip.x + 1.0 + (ip.y + 1.0) * 57.0), u.x),
        u.y
    );
}

vec4 quatFromAxisAngle(vec3 axis, float angle) {
    float halfAngle = angle * 0.5;
    float s = sin(halfAngle);
    return vec4(axis * s, cos(halfAngle));
}

vec3 rotate(vec3 v, vec4 q) {
    vec3 t = 2.0 * cross(q.xyz, v);
    return v + q.w * t + cross(q.xyz, t);
}

float layeredNoise(vec3 p, float amplitude, float frequency, float persistence, int octaves) {
    float n = 0.0;
    for (int i = 0; i < octaves; i++) {
        n += noise(p.xy * frequency) * amplitude;
        amplitude *= persistence;
        frequency *= 2.0;
       p = rotate(p, quatFromAxisAngle(normalize(vec3(1.0, 0.5, 0.0)), radians(45.0)));
    }
    return n;
}


vec3 colorBanding(float n) {
    vec3 color = vec3(0);
    if (n > 0.95) return vec3(0.0, 0.1, 0.5);
    else if (n > 0.85) color = vec3(0.0, 0.3, 0.7);
    else if (n > 0.75) color = vec3(0.0, 0.5, 0.8);
    else if (n > 0.65) color = vec3(0.0, 0.7, 0.7);
    else if (n > 0.55) color = vec3(0.8, 0.6, 0.4);
    else if (n > 0.45) color = vec3(0.0, 0.6, 0.0);
    else color = vec3(0.4, 0.6, 0.1);
    return color;
}

float sphere(vec3 ro, vec3 rd, float radius) {
    float b = dot(ro, rd);
    float c = dot(ro, ro) - radius * radius;
    float h = b * b - c;
    if (h < 0.0) return -1.0;
    // if (true) return 15.0;
    return -b - sqrt(h); // 返回交点距离
}

void mainImage(out vec4 fragColor, in vec2 fragCoord) {
    vec2 uv = (fragCoord - iResolution.xy * 0.5) / iResolution.y;
    // if(uv.y>0.0) {
	// 	return;
	// }
	vec3 ro_planet = vec3(0.0, 0.0, 2.4);
    vec3 ro_clouds = vec3(0.0, 0.0, 2.3);
    vec3 rd = normalize(vec3(uv, -1.0));
    float t_planet = sphere(ro_planet, rd, 1.0);
    float t_clouds = sphere(ro_clouds, rd, 1.0);
    if (t_clouds < 0.0) {
        fragColor = vec4(1.0, 1.0, 0.0, 0.0);
        // fragColor = vec4(1.0, 1.0, 0.0, 1.0);
    } else {
        vec3 pos = ro_planet + t_planet * rd;
        vec4 q = quatFromAxisAngle(normalize(vec3(1.0, 1.0, 0.0)), iTime);
        // pos = rotate(pos, q);
        vec3 norm = normalize(pos);
        float noiseValue = layeredNoise(norm * 3.1415, 0.75, 1.0, 0.5, 3);
        vec4 color = vec4(colorBanding(noiseValue), 1.0);

        if (t_planet < 0.0) {
            color = vec4(0.0);
			// color = vec4(0.0, 1.0, 0.0, 1.0);
        }
		
		fragColor = color;

    }
}

答:(思考过程太长,有兴趣看末尾截图)

ini 复制代码
import * as THREE from 'three';

// Replicate GLSL functions in JavaScript
function hash(n) {
    const s = Math.sin(n) * 53738.14534123;
    return s - Math.floor(s);
}

function mix(a, b, t) {
    return a + (b - a) * t;
}

function noise(x, y) {
    const ipx = Math.floor(x);
    const ipy = Math.floor(y);
    const ux = x - ipx;
    const uy = y - ipy;

    const smoothUx = ux * ux * (3 - 2 * ux);
    const smoothUy = uy * uy * (3 - 2 * uy);

    const a = hash(ipx + ipy * 57);
    const b = hash(ipx + 1 + ipy * 57);
    const c = hash(ipx + (ipy + 1) * 57);
    const d = hash(ipx + 1 + (ipy + 1) * 57);

    const ab = mix(a, b, smoothUx);
    const cd = mix(c, d, smoothUx);
    return mix(ab, cd, smoothUy);
}

function colorBanding(n) {
    if (n > 0.95) return new THREE.Color(0.0, 0.1, 0.5);
    else if (n > 0.85) return new THREE.Color(0.0, 0.3, 0.7);
    else if (n > 0.75) return new THREE.Color(0.0, 0.5, 0.8);
    else if (n > 0.65) return new THREE.Color(0.0, 0.7, 0.7);
    else if (n > 0.55) return new THREE.Color(0.8, 0.6, 0.4);
    else if (n > 0.45) return new THREE.Color(0.0, 0.6, 0.0);
    else return new THREE.Color(0.4, 0.6, 0.1);
}

function generatePlanetTexture(width = 1024, height = 512) {
    const canvas = document.createElement('canvas');
    canvas.width = width;
    canvas.height = height;
    const ctx = canvas.getContext('2d');
    const imageData = ctx.createImageData(width, height);
    const data = imageData.data;

    const octaves = 3;
    const amplitude = 0.75;
    const persistence = 0.5;
    const rotationAxis = new THREE.Vector3(1, 0.5, 0).normalize();
    const rotationAngle = 45 * Math.PI / 180;
    const rotationQuaternion = new THREE.Quaternion().setFromAxisAngle(rotationAxis, rotationAngle);

    for (let y = 0; y < height; y++) {
        for (let x = 0; x < width; x++) {
            const u = x / width;
            const v = y / height;

            // Convert to spherical coordinates
            const phi = u * 2 * Math.PI;
            const theta = v * Math.PI - Math.PI / 2;
            const normal = new THREE.Vector3(
                Math.cos(theta) * Math.cos(phi),
                Math.cos(theta) * Math.sin(phi),
                Math.sin(theta)
            );

            let p = normal.clone().multiplyScalar(Math.PI);
            let currentAmplitude = amplitude;
            let currentFrequency = 1.0;
            let noiseValue = 0;

            for (let i = 0; i < octaves; i++) {
                const nx = p.x * currentFrequency;
                const ny = p.y * currentFrequency;
                noiseValue += noise(nx, ny) * currentAmplitude;

                currentAmplitude *= persistence;
                currentFrequency *= 2.0;
                p.applyQuaternion(rotationQuaternion);
            }

            const color = colorBanding(noiseValue);
            const idx = (y * width + x) * 4;
            data[idx] = color.r * 255;
            data[idx + 1] = color.g * 255;
            data[idx + 2] = color.b * 255;
            data[idx + 3] = 255;
        }
    }

    ctx.putImageData(imageData, 0, 0);
    return canvas;
}

// Three.js setup
const scene = new THREE.Scene();
const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
const renderer = new THREE.WebGLRenderer({ antialias: true });
renderer.setSize(window.innerWidth, window.innerHeight);
document.body.appendChild(renderer.domElement);

// Create sphere with generated texture
const texture = new THREE.CanvasTexture(generatePlanetTexture());
const geometry = new THREE.SphereGeometry(1, 64, 64);
const material = new THREE.MeshBasicMaterial({ map: texture });
const sphere = new THREE.Mesh(geometry, material);
scene.add(sphere);

camera.position.z = 3;

function animate() {
    requestAnimationFrame(animate);
    sphere.rotation.y += 0.005;
    renderer.render(scene, camera);
}
animate();

// Handle window resize
window.addEventListener('resize', () => {
    camera.aspect = window.innerWidth / window.innerHeight;
    camera.updateProjectionMatrix();
    renderer.setSize(window.innerWidth, window.innerHeight);
});

效果评估与总结

无需修改就能成功运行,完美翻译shadertoy里的glsl代码并复刻效果。感觉 DeepSeek 太厉害了!

用了 THREE.CanvasTexture,比我想象的要简单。本以为要生成多张贴图或者拼Atlas,结果一张画布贴图解决,直接运用到球体,厉害。

SphereGeometry -- three.js docs

CanvasTexture -- three.js docs

优点:性能好。

缺点:1. 分辨率取决于贴图,肯定没纯噪声那么精细。

  1. 南北极会有拉伸?

把geometry改成平面,可以看贴图本身: geometry = new THREE.PlaneGeometry(64, 64);

竟有澳洲:

效果图:

( octave=3 效果)

( octave=8 效果)

码上掘金:

github :

手机坏了,二次验证APP无法使用,登录不了。。

等我修好手机再说。

相关推荐
李斯维11 天前
深度解析 Android Matrix 变换(二):组合变换 pre、post
android·canvas·图形学
李斯维19 天前
深入理解 Android Canvas 变换:缩放、旋转、平移全解析(一)
android·canvas·图形学
Uzuki1 个月前
Vulkan环境配置 | vscode+msvc 解决方案
vscode·游戏开发·图形学·c/c++
charon87782 个月前
Games202Lecture 6 Real-time Environment Mapping
图形学
白鹭float.3 个月前
【OpenGL/Assimp】渲染模型、半透明材质与封装光源
c++·图形学·opengl·assimp
白鹭float.3 个月前
【OpenGL/C++】面向对象扩展——测试环境
c++·图形学·opengl
三翼鸟数字化技术团队3 个月前
模型工作流:自动化的模型内部三角面剔除
计算机图形学·图形学
ttod_qzstudio4 个月前
Unity中Mesh重叠顶点合并参考及其应用
unity·图形学
Ian10254 个月前
《Learn Three.js》学习(3)光源
前端·javascript·学习·webgl·图形学·三维·三维光源