Cesium里的postProcessStages实现暗黑样式的天地图

参考许多可视化大屏,很多的样式都是暗黑的看起来非常有科技感,但大多数都不是天地图底图的,对此,笔者对于实现这个效果进行了研究,总结了下面的两个方法:

方法一:改CSS里的filter滤镜

设置div容器的id或者class,通过css里去改

推荐的属性值:

复制代码
brightness(0.88) contrast(0.95) grayscale(0) hue-rotate(180deg) opacity(1) saturate(2.5) sepia(0.5) invert(1)

效果还可以,地图上的地铁线的颜色也不会影响,但是唯一的缺点就是由于是加在div地图容器上的,地图的弹窗也会变为暗黑的特效,但是优点就是可以兼容所有的GIS前端地图框架,不管是二维的leaflet、openlayers,还是三维的Cesium、ArcGIS api for js。

方法二:对于Cesium,可以使用PostProcessStage

代码如下:

javascript 复制代码
  this.viewer.scene.postProcessStages.add(new Cesium.PostProcessStage({
            name: 'css-like-dark-filter',
            fragmentShader: `
       uniform sampler2D colorTexture;
in vec2 v_textureCoordinates;
out vec4 fragColor;

// 工具函数:RGB -> HSV
vec3 rgb2hsv(vec3 c) {
    vec4 K = vec4(0.0, -1.0 / 3.0, 2.0 / 3.0, -1.0);
    vec4 p = mix(vec4(c.bg, K.wz), vec4(c.gb, K.xy), step(c.b, c.g));
    vec4 q = mix(vec4(p.xyw, c.r), vec4(c.r, p.yzx), step(p.x, c.r));
    float d = q.x - min(q.w, q.y);
    float e = 1e-10;
    return vec3(abs(q.z + (q.w - q.y) / (6.0 * d + e)), d / (q.x + e), q.x);
}

// 工具函数:HSV -> RGB
vec3 hsv2rgb(vec3 c) {
    vec4 K = vec4(1.0, 2.0 / 3.0, 1.0 / 3.0, 3.0);
    vec3 p = abs(fract(c.xxx + K.xyz) * 6.0 - K.www);
    return c.z * mix(K.xxx, clamp(p - K.xxx, 0.0, 1.0), c.y);
}

void main() {
    vec4 texColor = texture(colorTexture, v_textureCoordinates);
    vec3 color = texColor.rgb;

    // 1. invert(1)
    color = 1.0 - color;

    // 2. sepia(0.5)
    vec3 sepia = vec3(
        dot(color, vec3(0.393, 0.769, 0.189)),
        dot(color, vec3(0.349, 0.686, 0.168)),
        dot(color, vec3(0.272, 0.534, 0.131))
    );
    color = mix(color, sepia, 0.5);

    // 3. saturate(2.5)
    float luminance = dot(color, vec3(0.299, 0.587, 0.114));
    vec3 gray = vec3(luminance);
    color = mix(gray, color, 2.5);

    // 4. hue-rotate(180deg) → add 0.5 to hue (since 360° = 1.0 in normalized hue)
    vec3 hsv = rgb2hsv(color);
    hsv.x = fract(hsv.x + 0.5); // 180° rotation
    color = hsv2rgb(hsv);

    // 5. contrast(0.95)
    color = (color - 0.5) * 0.95 + 0.5;

    // 6. brightness(0.88)
    color *= 0.88;

    // Clamp to [0,1] to avoid overflow
    color = clamp(color, 0.0, 1.0);

    fragColor = vec4(color, texColor.a);
}
    `,
            uniforms: {}
        }));
    }

代码里用反引号(`````)包裹的那段内容是 GLSL(OpenGL Shading Language)代码 ,具体来说,是一段 fragment shader(片段着色器)

  • GLSL 是一种用于编写 GPU 着色器程序的 C 风格语言。
  • 它运行在显卡上,用于控制图形渲染管线中的顶点(vertex shader)和像素/片段(fragment shader)处理。
  • 在 CesiumJS 中,PostProcessStage 允许你通过自定义 fragment shader 对整个场景画面进行后处理(post-processing),比如添加滤镜、模糊、夜视效果等。

效果:

不会改css,地图里的弹窗样式也不会改变。

相关推荐
yuki_uix3 分钟前
虚拟 DOM 与 Diff 算法——React 性能优化的底层逻辑
前端·react.js·面试
yuki_uix4 分钟前
从输入 URL 到页面显示——浏览器工作原理全解析
前端·面试
weixin_4080996721 分钟前
【完整教程】天诺脚本如何调用 OCR 文字识别 API?自动识别屏幕文字实战(附代码)
前端·人工智能·后端·ocr·api·天诺脚本·自动识别文字脚本
吴声子夜歌22 分钟前
ES6——Generator函数详解
前端·javascript·es6
吴声子夜歌24 分钟前
ES6——Set和Map详解
前端·javascript·es6
码喽7号1 小时前
vue学习四:Axios网络请求
前端·vue.js·学习
粥里有勺糖2 小时前
视野修炼-技术周刊第129期 | 上一次古法编程是什么时候
前端·javascript·github
whuhewei2 小时前
JS获取CSS动画的旋转角度
前端·javascript·css
蓝黑20202 小时前
Vue组件通信之v-model
前端·javascript·vue
像素之间2 小时前
为什么运行时要加set NODE_OPTIONS=--openssl-legacy-provider && vue-cli-service serve
前端·javascript·vue.js