UNPKG

gmap-ol

Version:

A set of helper classes for working with openLayers.

197 lines (177 loc) 7.74 kB
/* * @Author: xuhongling * @Date: 2022-08-16 17:37:59 * @Last Modified by: xuhongling * @Last Modified time: 2024-05-07 09:23:51 */ // 空白 base64 照片数据 const defaultLegendImage = ''; // 绘制文字到地图上面 function drawTextToMap(mapCanvas, mapContext, options) { const cW = mapCanvas.width; let clipHeight = mapCanvas.height; if (options.size) { clipHeight = options.size[1] ? options.size[1] : mapCanvas.height; clipHeight = clipHeight > mapCanvas.height ? mapCanvas.height : clipHeight; } const startY = (mapCanvas.height - clipHeight) / 2; mapContext.textAlign = 'center'; mapContext.shadowOffsetX = 0; mapContext.shadowOffsetY = 2; mapContext.shadowColor = 'rgba(240, 242, 245, 0.9)'; mapContext.shadowBlur = 4; if (options.title) { const titleName = options.title.name ? options.title.name : ''; mapContext.fillStyle = options.title.color ? options.title.color : '#416cf2'; mapContext.font = `bold ${options.title.fontSize ? options.title.fontSize : '26px'} sans-serif`; mapContext.fillText(titleName, cW / 2, 50 + startY); } if (options.subtitle) { const subTitleName = options.subtitle.name ? options.subtitle.name : ''; mapContext.fillStyle = options.subtitle.color ? options.subtitle.color : '#416cf2'; mapContext.font = `bold ${options.subtitle.fontSize ? options.subtitle.fontSize : '16px'} sans-serif`; mapContext.fillText(subTitleName, cW / 2, 90 + startY); } } // 把地图跟图例等合成图片 async function compositePictures(mapCanvas, mapContext, options) { // 加载图片,缓存到本地的canvas中(避免图片地址导致的下载跨域) /*img 规定要使用的图像、画布。 startX 可选。开始剪切的 x 坐标位置。 startY 可选。开始剪切的 y 坐标位置。 clipWidth 可选。被剪切图像的宽度。 clipHeight 可选。被剪切图像的高度。 x 在画布上放置图像的 x 坐标位置。 y 在画布上放置图像的 y 坐标位置。 width 可选。要使用的图像的宽度。(伸展或缩小图像) height 可选。要使用的图像的高度。(伸展或缩小图像) */ //用9参数的drawImage方法对图片进行裁减 let clipWidth = mapCanvas.width; let clipHeight = mapCanvas.height; if (options.imgSize) { clipWidth = options.imgSize[0] ? options.imgSize[0] : mapCanvas.width; clipHeight = options.imgSize[1] ? options.imgSize[1] : mapCanvas.height; clipWidth = clipWidth > mapCanvas.width ? mapCanvas.width : clipWidth; clipHeight = clipHeight > mapCanvas.height ? mapCanvas.height : clipHeight; } const startX = (mapCanvas.width - clipWidth) / 2; const startY = (mapCanvas.height - clipHeight) / 2; // 创建图例图片对象 let legendImg = new Image(); legendImg.setAttribute('crossOrigin', 'anonymous'); // 设置属性 legendImg.src = options.legend ? options.legend : defaultLegendImage; // 图例加载完成,为后面回去图例宽高用 await legendImageLoadingComplete(options); const imgWidth = legendImg ? legendImg.width : 0; const imgHeight = legendImg ? legendImg.height : 0; const imgX = startX + clipWidth - imgWidth - 10; const imgY = startY + clipHeight - imgHeight - 10; mapContext.drawImage(legendImg, imgX, imgY, imgWidth, imgHeight); // 如果导出报错说被污染,需要去地图的Layer下面的source加上 crossOrigin: 'anonymous', const type = options.type ? options.type : 'image/png'; const quality = options.quality ? options.quality : 0.8; const base64Image = await mapCanvas.toDataURL(type, quality); if (options?.down !== undefined && !options?.down ? false : true) { downloadByBlob(base64Image); } return base64Image; } async function legendImgLoad(options) { return new Promise((resolve) => { let legendImg = new Image(); legendImg.src = options.legend ? options.legend : defaultLegendImage; legendImg.onload = () => { resolve(); }; }); } async function legendImageLoadingComplete(options) { await legendImgLoad(options); } // 把地图下载成图片 function downloadByBlob(base64Image) { const image = new Image(); // 创建一个image标签 image.setAttribute('crossOrigin', 'anonymous'); // 设置属性 image.src = base64Image; // 设置src // 加载图片,缓存到本地的canvas中(避免图片地址导致的下载跨域) image.onload = () => { const blob = dataURItoBlob(base64Image); const url = URL.createObjectURL(blob); const name = `Map-Image-${Date.now()}`; downloadImage(url, name); URL.revokeObjectURL(url); // 用完释放URL对象 }; } // 下载图片到本地 function downloadImage(url, name) { const eleLink = document.createElement('a'); // 创建一个a标签 eleLink.download = name; // 下载命名 eleLink.href = url; // 下载地址 eleLink.click(); // 模拟点击 eleLink.remove(); // 模拟点击移除 } // base64 转 blob function dataURItoBlob(dataURI) { let byteString; if (dataURI.split(',')[0].indexOf('base64') >= 0) { byteString = atob(dataURI.split(',')[1]); } else { byteString = unescape(dataURI.split(',')[1]); } const mimeString = dataURI.split(',')[0].split(':')[1].split(';')[0]; const ia = new Uint8Array(byteString.length); for (let i = 0; i < byteString.length; i++) { ia[i] = byteString.charCodeAt(i); } return new Blob([ia], { type: mimeString }); } /** *导出地图成图片下载到本地 */ function exportMapToImage(olMap, options) { if (olMap === null || olMap === undefined) { return; // 终止函数执行 } // 创建画布 const mapCanvas = document.createElement('canvas'); const mapContext = mapCanvas.getContext('2d'); const size = olMap.getSize(); mapCanvas.width = size[0]; mapCanvas.height = size[1]; // 获取地图图层信息 const canvasEl = document.querySelectorAll('.ol-layer canvas'); const canvasElValues = Object.values(canvasEl); for (let i = 0; i < canvasElValues.length; i++) { const canvas = canvasElValues[i]; if (canvas.width > 0) { const opacity = canvas.parentNode.style.opacity; mapContext.globalAlpha = opacity === '' ? 1 : Number(opacity); const transform = canvas.style.transform; const matrix = transform .match(/^matrix\(([^(]*)\)$/)[1] .split(',') .map(Number); CanvasRenderingContext2D.prototype.setTransform.apply(mapContext, matrix); let clipWidth = mapCanvas.width; let clipHeight = mapCanvas.height; if (options.size) { clipWidth = options.size[0] ? options.size[0] : mapCanvas.width; clipHeight = options.size[1] ? options.size[1] : mapCanvas.height; clipWidth = clipWidth > mapCanvas.width ? mapCanvas.width : clipWidth; clipHeight = clipHeight > mapCanvas.height ? mapCanvas.height : clipHeight; } const startX = (canvas.width - clipWidth) / 2; const startY = (canvas.height - clipHeight) / 2; mapContext.rect(startX, startY, clipWidth, clipHeight); mapContext.clip(); mapContext.drawImage(canvas, 0, 0); } } // 绘制标题文字到地图上面 drawTextToMap(mapCanvas, mapContext, options); // 把地图图例合成图片 let base64ImageData = compositePictures(mapCanvas, mapContext, options); return base64ImageData; } export default exportMapToImage;