1<!DOCTYPE html> 2<!-- 3 * Copyright (c) 2010 The Chromium Authors. All rights reserved. Use of this 4 * source code is governed by a BSD-style license that can be found in the 5 * LICENSE file. 6 * 7 * Author: Eric Bidelman <ericbidelman@chromium.org> 8--> 9<html> 10<head> 11<title>Your Google Documents List</title> 12<script type="text/javascript" src="js/jquery-1.4.1.min.js"></script> 13<style type="text/css"> 14body { 15 font: 12px 'Myriad Pro', 'Tw Cen MT', Arial, Verdana, sans-serif; 16 color: #666666; 17 overflow-x: hidden; 18} 19ul { 20 padding: 0; 21 list-style: none; 22} 23li { 24 clear: both; 25 padding: 2px 0; 26} 27li div img { 28 margin: 0 5px; 29 vertical-align: middle; 30} 31li div { 32 text-overflow: ellipsis; 33 white-space: nowrap; 34 overflow: hidden; 35 width: 250px; 36 float: left; 37 padding: 2px 0; 38} 39li span { 40 margin-left: 5px; 41} 42li:hover { 43 background-color: #fffccc; 44} 45a { 46 color: #4E7DC2; 47 text-decoration: none; 48} 49a:hover { 50 color: #880000; 51 text-decoration: underline; 52} 53#butter { 54 color: #fff; 55 background-color: #000033; 56 padding: 5px 20px; 57 border-radius: 15px; 58 width: auto; 59 text-align: center; 60 float: right; 61 display: none; 62} 63#butter.error { 64 background-color: red; 65} 66#new_doc_container { 67 display: none; 68} 69#new_doc_container input[type='text'],textarea { 70 width: 100%; 71} 72#output { 73 width: 375px; 74 clear: both; 75} 76[contenteditable]:hover { 77 outline: 1px dotted #666; 78} 79.star { 80 margin-top: 1px; 81 margin-right: 3px; 82 width: 16px; 83 height: 16px; 84 background: no-repeat url() !important; 85} 86.star.selected { 87 background: no-repeat url() !important; 88} 89</style> 90</head> 91<body> 92 93<div style="height:15px;"> 94 <div style="float:left;"> 95 <a href="javascript:void(0);" onclick="gdocs.refreshDocs();return false;">Refresh list</a>, 96 <a href="javascript:void(0);" onclick="$('#new_doc_container').toggle();return false;">New Document</a> 97 </div> 98 <div id="butter">Fetching your docs</div> 99</div> 100<div id="new_doc_container"> 101 Create a: <select id="doc_type"> 102 <option value="document">document</option> 103 <option value="presentation">presentation</option> 104 <option value="spreadsheet">spreadsheet</option> 105 </select> 106 <input type="text" id="doc_title" placeholder="Enter a title"><br> 107 <textarea id="doc_content" placeholder="Enter document content"></textarea> 108 Star it? <input type="checkbox" id="doc_starred"> 109 <button onclick="gdocs.createDoc();" style="float:right;">Create new doc</button> 110</div> 111<div id="output"></div> 112 113<script type="text/javascript"> 114// Protected namespaces. 115var util = {}; 116var gdocs = {}; 117 118var bgPage = chrome.extension.getBackgroundPage(); 119var pollIntervalMax = 1000 * 60 * 60; // 1 hour 120var requestFailureCount = 0; // used for exponential backoff 121var requestTimeout = 1000 * 2; // 5 seconds 122 123var DEFAULT_MIMETYPES = { 124 'atom': 'application/atom+xml', 125 'document': 'text/plain', 126 'spreadsheet': 'text/csv', 127 'presentation': 'text/plain', 128 'pdf': 'application/pdf' 129}; 130 131// Persistent click handler for star icons. 132$('#doc_type').change(function() { 133 if ($(this).val() === 'presentation') { 134 $('#doc_content').attr('disabled', 'true') 135 .attr('placeholder', 'N/A for presentations'); 136 } else { 137 $('#doc_content').removeAttr('disabled') 138 .attr('placeholder', 'Enter document content'); 139 } 140}); 141 142 143// Persistent click handler for changing the title of a document. 144$('[contenteditable="true"]').live('blur', function(index) { 145 var index = $(this).parent().parent().attr('data-index'); 146 147 // Only make the XHR if the user chose a new title. 148 if ($(this).text() != bgPage.docs[index].title) { 149 bgPage.docs[index].title = $(this).text(); 150 gdocs.updateDoc(bgPage.docs[index]); 151 } 152}); 153 154// Persistent click handler for star icons. 155$('.star').live('click', function() { 156 $(this).toggleClass('selected'); 157 158 var index = $(this).parent().attr('data-index'); 159 bgPage.docs[index].starred = $(this).hasClass('selected'); 160 gdocs.updateDoc(bgPage.docs[index]); 161}); 162 163/** 164 * Class to compartmentalize properties of a Google document. 165 * @param {Object} entry A JSON representation of a DocList atom entry. 166 * @constructor 167 */ 168gdocs.GoogleDoc = function(entry) { 169 this.entry = entry; 170 this.title = entry.title.$t; 171 this.resourceId = entry.gd$resourceId.$t; 172 this.type = gdocs.getCategory( 173 entry.category, 'http://schemas.google.com/g/2005#kind'); 174 this.starred = gdocs.getCategory( 175 entry.category, 'http://schemas.google.com/g/2005/labels', 176 'http://schemas.google.com/g/2005/labels#starred') ? true : false; 177 this.link = { 178 'alternate': gdocs.getLink(entry.link, 'alternate').href 179 }; 180 this.contentSrc = entry.content.src; 181}; 182 183/** 184 * Sets up a future poll for the user's document list. 185 */ 186util.scheduleRequest = function() { 187 var exponent = Math.pow(2, requestFailureCount); 188 var delay = Math.min(bgPage.pollIntervalMin * exponent, 189 pollIntervalMax); 190 delay = Math.round(delay); 191 192 if (bgPage.oauth.hasToken()) { 193 var req = bgPage.window.setTimeout(function() { 194 gdocs.getDocumentList(); 195 util.scheduleRequest(); 196 }, delay); 197 bgPage.requests.push(req); 198 } 199}; 200 201/** 202 * Urlencodes a JSON object of key/value query parameters. 203 * @param {Object} parameters Key value pairs representing URL parameters. 204 * @return {string} query parameters concatenated together. 205 */ 206util.stringify = function(parameters) { 207 var params = []; 208 for(var p in parameters) { 209 params.push(encodeURIComponent(p) + '=' + 210 encodeURIComponent(parameters[p])); 211 } 212 return params.join('&'); 213}; 214 215/** 216 * Creates a JSON object of key/value pairs 217 * @param {string} paramStr A string of Url query parmeters. 218 * For example: max-results=5&startindex=2&showfolders=true 219 * @return {Object} The query parameters as key/value pairs. 220 */ 221util.unstringify = function(paramStr) { 222 var parts = paramStr.split('&'); 223 224 var params = {}; 225 for (var i = 0, pair; pair = parts[i]; ++i) { 226 var param = pair.split('='); 227 params[decodeURIComponent(param[0])] = decodeURIComponent(param[1]); 228 } 229 return params; 230}; 231 232/** 233 * Utility for displaying a message to the user. 234 * @param {string} msg The message. 235 */ 236util.displayMsg = function(msg) { 237 $('#butter').removeClass('error').text(msg).show(); 238}; 239 240/** 241 * Utility for removing any messages currently showing to the user. 242 */ 243util.hideMsg = function() { 244 $('#butter').fadeOut(1500); 245}; 246 247/** 248 * Utility for displaying an error to the user. 249 * @param {string} msg The message. 250 */ 251util.displayError = function(msg) { 252 util.displayMsg(msg); 253 $('#butter').addClass('error'); 254}; 255 256/** 257 * Returns the correct atom link corresponding to the 'rel' value passed in. 258 * @param {Array<Object>} links A list of atom link objects. 259 * @param {string} rel The rel value of the link to return. For example: 'next'. 260 * @return {string|null} The appropriate link for the 'rel' passed in, or null 261 * if one is not found. 262 */ 263gdocs.getLink = function(links, rel) { 264 for (var i = 0, link; link = links[i]; ++i) { 265 if (link.rel === rel) { 266 return link; 267 } 268 } 269 return null; 270}; 271 272/** 273 * Returns the correct atom category corresponding to the scheme/term passed in. 274 * @param {Array<Object>} categories A list of atom category objects. 275 * @param {string} scheme The category's scheme to look up. 276 * @param {opt_term?} An optional term value for the category to look up. 277 * @return {string|null} The appropriate category, or null if one is not found. 278 */ 279gdocs.getCategory = function(categories, scheme, opt_term) { 280 for (var i = 0, cat; cat = categories[i]; ++i) { 281 if (opt_term) { 282 if (cat.scheme === scheme && opt_term === cat.term) { 283 return cat; 284 } 285 } else if (cat.scheme === scheme) { 286 return cat; 287 } 288 } 289 return null; 290}; 291 292/** 293 * A generic error handler for failed XHR requests. 294 * @param {XMLHttpRequest} xhr The xhr request that failed. 295 * @param {string} textStatus The server's returned status. 296 */ 297gdocs.handleError = function(xhr, textStatus) { 298 util.displayError('Failed to fetch docs. Please try again.'); 299 ++requestFailureCount; 300}; 301 302/** 303 * A helper for constructing the raw Atom xml send in the body of an HTTP post. 304 * @param {XMLHttpRequest} xhr The xhr request that failed. 305 * @param {string} docTitle A title for the document. 306 * @param {string} docType The type of document to create. 307 * (eg. 'document', 'spreadsheet', etc.) 308 * @param {boolean?} opt_starred Whether the document should be starred. 309 * @return {string} The Atom xml as a string. 310 */ 311gdocs.constructAtomXml_ = function(docTitle, docType, opt_starred) { 312 var starred = opt_starred || null; 313 314 var starCat = ['<category scheme="http://schemas.google.com/g/2005/labels" ', 315 'term="http://schemas.google.com/g/2005/labels#starred" ', 316 'label="starred"/>'].join(''); 317 318 var atom = ["<?xml version='1.0' encoding='UTF-8'?>", 319 '<entry xmlns="http://www.w3.org/2005/Atom">', 320 '<category scheme="http://schemas.google.com/g/2005#kind"', 321 ' term="http://schemas.google.com/docs/2007#', docType, '"/>', 322 starred ? starCat : '', 323 '<title>', docTitle, '</title>', 324 '</entry>'].join(''); 325 return atom; 326}; 327 328/** 329 * A helper for constructing the body of a mime-mutlipart HTTP request. 330 * @param {string} title A title for the new document. 331 * @param {string} docType The type of document to create. 332 * (eg. 'document', 'spreadsheet', etc.) 333 * @param {string} body The body of the HTTP request. 334 * @param {string} contentType The Content-Type of the (non-Atom) portion of the 335 * http body. 336 * @param {boolean?} opt_starred Whether the document should be starred. 337 * @return {string} The Atom xml as a string. 338 */ 339gdocs.constructContentBody_ = function(title, docType, body, contentType, 340 opt_starred) { 341 var body = ['--END_OF_PART\r\n', 342 'Content-Type: application/atom+xml;\r\n\r\n', 343 gdocs.constructAtomXml_(title, docType, opt_starred), '\r\n', 344 '--END_OF_PART\r\n', 345 'Content-Type: ', contentType, '\r\n\r\n', 346 body, '\r\n', 347 '--END_OF_PART--\r\n'].join(''); 348 return body; 349}; 350 351/** 352 * Creates a new document in Google Docs. 353 */ 354gdocs.createDoc = function() { 355 var title = $.trim($('#doc_title').val()); 356 if (!title) { 357 alert('Please provide a title'); 358 return; 359 } 360 var content = $('#doc_content').val(); 361 var starred = $('#doc_starred').is(':checked'); 362 var docType = $('#doc_type').val(); 363 364 util.displayMsg('Creating doc...'); 365 366 var handleSuccess = function(resp, xhr) { 367 bgPage.docs.splice(0, 0, new gdocs.GoogleDoc(JSON.parse(resp).entry)); 368 369 gdocs.renderDocList(); 370 bgPage.setIcon({'text': bgPage.docs.length.toString()}); 371 372 $('#new_doc_container').hide(); 373 $('#doc_title').val(''); 374 $('#doc_content').val(''); 375 util.displayMsg('Document created!'); 376 util.hideMsg(); 377 378 requestFailureCount = 0; 379 }; 380 381 var params = { 382 'method': 'POST', 383 'headers': { 384 'GData-Version': '3.0', 385 'Content-Type': 'multipart/related; boundary=END_OF_PART', 386 }, 387 'parameters': {'alt': 'json'}, 388 'body': gdocs.constructContentBody_(title, docType, content, 389 DEFAULT_MIMETYPES[docType], starred) 390 }; 391 392 // Presentation can only be created from binary content. Instead, create a 393 // blank presentation. 394 if (docType === 'presentation') { 395 params['headers']['Content-Type'] = DEFAULT_MIMETYPES['atom']; 396 params['body'] = gdocs.constructAtomXml_(title, docType, starred); 397 } 398 399 bgPage.oauth.sendSignedRequest(bgPage.DOCLIST_FEED, handleSuccess, params); 400}; 401 402/** 403 * Updates a document's metadata (title, starred, etc.). 404 * @param {gdocs.GoogleDoc} googleDocObj An object containing the document to 405 * update. 406 */ 407gdocs.updateDoc = function(googleDocObj) { 408 var handleSuccess = function(resp) { 409 util.displayMsg('Updated!'); 410 util.hideMsg(); 411 requestFailureCount = 0; 412 }; 413 414 var params = { 415 'method': 'PUT', 416 'headers': { 417 'GData-Version': '3.0', 418 'Content-Type': 'application/atom+xml', 419 'If-Match': '*' 420 }, 421 'body': gdocs.constructAtomXml_(googleDocObj.title, googleDocObj.type, 422 googleDocObj.starred) 423 }; 424 425 var url = bgPage.DOCLIST_FEED + googleDocObj.resourceId; 426 bgPage.oauth.sendSignedRequest(url, handleSuccess, params); 427}; 428 429/** 430 * Deletes a document from the user's document list. 431 * @param {integer} index An index intro the background page's docs array. 432 */ 433gdocs.deleteDoc = function(index) { 434 var handleSuccess = function(resp, xhr) { 435 util.displayMsg('Document trashed!'); 436 util.hideMsg(); 437 requestFailureCount = 0; 438 bgPage.docs.splice(index, 1); 439 bgPage.setIcon({'text': bgPage.docs.length.toString()}); 440 } 441 442 var params = { 443 'method': 'DELETE', 444 'headers': { 445 'GData-Version': '3.0', 446 'If-Match': '*' 447 } 448 }; 449 450 $('#output li').eq(index).fadeOut('slow'); 451 452 bgPage.oauth.sendSignedRequest( 453 bgPage.DOCLIST_FEED + bgPage.docs[index].resourceId, 454 handleSuccess, params); 455}; 456 457/** 458 * Callback for processing the JSON feed returned by the DocList API. 459 * @param {string} response The server's response. 460 * @param {XMLHttpRequest} xhr The xhr request that was made. 461 */ 462gdocs.processDocListResults = function(response, xhr) { 463 if (xhr.status != 200) { 464 gdocs.handleError(xhr, response); 465 return; 466 } else { 467 requestFailureCount = 0; 468 } 469 470 var data = JSON.parse(response); 471 472 for (var i = 0, entry; entry = data.feed.entry[i]; ++i) { 473 bgPage.docs.push(new gdocs.GoogleDoc(entry)); 474 } 475 476 var nextLink = gdocs.getLink(data.feed.link, 'next'); 477 if (nextLink) { 478 gdocs.getDocumentList(nextLink.href); // Fetch next page of results. 479 } else { 480 gdocs.renderDocList(); 481 } 482}; 483 484/** 485 * Presents the in-memory documents that were fetched from the server as HTML. 486 */ 487gdocs.renderDocList = function() { 488 util.hideMsg(); 489 490 // Construct the iframe's HTML. 491 var html = []; 492 for (var i = 0, doc; doc = bgPage.docs[i]; ++i) { 493 // If we have an arbitrary file, use generic file icon. 494 var type = doc.type.label; 495 if (doc.type.term == 'http://schemas.google.com/docs/2007#file') { 496 type = 'file'; 497 } 498 499 var starred = doc.starred ? ' selected' : ''; 500 html.push( 501 '<li data-index="', i , '"><div class="star', starred, '"></div>', 502 '<div><img src="img/icons/', type, '.gif">', 503 '<span contenteditable="true" class="doc_title"></span></div>', 504 '<span>[<a href="', doc.link['alternate'], 505 '" target="_new">view</a> | <a href="javascript:void(0);" ', 506 'onclick="gdocs.deleteDoc(',i, 507 ');return false;">delete</a>]','</span></li>'); 508 } 509 $('#output').html('<ul>' + html.join('') + '</ul>'); 510 511 // Set each span's innerText to be the doc title. We're filling this after 512 // the html has been rendered to the page prevent XSS attacks when using 513 // innerHTML. 514 $('#output li span.doc_title').each(function(i, ul) { 515 $(ul).text(bgPage.docs[i].title); 516 }); 517 518 bgPage.setIcon({'text': bgPage.docs.length.toString()}); 519}; 520 521/** 522 * Fetches the user's document list. 523 * @param {string?} opt_url A url to query the doclist API with. If omitted, 524 * the main doclist feed uri is used. 525 */ 526gdocs.getDocumentList = function(opt_url) { 527 var url = opt_url || null; 528 529 var params = { 530 'headers': { 531 'GData-Version': '3.0' 532 } 533 }; 534 535 if (!url) { 536 util.displayMsg('Fetching your docs'); 537 bgPage.setIcon({'text': '...'}); 538 539 bgPage.docs = []; // Clear document list. We're doing a refresh. 540 541 url = bgPage.DOCLIST_FEED; 542 params['parameters'] = { 543 'alt': 'json', 544 'showfolders': 'true' 545 }; 546 } else { 547 util.displayMsg($('#butter').text() + '.'); 548 549 var parts = url.split('?'); 550 if (parts.length > 1) { 551 url = parts[0]; // Extract base URI. Params are passed in separately. 552 params['parameters'] = util.unstringify(parts[1]); 553 } 554 } 555 556 bgPage.oauth.sendSignedRequest(url, gdocs.processDocListResults, params); 557}; 558 559/** 560 * Refreshes the user's document list. 561 */ 562gdocs.refreshDocs = function() { 563 bgPage.clearPendingRequests(); 564 gdocs.getDocumentList(); 565 util.scheduleRequest(); 566}; 567 568 569bgPage.oauth.authorize(function() { 570 if (!bgPage.docs.length) { 571 gdocs.getDocumentList(); 572 } else { 573 gdocs.renderDocList(); 574 } 575 util.scheduleRequest(); 576}); 577</script> 578</body> 579</html> 580