- Accessible Map
- Advanced Mapbox Vector Tiles
- 高级地图定位
- 地图加载 GIF 动图
- ArcGIS REST Feature Service
- Attributions
- 波段对比度调节
- Bing Maps
- Box Selection
- Canvas Tiles
- CartoDB source example
- Change Tile Layer Style
- Cloud Optimized GeoTIFF (COG)
- Cloud Optimized GeoTIFF (COG) from a Blob
- Clustered Features
- Color Manipulation
- Constrained Extent
- Constrained Zoom
- Custom Animation
- Custom Canvas Tiles
- 自定义圆形渲染
- 自定义控件
- 自定义拖拽上传展示数据(KMZ)
- 自定义拖拽上传展示数据 (MVT 预览)
- 自定义触发器渲染
- 自定义交互
- 地图自定义 dom 元素(element)
- 自定义鹰眼
- 自定义面状图形要素样式
- 自定义瓦片 WMS
- 自定义提示信息
- d3 集成
- 数据瓦片
- 设备定位
- 拖拽加载数据
- 拖拽 Image Vector
- 拖拽,旋转和缩放
- 绘制和修改图形要素(Features)
- 绘制和编辑测地线(Geodesic)圆
- 绘制图形要素(Feature)
- 绘制规则图形
- 动态聚合
- 动画展示数据
- 地震位置聚合图
- 地震位置热力图
- 渲染 KML 中的地震数据
- 自定义图形渲染地震数据
- 可编辑的 ArcGIS REST 图形要素服务
- EPSG:4326
- 导出 PDF 示例
- 范围(Extent)交互
- 外部地图
- 使用 WebGL 过滤图形要素
- 航班飞行动画
- Fractal 渲染
- 自由绘制
- 全屏控件
- 带有扩展element的全屏控件
- 全屏拖拽,旋转,缩放
- Geographic Coordinates
- Geographic Editing
- GeoJSON
- 集成 geojson-vt
- 地理位置
- GeoTIFF 瓦片金字塔
- GeoTIFF with Overviews
- GPX 数据
- 高 DPI WMTS
- 点击容许偏差
- Icon 比例
- Icon 颜色
- Icon 位置修改
- Icon 像素操作
- Icon Sprites with WebGL
- Icon 渲染
- IGC 数据
- IGN WMTS
- IIIF 影像 API
- ArcGIS 图像地图服务
- 图片过滤
- 图像载入事件
- 图像重投影
- 即时渲染 (地理)
- 插值
- 集成 JSTS
- KML
- Layer 裁剪
- 图层组
- 图层 最小/最大 分辨率
- 图层透明度
- 图层鼠标透视
- 图层卷帘
- 图层卷帘 (WebGL)
- 图层层级(Z-Index)
- 图层缩放限制
- 懒加载资源
- 限制图层范围
- 线段箭头
- 载入中旋转动画
- 局部的 OpenStreetMap
- 鼠标放大镜效果
- 地图打印(导出)
- 地图经纬网格
- 地图链接地址
- Mapbox 矢量图层
- Mapbox 矢量瓦片
- Mapbox-gl 图层
- MapGuide Untiled
- 地图点图标动画
- 测量
- 使用矢量样式测量
- 修改图形要素(Features)
- 修改图形要素测试
- 显示鼠标位置
- 地图平移(拖拽)结束事件
- 定位控件
- NDVI from a Sentinel 2 COG
- 动态颜色斜率 NDVI
- NDVI+NDWI from two 16-bit COGs
- OGC 地图瓦片
- OGC 地图瓦片 (地理)
- OGC 矢量瓦片
- OpenStreetMap 重投影
- 带有比例尺控件的 OpenStreetMap 重投影
- OSM 矢量瓦片
- OSM XML
- 覆盖物(Overlay)
- 鹰眼地图控件
- 页面滚动
- 拖拽和页面滚动
- Permalink
- Pinch Zoom
- Popup
- Preload Tiles
- 按比例导出地图
- 投影和比例尺
- 栅格重投影
- 栅格资源
- 区域增长
- Regular Shapes
- Render geometries to a canvas
- Rendering 16-bit NumpyTiles
- Reprojection with EPSG.io Search
- Reusable Source
- Rich Text Labels
- Scale and Rotate using Modify Interaction
- Scale Line
- Sea Level
- Sea Level (with WebGL)
- Select Features
- Select Features by Hover
- Select multiple Features
- Semi-Transparent Layer
- Shaded Relief
- Shaded Relief (with WebGL)
- Shared Views
- Simple Map
- Single Image WMS
- Single Image WMS with Proj4js
- Smoothing lines using Chaikins algorithm
- Snap Interaction
- Sphere Mollweide
- Stamen Tiles
- Static Image
- Street Labels
- Style renderer
- Styling feature with CanvasGradient or CanvasPattern
- SVG Layer
- Synthetic Lines
- Synthetic Points
- Teleporting Maps
- Tile Load Events
- Tile Transitions
- Tiled ArcGIS MapServer
- Tiled WMS
- Tiled WMS Wrapping
- TileJSON
- Timezones in KML
- Tissot Indicatrix
- TopoJSON
- topolis integration
- Tracing around a polygon
- Translate Features
- turf.js
- UTFGrid
- Vector Clipping Layer
- Vector Image Layer
- Vector Label Decluttering
- Vector Labels
- Vector Labels - Justify Text
- Vector Layer
- Vector Layer Hit Detection
- Vector Tile Info
- Vector Tile Selection
- Vector tiles created from a Mapbox Style object
- Vector tiles in EPSG:4326
- Vector tiles rendered in an offscreen canvas
- View Animation
- View Min-Zoom
- View Padding
- View Rotation
- WebGL points layer
- WebGL Tile Layer Styles
- WebGL Tiles
- WebGL Vector Layer
- WFS
- WFS - GetFeature
- WFS with geographic coordinates
- Wind Arrows
- WKB
- WKT
- WMS 512x256 Tiles
- WMS Capabilities Parsing
- WMS GetFeatureInfo (Image Layer)
- WMS GetFeatureInfo (Layers)
- WMS GetFeatureInfo (Tile Layer)
- WMS GetLegendGraphic
- WMS 时序渲染
- 无投影 WMS
- WMTS
- WMTS Capabilities 解析
- 从 Capabilities 加载 WMTS
- WMTS 瓦片转换
- XYZ
- XYZ Esri
- XYZ 视网膜瓦片
- 缩放滑块
- Zoomify
Shaded Relief
Calculate shaded relief from elevation data
This example uses a ol/source/Raster
to generate data based on another source. The raster source accepts any number of input sources (tile or image based) and runs a pipeline of operations on the input data. The return from the final operation is used as the data for the output source.
In this case, a single tiled source of elevation data is used as input. The shaded relief is calculated in a single "image" operation. By setting operationType: 'image'
on the raster source, operations are called with an ImageData
object for each of the input sources. Operations are also called with a general purpose data
object. In this example, the sun elevation and azimuth data from the inputs above are assigned to this data
object and accessed in the shading operation. The shading operation returns an array of ImageData
objects. When the raster source is used by an image layer, the first ImageData
object returned by the last operation in the pipeline is used for rendering.
main.js
import 'ol/ol.css';
import Map from 'ol/Map';
import View from 'ol/View';
import {Image as ImageLayer, Tile as TileLayer} from 'ol/layer';
import {OSM, Raster, XYZ} from 'ol/source';
/**
* Generates a shaded relief image given elevation data. Uses a 3x3
* neighborhood for determining slope and aspect.
* @param {Array<ImageData>} inputs Array of input images.
* @param {Object} data Data added in the "beforeoperations" event.
* @return {ImageData} Output image.
*/
function shade(inputs, data) {
const elevationImage = inputs[0];
const width = elevationImage.width;
const height = elevationImage.height;
const elevationData = elevationImage.data;
const shadeData = new Uint8ClampedArray(elevationData.length);
const dp = data.resolution * 2;
const maxX = width - 1;
const maxY = height - 1;
const pixel = [0, 0, 0, 0];
const twoPi = 2 * Math.PI;
const halfPi = Math.PI / 2;
const sunEl = (Math.PI * data.sunEl) / 180;
const sunAz = (Math.PI * data.sunAz) / 180;
const cosSunEl = Math.cos(sunEl);
const sinSunEl = Math.sin(sunEl);
let pixelX,
pixelY,
x0,
x1,
y0,
y1,
offset,
z0,
z1,
dzdx,
dzdy,
slope,
aspect,
cosIncidence,
scaled;
function calculateElevation(pixel) {
// The method used to extract elevations from the DEM.
// In this case the format used is
// red + green * 2 + blue * 3
//
// Other frequently used methods include the Mapbox format
// (red * 256 * 256 + green * 256 + blue) * 0.1 - 10000
// and the Terrarium format
// (red * 256 + green + blue / 256) - 32768
//
return pixel[0] + pixel[1] * 2 + pixel[2] * 3;
}
for (pixelY = 0; pixelY <= maxY; ++pixelY) {
y0 = pixelY === 0 ? 0 : pixelY - 1;
y1 = pixelY === maxY ? maxY : pixelY + 1;
for (pixelX = 0; pixelX <= maxX; ++pixelX) {
x0 = pixelX === 0 ? 0 : pixelX - 1;
x1 = pixelX === maxX ? maxX : pixelX + 1;
// determine elevation for (x0, pixelY)
offset = (pixelY * width + x0) * 4;
pixel[0] = elevationData[offset];
pixel[1] = elevationData[offset + 1];
pixel[2] = elevationData[offset + 2];
pixel[3] = elevationData[offset + 3];
z0 = data.vert * calculateElevation(pixel);
// determine elevation for (x1, pixelY)
offset = (pixelY * width + x1) * 4;
pixel[0] = elevationData[offset];
pixel[1] = elevationData[offset + 1];
pixel[2] = elevationData[offset + 2];
pixel[3] = elevationData[offset + 3];
z1 = data.vert * calculateElevation(pixel);
dzdx = (z1 - z0) / dp;
// determine elevation for (pixelX, y0)
offset = (y0 * width + pixelX) * 4;
pixel[0] = elevationData[offset];
pixel[1] = elevationData[offset + 1];
pixel[2] = elevationData[offset + 2];
pixel[3] = elevationData[offset + 3];
z0 = data.vert * calculateElevation(pixel);
// determine elevation for (pixelX, y1)
offset = (y1 * width + pixelX) * 4;
pixel[0] = elevationData[offset];
pixel[1] = elevationData[offset + 1];
pixel[2] = elevationData[offset + 2];
pixel[3] = elevationData[offset + 3];
z1 = data.vert * calculateElevation(pixel);
dzdy = (z1 - z0) / dp;
slope = Math.atan(Math.sqrt(dzdx * dzdx + dzdy * dzdy));
aspect = Math.atan2(dzdy, -dzdx);
if (aspect < 0) {
aspect = halfPi - aspect;
} else if (aspect > halfPi) {
aspect = twoPi - aspect + halfPi;
} else {
aspect = halfPi - aspect;
}
cosIncidence =
sinSunEl * Math.cos(slope) +
cosSunEl * Math.sin(slope) * Math.cos(sunAz - aspect);
offset = (pixelY * width + pixelX) * 4;
scaled = 255 * cosIncidence;
shadeData[offset] = scaled;
shadeData[offset + 1] = scaled;
shadeData[offset + 2] = scaled;
shadeData[offset + 3] = elevationData[offset + 3];
}
}
return {data: shadeData, width: width, height: height};
}
const elevation = new XYZ({
url: 'https://{a-d}.tiles.mapbox.com/v3/aj.sf-dem/{z}/{x}/{y}.png',
crossOrigin: 'anonymous',
});
const raster = new Raster({
sources: [elevation],
operationType: 'image',
operation: shade,
});
const map = new Map({
target: 'map',
layers: [
new TileLayer({
source: new OSM(),
}),
new ImageLayer({
opacity: 0.3,
source: raster,
}),
],
view: new View({
extent: [-13675026, 4439648, -13580856, 4580292],
center: [-13615645, 4497969],
minZoom: 10,
maxZoom: 16,
zoom: 13,
}),
});
const controlIds = ['vert', 'sunEl', 'sunAz'];
const controls = {};
controlIds.forEach(function (id) {
const control = document.getElementById(id);
const output = document.getElementById(id + 'Out');
const listener = function () {
output.innerText = control.value;
raster.changed();
};
control.addEventListener('input', listener);
control.addEventListener('change', listener);
output.innerText = control.value;
controls[id] = control;
});
raster.on('beforeoperations', function (event) {
// the event.data object will be passed to operations
const data = event.data;
data.resolution = event.resolution;
for (const id in controls) {
data[id] = Number(controls[id].value);
}
});
index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Shaded Relief</title>
<!-- Pointer events polyfill for old browsers, see https://caniuse.com/#feat=pointer -->
<script src="https://unpkg.com/elm-pep"></script>
<!-- The line below is only needed for old environments like Internet Explorer and Android 4.x -->
<script src="https://cdn.polyfill.io/v3/polyfill.min.js?features=fetch,requestAnimationFrame,Element.prototype.classList,URL,TextDecoder,Number.isInteger"></script>
<style>
.map {
width: 100%;
height:400px;
}
table.controls td {
padding: 2px 5px;
}
table.controls td:nth-child(3) {
text-align: right;
min-width: 3em;
}
</style>
</head>
<body>
<div id="map" class="map"></div>
<table class="controls">
<tr>
<td><label for="vert">vertical exaggeration:</label></td>
<td><input id="vert" type="range" min="1" max="5" value="1"/></td>
<td><span id="vertOut"></span> x</td>
</tr>
<tr>
<td><label for="sunEl">sun elevation:</label></td>
<td><input id="sunEl" type="range" min="0" max="90" value="45"/></td>
<td><span id="sunElOut"></span> °</td>
</tr>
<tr>
<td><label for="sunAz">sun azimuth:</label></td>
<td><input id="sunAz" type="range" min="0" max="360" value="45"/></td>
<td><span id="sunAzOut"></span> °</td>
</tr>
</table>
<script src="main.js"></script>
</body>
</html>
package.json
{
"name": "shaded-relief",
"dependencies": {
"ol": "7.1.0"
},
"devDependencies": {
"parcel": "^2.0.0-beta.1"
},
"scripts": {
"start": "parcel index.html",
"build": "parcel build --public-url . index.html"
}
}
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论