Advanced topics

Access to Canvas Pixels

It is a great idea to use images from our CDN on HTML canvas. Unfortunately browsers restrict access to canvas if an image was downloaded from untrusted source. By default only images from the same origin are treated as trusted. If you’ll draw the image from other origins, the canvas will be marked as «dirty» and no pixel read will be possible from the canvas.

Calling getImageData() or toDataURL() will lead to SecurityError:

canvas.toDataURL();
// Uncaught DOMException: Failed to execute 'toDataURL' on 'HTMLCanvasElement': Tainted canvases may not be exported.
canvas.getContext('2d').getImageData(0, 0, 100, 100);
// Uncaught DOMException: Failed to execute 'getImageData' on 'CanvasRenderingContext2D': The canvas has been tainted by cross-origin data.

One of solutions is using crossOrigin image property to ask browser to request the image in a safe way.

$(function() {
  var imageUrl = 'https://ucarecdn.com/c5b7dd84-c0e2-48ff-babc-d23939f2c6b4/-/preview/240x240/-/quality/best/';
   
  var canvas = document.getElementById('s1-canvas');
  var img = new Image();
   
  img.crossOrigin = "Anonymous";
   
  img.onload = function() {
    canvas.width = this.width;
    canvas.height = this.height;
    canvas.getContext('2d').drawImage(this, 0, 0);
   
    canvas.toDataURL();
    canvas.getContext('2d').getImageData(0, 0, 100, 100);
  };
  img.src = imageUrl;
});
  
<canvas id="s1-canvas"></canvas>
  

Unfortunately this property is available only in Chrome and Firefox. But it is very easy to make such request manually with the help of XMLHttpRequest.

$(function() 
  function requestImage(imageUrl, callback) {
    var req = new XMLHttpRequest();
    req.onload = function() {
      var img = new Image();
      img.onload = function() {
        URL.revokeObjectURL(this.src);
        callback(img);
      };
      img.src = URL.createObjectURL(req.response);
    };
    req.open("get", imageUrl, true);
    req.responseType = 'blob';
    req.send();
  }
 
  function getAverageColor(canvas) {
    var ctx = canvas.getContext('2d');
    var data = ctx.getImageData(0, 0, canvas.width, canvas.height).data;
    var r = 0, g = 0, b = 0, a = 0;
    for (i = 0; i < data.length; i += 4) {
      r += data[i+0] * data[i+3];
      g += data[i+1] * data[i+3];
      b += data[i+2] * data[i+3];
      a += data[i+3];
    }
    return [r / a, g / a, b / a];
  }
 
  var canvas = document.getElementById('s2-canvas');
  var URL = window.URL || window.webkitURL;
 
  $('#s2-average-color').on('click', function() {
    var imageUrl = $('#s2-source').val();
    requestImage(imageUrl, function(img) {
      canvas.width = img.width;
      canvas.height = img.height;
      canvas.getContext('2d').drawImage(img, 0, 0);
 
      var colors = $.map(getAverageColor(canvas), function(x) {
        return Math.round(x).toString(16);
      });
      alert("The average color is #" + colors.join(''));
    })
  });
});
  
<input type="text" size="80" id="s2-source" value="https://ucarecdn.com/c5b7dd84-c0e2-48ff-babc-d23939f2c6b4/-/preview/250x250/"><br>
<button id="s2-average-color">Get average color</button><br>
<canvas id="s2-canvas"></canvas>
  


This works well in Chrome 19+, Firefox 6+, IE 10+, Safari 6+ and Opera 15+.

Uploaded Files Browser

Often you may want to allow user to choose previously uploaded files again. Uploadcare doesn’t store relations between users and files and also doesn’t allow the widget to list all files in the project for security reasons.

There are two ways to handle the files-to-user relations:

  1. You can store this information on your side and load list of user’s files using AJAX requests to your server. This is relatively complex way because you’ll need to implement some storage and endpoint which will return file list from this storage.
  2. List of previously uploaded files can be stored locally on user’s computer in browser’s local storage. This way user will not be able to choose files uploaded from other devices, but this doesn’t require any support on server side.

Following example shows how to add local storage history tab with following features:

  • Lists are separated by public key. If you are using widgets with two or more public keys, only files from current project will be shown.
  • List of last 50 files are shown.
  • If widget has images-only flag, only image files will be shown.
  • User can remove files from history.
