简体   繁体   English

JS-canvas.toDataURL失败

[英]JS - canvas.toDataURL Fails

I've been working with some google maps recently, and I needed to rotate the icon for a map marker. 我最近一直在使用一些Google地图,并且需要旋转地图标记的图标。 I did some research, and came across This Answer . 我做了一些研究,发现了“ 答案”

Unfortunately, it's not working properly. 不幸的是,它不能正常工作。 It won't render the image I need. 它不会渲染我需要的图像。 The URL produced by canvas.toDataURL is a blank image (of the correct size). canvas.toDataURL生成的URL是空白图像(大小正确)。

Here is the code: 这是代码:

var latlong0=new google.maps.LatLng(44.422036688671, -73.857908744819);    function initialize() {
  var mapOptions = {
    center: { lat: 44.422036688671, lng: -73.857908744819},
    zoom: 0
  };
  var map = new google.maps.Map(document.getElementById('map-canvas'),
      mapOptions);
  var airplane = 'img/airplane_icon.png';

  var infowindow0=new google.maps.InfoWindow({
      content: `
      <div>
        <p class='spectxt'>&lt;Flight #&gt;</p>
    <div style='font-size:10px;'>
          <p class='gentxtl'>Airline: <span class='acptxt'>Aero Test Ltd</span></p>
      <p class='gentxtl'>Aircraft: <span class='acptxt'>Boeing 717-200 HGW</span></p>
      <p class='gentxtl'>From: <span class='acptxt'>YYZ</span></p>
      <p class='gentxtl'>To: <span class='acptxt'>YHZ</span></p>
        </div>
      </div>
      `
    });

  var mark0=new google.maps.Marker({
    position: latlong0,
    map: map,
    title:'Aircraft',
    icon: {
            url: RotateIcon
                .makeIcon(
                    '/img/airplane_icon.png')
                .setRotation({deg: 78})
                .getUrl()
        }
  });
  mark0.setMap(map);mark0.addListener('click', function() {
      infowindow0.open(map, mark0);
    });    }
google.maps.event.addDomListener(window, 'load', initialize);

The RotateIcon code: RotateIcon代码:

  var RotateIcon = function(options){
    this.options = options || {};
    this.rImg = options.img || new Image();
    this.rImg.src = this.rImg.src || this.options.url || '';
    this.options.width = this.options.width || this.rImg.width || 50;
    this.options.height = this.options.height || this.rImg.height || 50;
    var canvas = document.createElement("canvas");
    canvas.width = this.options.width;
    canvas.height = this.options.height;
    this.context = canvas.getContext("2d");
    this.canvas = canvas;
};
RotateIcon.makeIcon = function(url) {
    return new RotateIcon({url: url});
};
RotateIcon.prototype.setRotation = function(options){
    var canvas = this.context,
        angle = options.deg ? options.deg * Math.PI / 180:
            options.rad,
        centerX = this.options.width/2,
        centerY = this.options.height/2;

    canvas.clearRect(0, 0, this.options.width, this.options.height);
    canvas.save();
    canvas.translate(centerX, centerY);
    canvas.rotate(angle);
    canvas.translate(-centerX, -centerY);
    canvas.drawImage(this.rImg, 0, 0);
    canvas.restore();
    return this;
};
RotateIcon.prototype.getUrl = function(){
    return this.canvas.toDataURL('image/png');
};

The image is on the same domain as the web page, and both resources are regular http (no https). 该图像与网页位于同一域中,并且两个资源均为常规http(无https)。 Chrome DevTools reports no warnings or errors. Chrome DevTools不报告警告或错误。 canvas.toDataURL outputs this canvas.toDataURL输出

I'm using Chrome 48 on Windows 8.1 我在Windows 8.1上使用Chrome 48

I've noticed something else odd with it. 我注意到其他奇怪的地方。 If I change the URL to /img/logo.png , it creates the image I want, but with the wrong sizing. 如果将URL更改为/img/logo.png ,它将创建所需的图像,但尺寸错误。 canvas.toDataURL outputs this canvas.toDataURL输出

-EDIT- -编辑-
/img/airplane_icon.png can be found here /img/airplane_icon.png可以在这里找到
/img/logo.png can be found here /img/logo.png可以在这里找到

