2010-12-02 02:41:51 +00:00
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<script type="text/javascript" src="base64.js"></script>
|
|
|
|
<script type="text/javascript" src="jquery-1.4.2.min.js"></script>
|
|
|
|
<script type="text/javascript" src="chrome_ex_oauthsimple.js"></script>
|
|
|
|
<script type="text/javascript" src="chrome_ex_oauth.js"></script>
|
2010-12-02 04:35:13 +00:00
|
|
|
<script type="text/javascript" src="Crypto.js"></script>
|
|
|
|
<script type="text/javascript" src="SHA1.js"></script>
|
2010-12-02 02:41:51 +00:00
|
|
|
|
|
|
|
<script type="text/javascript" charset="utf-8">
|
|
|
|
|
|
|
|
var OAUTH = ChromeExOAuth.initBackgroundPage({
|
|
|
|
'request_url' : 'https://www.google.com/accounts/OAuthGetRequestToken',
|
|
|
|
'authorize_url' : 'https://www.google.com/accounts/OAuthAuthorizeToken',
|
|
|
|
'access_url' : 'https://www.google.com/accounts/OAuthGetAccessToken',
|
|
|
|
'consumer_key' : 'anonymous',
|
|
|
|
'consumer_secret' : 'anonymous',
|
|
|
|
'scope' : 'http://picasaweb.google.com/data/',
|
|
|
|
'app_name' : 'Clip It Good: Chrome Extension'
|
|
|
|
});
|
|
|
|
|
|
|
|
// Constants for various album types.
|
|
|
|
var PICASA = 'picasa';
|
2010-12-02 04:35:13 +00:00
|
|
|
var CAMLISTORE = 'camlistore';
|
2010-12-02 02:41:51 +00:00
|
|
|
var ALBUM_TYPE_STRING = {
|
2010-12-02 04:35:13 +00:00
|
|
|
picasa: 'Picasa Web Albums',
|
|
|
|
camlistore: 'Camlistore'
|
2010-12-02 02:41:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Preferences
|
|
|
|
var ALBUM_CONFIG = {}; // 'type' -> ('id' -> 'name')
|
2010-12-02 06:15:27 +00:00
|
|
|
var ALBUM_OPTIONS = {}; // 'type' -> ('id' -> optionsDict)
|
2010-12-02 02:41:51 +00:00
|
|
|
|
|
|
|
function loadAlbumConfig() {
|
|
|
|
var newAlbumConfig = localStorage.getItem('config:albums');
|
|
|
|
if (newAlbumConfig) {
|
|
|
|
ALBUM_CONFIG = $.parseJSON(newAlbumConfig);
|
|
|
|
}
|
2010-12-02 06:15:27 +00:00
|
|
|
|
|
|
|
var newAlbumOptions = localStorage.getItem('config:albumOptions');
|
|
|
|
if (newAlbumOptions) {
|
|
|
|
ALBUM_OPTIONS = $.parseJSON(newAlbumOptions);
|
|
|
|
}
|
2010-12-02 02:41:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function saveAlbumConfig() {
|
|
|
|
localStorage.setItem('config:albums', JSON.stringify(ALBUM_CONFIG));
|
2010-12-02 06:15:27 +00:00
|
|
|
localStorage.setItem('config:albumOptions', JSON.stringify(ALBUM_OPTIONS));
|
2010-12-02 02:41:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sort albums by name.
|
|
|
|
function getSortedAlbums(albumIdNameDict) {
|
|
|
|
var albumIdNameArray = [];
|
|
|
|
$.each(albumIdNameDict, function(id, name) {
|
|
|
|
albumIdNameArray.push({'id': id, 'name': name});
|
|
|
|
});
|
|
|
|
albumIdNameArray.sort(function(a, b) {
|
|
|
|
if (a['name'] < b['name']) {
|
|
|
|
return -1;
|
|
|
|
} else if (a['name'] > b['name']) {
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return albumIdNameArray;
|
|
|
|
}
|
|
|
|
|
|
|
|
function setupMenus() {
|
|
|
|
loadAlbumConfig();
|
|
|
|
|
|
|
|
chrome.contextMenus.removeAll(function() {
|
|
|
|
$.each(ALBUM_CONFIG, function(albumType, albumIdNameDict) {
|
|
|
|
chrome.contextMenus.create({
|
|
|
|
title: ALBUM_TYPE_STRING[albumType],
|
|
|
|
contexts: ['image']
|
|
|
|
});
|
|
|
|
chrome.contextMenus.create({
|
|
|
|
type: 'separator',
|
|
|
|
contexts: ['image']
|
|
|
|
});
|
|
|
|
|
|
|
|
$.each(getSortedAlbums(albumIdNameDict), function(index, albumDict) {
|
|
|
|
chrome.contextMenus.create({
|
|
|
|
title: albumDict.name,
|
|
|
|
contexts: ['image'],
|
|
|
|
onclick: function(data, tab) {
|
|
|
|
return handleMenuClick(
|
2010-12-02 04:35:13 +00:00
|
|
|
albumType, albumDict.name, albumDict.id, data, tab);
|
2010-12-02 02:41:51 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2010-12-02 03:51:14 +00:00
|
|
|
// Upload actions
|
|
|
|
var ALBUM_TYPE_UPLOAD_FUNC = {
|
|
|
|
|
2010-12-02 05:29:09 +00:00
|
|
|
picasa: function(albumId, albumName, dataSrcUrl, dataBuffer, uploadDone) {
|
2010-12-02 04:35:13 +00:00
|
|
|
var builder = new BlobBuilder();
|
|
|
|
builder.append(dataBuffer);
|
|
|
|
|
2010-12-02 05:29:09 +00:00
|
|
|
function complete(resp, xhr) {
|
|
|
|
if (!(xhr.status >= 200 && xhr.status <= 299)) {
|
|
|
|
alert('Error: Response status = ' + xhr.status +
|
|
|
|
', response body = "' + xhr.responseText + '"');
|
|
|
|
} else {
|
|
|
|
uploadDone();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-02 03:51:14 +00:00
|
|
|
OAUTH.authorize(function() {
|
|
|
|
OAUTH.sendSignedRequest(
|
|
|
|
'http://picasaweb.google.com/data/feed/api/' +
|
|
|
|
'user/default/albumid/' + albumId,
|
2010-12-02 05:29:09 +00:00
|
|
|
complete,
|
2010-12-02 03:51:14 +00:00
|
|
|
{
|
|
|
|
method: 'POST',
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'image/png',
|
|
|
|
'Slug': dataSrcUrl
|
|
|
|
},
|
|
|
|
parameters: {
|
|
|
|
alt: 'json'
|
|
|
|
},
|
2010-12-02 04:35:13 +00:00
|
|
|
body: builder.getBlob('image/png')
|
2010-12-02 03:51:14 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
});
|
2010-12-02 04:35:13 +00:00
|
|
|
},
|
|
|
|
|
2010-12-02 05:29:09 +00:00
|
|
|
camlistore: function(albumId, albumName, dataSrcUrl,
|
|
|
|
dataBuffer, uploadDone) {
|
2010-12-02 05:09:35 +00:00
|
|
|
var hash = Crypto.SHA1(new Uint8Array(dataBuffer, 0));
|
2010-12-02 05:29:09 +00:00
|
|
|
var blobRef = 'sha1-' + hash;
|
2010-12-02 06:15:27 +00:00
|
|
|
var options = ALBUM_OPTIONS[CAMLISTORE][albumId];
|
2010-12-02 05:09:35 +00:00
|
|
|
|
|
|
|
function doUpload(uploadUrl) {
|
|
|
|
// XXX Use real random boundary.
|
|
|
|
var boundary = 'randomboundaryXYZ';
|
|
|
|
var contentType = 'multipart/form-data; boundary=' + boundary;
|
|
|
|
|
|
|
|
var header =
|
|
|
|
'--' + boundary + '\r\n' +
|
|
|
|
'Content-Type: application/octet-stream\r\n' +
|
2010-12-02 05:29:09 +00:00
|
|
|
'Content-Disposition: form-data; name="' + blobRef +
|
2010-12-02 05:09:35 +00:00
|
|
|
'"; filename="1"\r\n\r\n'
|
|
|
|
var footer = '\r\n--' + boundary + '--\r\n';
|
|
|
|
|
|
|
|
var builder = new BlobBuilder();
|
|
|
|
builder.append(header);
|
|
|
|
builder.append(dataBuffer);
|
|
|
|
builder.append(footer);
|
|
|
|
var payload = builder.getBlob(contentType);
|
|
|
|
|
|
|
|
var uploadXhr = new XMLHttpRequest();
|
2010-12-02 06:15:27 +00:00
|
|
|
uploadXhr.open('POST', uploadUrl, true,
|
|
|
|
options.username, options.password);
|
2010-12-02 05:09:35 +00:00
|
|
|
uploadXhr.onreadystatechange = function() {
|
|
|
|
if (uploadXhr.readyState == XMLHttpRequest.DONE &&
|
|
|
|
uploadXhr.status == 200) {
|
2010-12-02 05:29:09 +00:00
|
|
|
// XXX Check for bad response format (not JSON).
|
2010-12-02 05:09:35 +00:00
|
|
|
var responseJson = $.parseJSON(uploadXhr.responseText)
|
2010-12-02 05:29:09 +00:00
|
|
|
|
|
|
|
if (responseJson.received &&
|
|
|
|
responseJson.received.length == 1 &&
|
|
|
|
responseJson.received[0].blobRef == blobRef) {
|
|
|
|
console.log('Successful upload: ' + blobRef);
|
|
|
|
uploadDone();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
alert('Camlistore upload response did not verify blob "' +
|
|
|
|
blobRef + '": ' + uploadXhr.responseText);
|
2010-12-02 05:09:35 +00:00
|
|
|
}
|
2010-12-02 05:29:09 +00:00
|
|
|
// XXX: Handle request errors
|
2010-12-02 05:09:35 +00:00
|
|
|
}
|
|
|
|
uploadXhr.setRequestHeader('Content-Type', contentType);
|
|
|
|
uploadXhr.send(payload);
|
|
|
|
}
|
|
|
|
|
2011-02-08 16:24:16 +00:00
|
|
|
var statXhr = new XMLHttpRequest();
|
|
|
|
statXhr.open('POST', albumId + '/camli/stat', true,
|
2010-12-02 06:15:27 +00:00
|
|
|
options.username, options.password);
|
2011-02-08 16:24:16 +00:00
|
|
|
statXhr.onreadystatechange = function() {
|
|
|
|
if (statXhr.readyState == XMLHttpRequest.DONE &&
|
|
|
|
statXhr.status == 200) {
|
2010-12-02 05:29:09 +00:00
|
|
|
// XXX Check for bad response format (not JSON).
|
2011-02-08 16:24:16 +00:00
|
|
|
var responseJson = $.parseJSON(statXhr.responseText);
|
2010-12-02 05:29:09 +00:00
|
|
|
|
2011-02-08 16:24:16 +00:00
|
|
|
if (responseJson.stat &&
|
|
|
|
responseJson.stat.length == 1 &&
|
|
|
|
responseJson.stat[0].blobRef == blobRef) {
|
2010-12-02 05:29:09 +00:00
|
|
|
console.log('Blob already present: ' + blobRef);
|
|
|
|
uploadDone();
|
|
|
|
return;
|
|
|
|
}
|
2010-12-02 05:09:35 +00:00
|
|
|
|
|
|
|
var uploadUrl = responseJson.uploadUrl;
|
|
|
|
if (!uploadUrl) {
|
2011-02-08 16:24:16 +00:00
|
|
|
alert('Camlistore stat response missing "uploadUrl": ' +
|
|
|
|
statXhr.responseText);
|
2010-12-02 05:29:09 +00:00
|
|
|
return;
|
2010-12-02 05:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
doUpload(uploadUrl);
|
|
|
|
}
|
2010-12-02 05:29:09 +00:00
|
|
|
// XXX: Handle request errors
|
2010-12-02 05:09:35 +00:00
|
|
|
}
|
2011-02-08 16:24:16 +00:00
|
|
|
statXhr.setRequestHeader(
|
2010-12-02 05:09:35 +00:00
|
|
|
'Content-Type', 'application/x-www-form-urlencoded');
|
2011-02-08 16:24:16 +00:00
|
|
|
statXhr.send('camliversion=1&blob1=' + blobRef);
|
2010-12-02 03:51:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
function handleMenuClick(albumType, albumName, albumId, data, tab) {
|
2010-12-02 02:41:51 +00:00
|
|
|
chrome.pageAction.setTitle({
|
|
|
|
tabId: tab.id,
|
|
|
|
title: 'Clip It Good: Uploading (' + data.srcUrl.substr(0, 100) + ')'
|
|
|
|
});
|
|
|
|
chrome.pageAction.show(tab.id);
|
|
|
|
|
|
|
|
var img = document.createElement('img');
|
|
|
|
img.onload = function() {
|
|
|
|
var canvas = document.createElement('canvas');
|
|
|
|
canvas.width = img.width;
|
|
|
|
canvas.height = img.height;
|
|
|
|
canvas.getContext('2d').drawImage(img, 0, 0);
|
|
|
|
|
|
|
|
var dataUrl = canvas.toDataURL();
|
|
|
|
var dataUrlAdjusted = dataUrl.replace('data:image/png;base64,', '');
|
|
|
|
|
2010-12-02 04:35:13 +00:00
|
|
|
var arrayBuffer = Base64.decode(dataUrlAdjusted).buffer;
|
2010-12-02 02:41:51 +00:00
|
|
|
|
2010-12-02 05:29:09 +00:00
|
|
|
function uploadDone() {
|
2010-12-02 02:41:51 +00:00
|
|
|
chrome.pageAction.hide(tab.id);
|
2010-12-02 05:29:09 +00:00
|
|
|
}
|
2010-12-02 02:41:51 +00:00
|
|
|
|
2010-12-02 03:51:14 +00:00
|
|
|
var uploadFunc = ALBUM_TYPE_UPLOAD_FUNC[albumType];
|
2010-12-02 05:29:09 +00:00
|
|
|
uploadFunc(albumId, albumName, data.srcUrl, arrayBuffer, uploadDone);
|
2010-12-02 02:41:51 +00:00
|
|
|
} // end onload
|
|
|
|
|
|
|
|
img.src = data.srcUrl;
|
|
|
|
}
|
|
|
|
|
|
|
|
$(document).ready( function() {
|
|
|
|
setupMenus();
|
|
|
|
});
|
|
|
|
</script>
|
|
|
|
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
</body>
|
|
|
|
</html>
|