uploadcare.registerTab('localhistory', function(container, button, dialogApi, settings, name) {
 
  function loadItems() {
    // Format: UUID isImage size filename
    var items = localStorage.getItem(localKey);
    if ( ! items) {
      return [];
    }
    items = items.split("\n");
    for (var i = 0; i < items.length; i++) {
      var v = items[i].split(' ');
      v.splice(3, v.length, v.slice(3, v.length).join(' '));
      v[1] = !!parseInt(v[1]);
      v[2] = parseInt(v[2]);
      items[i] = v;
    }
    return items;
  }
 
  function saveItems(items) {
    items = items.slice(0, 100);
    for (var i = 0; i < items.length; i++) {
      var v = items[i].slice();
      v[1] = 0 + v[1]; // bool → int
      items[i] = v.join(" ");
    }
    localStorage.setItem(localKey, items.join("\n"));
  }
 
  function addItem(fileInfo) {
    var items = loadItems();
    items = $.grep(items, function(v) {
      return v[0] !== fileInfo.uuid;
    });
    var item = [fileInfo.uuid, fileInfo.isImage, fileInfo.size, fileInfo.name];
    items.unshift(item);
    saveItems(items);
  }
 
  function removeItem(item) {
    var items = loadItems();
    items = $.grep(items, function(v) {
      return v[0] !== item[0];
    });
    saveItems(items);
  }
 
  function makeItem(data) {
    var html = $('<div class="uploadcare-file-item"></div>').append([
      $('<div class="uploadcare-file-item__preview"></div>').append(
        data[1]
        ?
        $('<img/>', {
          src: settings.cdnBase + "/" + data[0] + "/-/quality/lightest/" + (
            settings.imagesOnly
              ? "-/preview/340x340/"
              : "-/scale_crop/110x110/center/"
          )
        })
        :
        $('<div class="uploadcare-file-icon"></div>')
      ),
      $('<div class="uploadcare-file-item__name"></div>').text(data[3]),
      $('<div class="uploadcare-file-item__size"></div>').text(uc.utils.readableFileSize(data[2])),
      $('<div class="uploadcare-file-item__remove"></div>').append(
        $('<div class="uploadcare-remove"></div>')
          .on('click', function() {
            html.remove();
            removeItem(data);
          })
      )
    ]).on('click', function(e) {
        dialogApi.addFiles('uploaded', [data[0]]);
        e.preventDefault();
      });
    return html;
  }
 
  function populate(container) {
    var items = loadItems();
    for (var i = items.length - 1; i >= 0; i--) {
      var item = items[i];
      if (settings.imagesOnly && ! item[1]) {
        continue;
      }
      container.prepend(makeItem(item));
    }
  }
 
  var localStorage = window.localStorage;
  var localKey = "uploadcare_" + settings.publicKey;
  var $ = uploadcare.jQuery;
  var uc;
  if ( ! localStorage) {
    button.hide();
    return;
  }
  uploadcare.plugin(function(uploadcare) {
    uc = uploadcare;
  });
  dialogApi.fileColl.onAdd.add(function(file) {
    file.done(function(fileInfo) {
      addItem(fileInfo);
    })
  });
  populate($('<div class="uploadcare-file-list"></div>')
    .toggleClass('uploadcare-file-list_table', !settings.imagesOnly)
    .toggleClass('uploadcare-file-list_tiles', settings.imagesOnly)
    .appendTo($('<div class="uploadcare-dialog-padding">\
      <div class="uploadcare-dialog-title">Previous uploaded files</div>\
    </div>'
)
    .appendTo(container)));
});
 
UPLOADCARE_LOCALE_TRANSLATIONS = {
  dialog: {tabs: {names: {
    localhistory: 'History'
  }}}
};
.uploadcare-dialog-tab-localhistory:before {
  background-image: url() !important;
  background-position: 0 0;
}
.uploadcare-dialog-tab_current.uploadcare-dialog-tab-localhistory:before {
  background-position: 0 50px;
}
.uploadcare-dialog-tabs-panel-localhistory .uploadcare-file-item {
  cursor: pointer;
}
<p>
  Images only with crop:
  <input type="hidden" role="uploadcare-uploader" data-tabs="file camera dropbox localhistory" data-images-only="" data-crop="">
</p>
<p>
  Multiple widget:
  <input type="hidden" role="uploadcare-uploader" data-tabs="file camera dropbox localhistory" data-multiple="">
</p>
  

Images only with crop:

Multiple widget:

Questions?

We're always happy to help with code or other questions you might have! Search our site for more information or send us an email!