-EDIT 2- 编辑2
So, I made a test page , and found it working.... So I went back to my first page, and noticed that if I have <img src="/img/airplane_icon.png"> after the map canvas, the page will work, but if I remove it, the page will not. 因此,我做了一个测试页 ,发现它可以正常工作。...所以我回到第一页,注意到如果在地图画布后有<img src="/img/airplane_icon.png">页面可以使用,但是如果我将其删除,该页面将无法使用。 Strange. 奇怪。 For now I'm just going to add a display:none to the img, but does anyone know why the img has to be included for the page to work? 现在,我只想在img上添加display:none ,但是没有人知道为什么必须包含img才能使页面正常工作吗?

Seems like a problem with the asynchronous load of the image if it is not already available in the cache. 如果图像的异步加载在缓存中尚不可用,则好像出现问题。 One option to address that is to load the icon as a data URI: 一种解决方法是将图标作为数据URI加载:

var airplaneIcon = new RotateIcon({
  url: '',
  height: 50,
  width: 50
})
var mark0 = new google.maps.Marker({
  position: latlong0,
  map: map,
  title: 'Aircraft',
  icon: {
  // size: new google.maps.Size(128, 128),
    scaledSize: new google.maps.Size(20, 20),
    anchor: new google.maps.Point(10, 10),
    url: airplaneIcon.setRotation({
        deg: 78
    })
    .getUrl()
  }
});

proof of concept fiddle 概念证明小提琴

code snippet: 代码段:

 var RotateIcon = function(options) { this.options = options || {}; this.rImg = options.img || new Image(); this.rImg.src = this.rImg.src || this.options.url || ''; this.options.width = this.options.width || this.rImg.width || 50; this.options.height = this.options.height || this.rImg.height || 50; var canvas = document.createElement("canvas"); canvas.width = this.options.width; canvas.height = this.options.height; this.context = canvas.getContext("2d"); this.canvas = canvas; }; RotateIcon.makeIcon = function(url) { return new RotateIcon({ url: url }); }; RotateIcon.prototype.setRotation = function(options) { var canvas = this.context, angle = options.deg ? options.deg * Math.PI / 180 : options.rad, centerX = this.options.width / 2, centerY = this.options.height / 2; canvas.clearRect(0, 0, this.options.width, this.options.height); canvas.save(); canvas.translate(centerX, centerY); canvas.rotate(angle); canvas.translate(-centerX, -centerY); canvas.drawImage(this.rImg, 0, 0); canvas.restore(); return this; }; RotateIcon.prototype.getUrl = function() { return this.canvas.toDataURL('image/png'); }; var map; var latlong0 = new google.maps.LatLng(44.422036688671, -73.857908744819); function initialize() { var mapOptions = { center: { lat: 44.422036688671, lng: -73.857908744819 }, zoom: 5 }; map = new google.maps.Map(document.getElementById('map-canvas'), mapOptions); var airplane = ''; var infowindow0 = new google.maps.InfoWindow({ content: ` <div> <p class='spectxt'>&lt;Flight #&gt;</p> <div style='font-size:10px;'> <p class='gentxtl'>Airline: <span class='acptxt'>Aero Test Ltd</span></p> <p class='gentxtl'>Aircraft: <span class='acptxt'>Boeing 717-200 HGW</span></p> <p class='gentxtl'>From: <span class='acptxt'>YYZ</span></p> <p class='gentxtl'>To: <span class='acptxt'>YHZ</span></p> </div> </div> ` }); var airplaneIcon = new RotateIcon({ url: '', height: 50, width: 50 }) var mark0 = new google.maps.Marker({ position: latlong0, map: map, title: 'Aircraft', icon: { // size: new google.maps.Size(128, 128), scaledSize: new google.maps.Size(20, 20), anchor: new google.maps.Point(10, 10), url: airplaneIcon.setRotation({ deg: 78 }) .getUrl() } }); mark0.setMap(map); mark0.addListener('click', function() { infowindow0.open(map, mark0); }); } google.maps.event.addDomListener(window, 'load', initialize); 
 html, body, #map-canvas { height: 100%; width: 100%; margin: 0px; padding: 0px } 
 <script src="https://maps.googleapis.com/maps/api/js"></script> <div id="map-canvas"></div> 

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM