├── .gitignore ├── README.md ├── img ├── install.gif └── screenshot.png ├── npm.coffee ├── package.json ├── sheet.coffee └── sheetExample.framer ├── .gitignore ├── app.coffee ├── framer ├── .bookmark ├── coffee-script.js ├── config.json ├── framer.generated.js ├── framer.init.js ├── framer.js ├── framer.js.map ├── framer.modules.js ├── images │ ├── cursor-active.png │ ├── cursor-active@2x.png │ ├── cursor.png │ ├── cursor@2x.png │ ├── icon-120.png │ ├── icon-152.png │ ├── icon-180.png │ ├── icon-192.png │ └── icon-76.png ├── style.css └── version ├── images └── .gitkeep ├── index.html └── modules ├── npm.coffee └── sheet.coffee /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | */node_modules 3 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Framer-Sheet 2 | 3 | Import information from Google Sheets into Framer. 4 | 5 | ![Sheets](img/screenshot.png) 6 | 7 | ## Add it to your Framer Studio project 8 | 9 | * Create and save a new Framer project (if you don't have one started already) 10 | * [Download](https://github.com/andrewliebchen/framer-sheet/archive/master.zip) or clone this repository 11 | * Copy `sheet.coffee` and `npm.coffee` to your `/modules` directory 12 | * Open your Terminal, drag your framer project into the Terminal window and press enter. The path in the Terminal will update to your Framer project 13 | * Type `npm install tabletop` to get the dependency from NPM 14 | 15 | ![Install](img/install.gif) 16 | 17 | * Import the module into your project by adding `{ Sheet } = require 'sheet'` to the top of your project's code 18 | 19 | ## How to use it 20 | 21 | Getting started is pretty easy. Follow the instructions above to install the module. You'll also need a [Google Sheets](http://drive.google.com/) document to import. Here's how to set up your sheet: 22 | 23 | Go up to the `File` menu and pick `Publish to the web`. Fiddle with the options, then click `Start` publishing. A URL will appear, something like `https://docs.google.com/spreadsheets/d/1sbyMINQHPsJctjAtMW0lCfLrcpMqoGMOJj6AN-sNQrc/pubhtml` (of course, it might look a little different). 24 | 25 | In the example URL above, you're interested in the key, which is between the `/d/` and `/pubhtml`, so `1sbyMINQHPsJctjAtMW0lCfLrcpMqoGMOJj6AN-sNQrc`. Copy this value. 26 | 27 | In your Framer project, instantiate a new instance in your project: 28 | 29 | ```coffeescript 30 | { Sheet } = require 'sheet' 31 | 32 | sheet = new Sheet 33 | key: '1sbyMINQHPsJctjAtMW0lCfLrcpMqoGMOJj6AN-sNQrc' 34 | ``` 35 | 36 | Now, to actually GET the data from your sheet, call the `get` method: 37 | 38 | ```coffeescript 39 | sheet.get((data, sheet) -> 40 | print data 41 | ) 42 | ``` 43 | 44 | ...which should print an array of objects (a collection) like... 45 | 46 | ``` 47 | » [{Name:"Carrot", Category:"Vegetable", Healthiness:"Adequate"}, 48 | {Name:"Pork Shoulder", Category:"Meat", Healthiness:"Questionable"}, 49 | {Name:"Bubblegum", Category:"Candy", Healthiness:"Super High"}] 50 | ``` 51 | 52 | Pretty cool, huh? You'll see in the example project, I've used Lodash's `map` to split the collection into rows which and cells for a table. 53 | 54 | ```coffeescript 55 | sheet.get((data, sheet) -> 56 | _.map data, (row, i) -> 57 | @row = new Layer 58 | name: 'row' 59 | parent: table 60 | # ... 61 | 62 | @name = new Layer 63 | html: row.Name 64 | name: "cell:#{row.Name}" 65 | parent: @row 66 | # ... 67 | ``` 68 | 69 | This is only the beginning...what else can you think to do? 70 | 71 | ## More information 72 | 73 | This module makes use of [Tabletop.js](https://github.com/jsoma/tabletop) for the heavy lifting. For more information, check out their documentation, including a section on how to format your Google Sheet. 74 | -------------------------------------------------------------------------------- /img/install.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/img/install.gif -------------------------------------------------------------------------------- /img/screenshot.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/img/screenshot.png -------------------------------------------------------------------------------- /npm.coffee: -------------------------------------------------------------------------------- 1 | exports.Tabletop = require 'tabletop' 2 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "framer-sheet", 3 | "version": "1.0.0", 4 | "description": "Import information from Google Sheets into Framer.", 5 | "main": "app.coffee", 6 | "scripts": { 7 | "test": "echo \"Error: no test specified\" && exit 1" 8 | }, 9 | "repository": { 10 | "type": "git", 11 | "url": "git+https://github.com/andrewliebchen/framer-sheet.git" 12 | }, 13 | "keywords": [ 14 | "Framer", 15 | "Google", 16 | "Sheets", 17 | "Framer", 18 | "module", 19 | "FramerJS", 20 | "Framer", 21 | "Studio" 22 | ], 23 | "author": "Andrew Liebchen ", 24 | "license": "MIT", 25 | "bugs": { 26 | "url": "https://github.com/andrewliebchen/framer-sheet/issues" 27 | }, 28 | "homepage": "https://github.com/andrewliebchen/framer-sheet#readme", 29 | "devDependencies": { 30 | "tabletop": "^1.5.1" 31 | } 32 | } 33 | -------------------------------------------------------------------------------- /sheet.coffee: -------------------------------------------------------------------------------- 1 | { Tabletop } = require 'npm' 2 | 3 | 4 | class exports.Sheet 5 | constructor: (options) -> 6 | @_key = options.key 7 | 8 | @get = (callback) => 9 | Tabletop.init { 10 | key: @_key 11 | simpleSheet: true 12 | callback: (data, sheet) => 13 | return callback(data, sheet) 14 | } 15 | -------------------------------------------------------------------------------- /sheetExample.framer/.gitignore: -------------------------------------------------------------------------------- 1 | # Framer Git Ignore 2 | 3 | # General OSX 4 | 5 | .DS_Store 6 | .AppleDouble 7 | .LSOverride 8 | 9 | # Icon must end with two \r 10 | Icon 11 | 12 | # Thumbnails 13 | ._* 14 | 15 | # Files that might appear in the root of a volume 16 | .DocumentRevisions-V100 17 | .fseventsd 18 | .Spotlight-V100 19 | .TemporaryItems 20 | .Trashes 21 | .VolumeIcon.icns 22 | 23 | # Directories potentially created on remote AFP share 24 | .AppleDB 25 | .AppleDesktop 26 | Network Trash Folder 27 | Temporary Items 28 | .apdisk 29 | 30 | # Framer Specific 31 | .*.html 32 | .app.js 33 | framer/*.old* 34 | framer/.*.hash 35 | framer/backup.coffee 36 | framer/backups/* 37 | framer/manifest.txt 38 | framer/metadata.json 39 | framer/preview.png 40 | framer/social-880x460.png 41 | framer/social-1200x630.png 42 | -------------------------------------------------------------------------------- /sheetExample.framer/app.coffee: -------------------------------------------------------------------------------- 1 | { Sheet } = require 'sheet' 2 | 3 | Framer.Defaults.Layer = 4 | height: 40 5 | backgroundColor: null 6 | color: 'black' 7 | style: 8 | 'font-size': '16px' 9 | 'line-height': '40px' 10 | 'padding': '0 0.5em' 11 | 12 | 13 | sheet = new Sheet 14 | key: '0AmYzu_s7QHsmdDNZUzRlYldnWTZCLXdrMXlYQzVxSFE' 15 | 16 | table = new Layer 17 | 18 | sheet.get((data, sheet) -> 19 | _.map data, (row, i) -> 20 | @row = new Layer 21 | name: 'row' 22 | parent: table 23 | backgroundColor: null 24 | width: 600 25 | backgroundColor: if i % 2 then '#f0f0f0' 26 | x: 10 27 | y: 40 * i 28 | 29 | 30 | @name = new Layer 31 | html: row.Name 32 | name: "cell:#{row.Name}" 33 | parent: @row 34 | 35 | @category = new Layer 36 | html: row.Category 37 | name: "cell:#{row.Category}" 38 | parent: @row 39 | x: 200 40 | 41 | @healthiness = new Layer 42 | html: row.Healthiness 43 | name: "cell:#{row.Healthiness}" 44 | parent: @row 45 | height: 40 46 | x: 400 47 | ) 48 | -------------------------------------------------------------------------------- /sheetExample.framer/framer/.bookmark: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/.bookmark -------------------------------------------------------------------------------- /sheetExample.framer/framer/config.json: -------------------------------------------------------------------------------- 1 | { 2 | "propertyPanelToggleStates" : { 3 | 4 | }, 5 | "deviceOrientation" : 0, 6 | "sharedPrototype" : 0, 7 | "contentScale" : 1, 8 | "deviceType" : "fullscreen", 9 | "selectedHand" : "", 10 | "updateDelay" : 0.3, 11 | "deviceScale" : 1, 12 | "foldedCodeRanges" : [ 13 | 14 | ], 15 | "orientation" : 0, 16 | "projectId" : "C33ADB9E-0989-4525-B853-CFA5081510A5" 17 | } -------------------------------------------------------------------------------- /sheetExample.framer/framer/framer.generated.js: -------------------------------------------------------------------------------- 1 | // This is autogenerated by Framer 2 | 3 | 4 | if (!window.Framer && window._bridge) {window._bridge('runtime.error', {message:'[framer.js] Framer library missing or corrupt. Select File → Update Framer Library.'})} 5 | if (DeviceComponent) {DeviceComponent.Devices["iphone-6-silver"].deviceImageJP2 = false}; 6 | if (window.Framer) {window.Framer.Defaults.DeviceView = {"deviceScale":1,"selectedHand":"","deviceType":"fullscreen","contentScale":1,"orientation":0}; 7 | } 8 | if (window.Framer) {window.Framer.Defaults.DeviceComponent = {"deviceScale":1,"selectedHand":"","deviceType":"fullscreen","contentScale":1,"orientation":0}; 9 | } 10 | window.FramerStudioInfo = {"deviceImagesUrl":"\/_server\/resources\/DeviceImages","documentTitle":"sheetExample.framer"}; 11 | 12 | Framer.Device = new Framer.DeviceView(); 13 | Framer.Device.setupContext(); -------------------------------------------------------------------------------- /sheetExample.framer/framer/framer.init.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | 3 | function isFileLoadingAllowed() { 4 | return (window.location.protocol.indexOf("file") == -1) 5 | } 6 | 7 | function isHomeScreened() { 8 | return ("standalone" in window.navigator) && window.navigator.standalone == true 9 | } 10 | 11 | function isCompatibleBrowser() { 12 | return Utils.isWebKit() 13 | } 14 | 15 | var alertNode; 16 | 17 | function dismissAlert() { 18 | alertNode.parentElement.removeChild(alertNode) 19 | loadProject() 20 | } 21 | 22 | function showAlert(html) { 23 | 24 | alertNode = document.createElement("div") 25 | 26 | alertNode.classList.add("framerAlertBackground") 27 | alertNode.innerHTML = html 28 | 29 | document.addEventListener("DOMContentLoaded", function(event) { 30 | document.body.appendChild(alertNode) 31 | }) 32 | 33 | window.dismissAlert = dismissAlert; 34 | } 35 | 36 | function showBrowserAlert() { 37 | var html = "" 38 | html += "
" 39 | html += "Error: Not A WebKit Browser" 40 | html += "Your browser is not supported.
Please use Safari or Chrome.
" 41 | html += "Try anyway" 42 | html += "
" 43 | 44 | showAlert(html) 45 | } 46 | 47 | function showFileLoadingAlert() { 48 | var html = "" 49 | html += "
" 50 | html += "Error: Local File Restrictions" 51 | html += "Preview this prototype with Framer Mirror or learn more about " 52 | html += "file restrictions.
" 53 | html += "Try anyway" 54 | html += "
" 55 | 56 | showAlert(html) 57 | } 58 | 59 | function loadProject() { 60 | CoffeeScript.load("app.coffee") 61 | } 62 | 63 | function setDefaultPageTitle() { 64 | // If no title was set we set it to the project folder name so 65 | // you get a nice name on iOS if you bookmark to desktop. 66 | document.addEventListener("DOMContentLoaded", function() { 67 | if (document.title == "") { 68 | if (window.FramerStudioInfo && window.FramerStudioInfo.documentTitle) { 69 | document.title = window.FramerStudioInfo.documentTitle 70 | } else { 71 | document.title = window.location.pathname.replace(/\//g, "") 72 | } 73 | } 74 | }) 75 | } 76 | 77 | function init() { 78 | 79 | if (Utils.isFramerStudio()) { 80 | return 81 | } 82 | 83 | setDefaultPageTitle() 84 | 85 | if (!isCompatibleBrowser()) { 86 | return showBrowserAlert() 87 | } 88 | 89 | if (!isFileLoadingAllowed()) { 90 | return showFileLoadingAlert() 91 | } 92 | 93 | loadProject() 94 | 95 | } 96 | 97 | init() 98 | 99 | })() 100 | -------------------------------------------------------------------------------- /sheetExample.framer/framer/framer.modules.js: -------------------------------------------------------------------------------- 1 | require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o 1) { 138 | for (var i = 1; i < arguments.length; i++) { 139 | args[i - 1] = arguments[i]; 140 | } 141 | } 142 | queue.push(new Item(fun, args)); 143 | if (queue.length === 1 && !draining) { 144 | runTimeout(drainQueue); 145 | } 146 | }; 147 | 148 | // v8 likes predictible objects 149 | function Item(fun, array) { 150 | this.fun = fun; 151 | this.array = array; 152 | } 153 | Item.prototype.run = function () { 154 | this.fun.apply(null, this.array); 155 | }; 156 | process.title = 'browser'; 157 | process.browser = true; 158 | process.env = {}; 159 | process.argv = []; 160 | process.version = ''; // empty string to avoid regexp issues 161 | process.versions = {}; 162 | 163 | function noop() {} 164 | 165 | process.on = noop; 166 | process.addListener = noop; 167 | process.once = noop; 168 | process.off = noop; 169 | process.removeListener = noop; 170 | process.removeAllListeners = noop; 171 | process.emit = noop; 172 | 173 | process.binding = function (name) { 174 | throw new Error('process.binding is not supported'); 175 | }; 176 | 177 | process.cwd = function () { return '/' }; 178 | process.chdir = function (dir) { 179 | throw new Error('process.chdir is not supported'); 180 | }; 181 | process.umask = function() { return 0; }; 182 | 183 | },{}],2:[function(require,module,exports){ 184 | (function (process){ 185 | (function() { 186 | 'use strict'; 187 | 188 | var inNodeJS = false; 189 | if (typeof process !== 'undefined' && !process.browser) { 190 | inNodeJS = true; 191 | var request = require('request'.trim()); //prevents browserify from bundling the module 192 | } 193 | 194 | var supportsCORS = false; 195 | var inLegacyIE = false; 196 | try { 197 | var testXHR = new XMLHttpRequest(); 198 | if (typeof testXHR.withCredentials !== 'undefined') { 199 | supportsCORS = true; 200 | } else { 201 | if ('XDomainRequest' in window) { 202 | supportsCORS = true; 203 | inLegacyIE = true; 204 | } 205 | } 206 | } catch (e) { } 207 | 208 | // Create a simple indexOf function for support 209 | // of older browsers. Uses native indexOf if 210 | // available. Code similar to underscores. 211 | // By making a separate function, instead of adding 212 | // to the prototype, we will not break bad for loops 213 | // in older browsers 214 | var indexOfProto = Array.prototype.indexOf; 215 | var ttIndexOf = function(array, item) { 216 | var i = 0, l = array.length; 217 | 218 | if (indexOfProto && array.indexOf === indexOfProto) { 219 | return array.indexOf(item); 220 | } 221 | 222 | for (; i < l; i++) { 223 | if (array[i] === item) { 224 | return i; 225 | } 226 | } 227 | return -1; 228 | }; 229 | 230 | /* 231 | Initialize with Tabletop.init( { key: '0AjAPaAU9MeLFdHUxTlJiVVRYNGRJQnRmSnQwTlpoUXc' } ) 232 | OR! 233 | Initialize with Tabletop.init( { key: 'https://docs.google.com/spreadsheet/pub?hl=en_US&hl=en_US&key=0AjAPaAU9MeLFdHUxTlJiVVRYNGRJQnRmSnQwTlpoUXc&output=html&widget=true' } ) 234 | OR! 235 | Initialize with Tabletop.init('0AjAPaAU9MeLFdHUxTlJiVVRYNGRJQnRmSnQwTlpoUXc') 236 | */ 237 | 238 | var Tabletop = function(options) { 239 | // Make sure Tabletop is being used as a constructor no matter what. 240 | if(!this || !(this instanceof Tabletop)) { 241 | return new Tabletop(options); 242 | } 243 | 244 | if(typeof(options) === 'string') { 245 | options = { key : options }; 246 | } 247 | 248 | this.callback = options.callback; 249 | this.wanted = options.wanted || []; 250 | this.key = options.key; 251 | this.simpleSheet = !!options.simpleSheet; 252 | this.parseNumbers = !!options.parseNumbers; 253 | this.wait = !!options.wait; 254 | this.reverse = !!options.reverse; 255 | this.postProcess = options.postProcess; 256 | this.debug = !!options.debug; 257 | this.query = options.query || ''; 258 | this.orderby = options.orderby; 259 | this.endpoint = options.endpoint || 'https://spreadsheets.google.com'; 260 | this.singleton = !!options.singleton; 261 | this.simpleUrl = !!(options.simpleUrl || options.simple_url); //jshint ignore:line 262 | this.callbackContext = options.callbackContext; 263 | // Default to on, unless there's a proxy, in which case it's default off 264 | this.prettyColumnNames = typeof(options.prettyColumnNames) === 'undefined' ? !options.proxy : options.prettyColumnNames; 265 | 266 | if(typeof(options.proxy) !== 'undefined') { 267 | // Remove trailing slash, it will break the app 268 | this.endpoint = options.proxy.replace(/\/$/,''); 269 | this.simpleUrl = true; 270 | this.singleton = true; 271 | // Let's only use CORS (straight JSON request) when 272 | // fetching straight from Google 273 | supportsCORS = false; 274 | } 275 | 276 | this.parameterize = options.parameterize || false; 277 | 278 | if (this.singleton) { 279 | if (typeof(Tabletop.singleton) !== 'undefined') { 280 | this.log('WARNING! Tabletop singleton already defined'); 281 | } 282 | Tabletop.singleton = this; 283 | } 284 | 285 | /* Be friendly about what you accept */ 286 | if (/key=/.test(this.key)) { 287 | this.log('You passed an old Google Docs url as the key! Attempting to parse.'); 288 | this.key = this.key.match('key=(.*?)(&|#|$)')[1]; 289 | } 290 | 291 | if (/pubhtml/.test(this.key)) { 292 | this.log('You passed a new Google Spreadsheets url as the key! Attempting to parse.'); 293 | this.key = this.key.match('d\\/(.*?)\\/pubhtml')[1]; 294 | } 295 | 296 | if(/spreadsheets\/d/.test(this.key)) { 297 | this.log('You passed the most recent version of Google Spreadsheets url as the key! Attempting to parse.'); 298 | this.key = this.key.match('d\\/(.*?)\/')[1]; 299 | } 300 | 301 | if (!this.key) { 302 | this.log('You need to pass Tabletop a key!'); 303 | return; 304 | } 305 | 306 | this.log('Initializing with key ' + this.key); 307 | 308 | this.models = {}; 309 | this.modelNames = []; 310 | this.model_names = this.modelNames; //jshint ignore:line 311 | 312 | this.baseJsonPath = '/feeds/worksheets/' + this.key + '/public/basic?alt='; 313 | 314 | if (inNodeJS || supportsCORS) { 315 | this.baseJsonPath += 'json'; 316 | } else { 317 | this.baseJsonPath += 'json-in-script'; 318 | } 319 | 320 | if(!this.wait) { 321 | this.fetch(); 322 | } 323 | }; 324 | 325 | // A global storage for callbacks. 326 | Tabletop.callbacks = {}; 327 | 328 | // Backwards compatibility. 329 | Tabletop.init = function(options) { 330 | return new Tabletop(options); 331 | }; 332 | 333 | Tabletop.sheets = function() { 334 | this.log('Times have changed! You\'ll want to use var tabletop = Tabletop.init(...); tabletop.sheets(...); instead of Tabletop.sheets(...)'); 335 | }; 336 | 337 | Tabletop.prototype = { 338 | 339 | fetch: function(callback) { 340 | if (typeof(callback) !== 'undefined') { 341 | this.callback = callback; 342 | } 343 | this.requestData(this.baseJsonPath, this.loadSheets); 344 | }, 345 | 346 | /* 347 | This will call the environment appropriate request method. 348 | 349 | In browser it will use JSON-P, in node it will use request() 350 | */ 351 | requestData: function(path, callback) { 352 | this.log('Requesting', path); 353 | 354 | if (inNodeJS) { 355 | this.serverSideFetch(path, callback); 356 | } else { 357 | //CORS only works in IE8/9 across the same protocol 358 | //You must have your server on HTTPS to talk to Google, or it'll fall back on injection 359 | var protocol = this.endpoint.split('//').shift() || 'http'; 360 | if (supportsCORS && (!inLegacyIE || protocol === location.protocol)) { 361 | this.xhrFetch(path, callback); 362 | } else { 363 | this.injectScript(path, callback); 364 | } 365 | } 366 | }, 367 | 368 | /* 369 | Use Cross-Origin XMLHttpRequest to get the data in browsers that support it. 370 | */ 371 | xhrFetch: function(path, callback) { 372 | //support IE8's separate cross-domain object 373 | var xhr = inLegacyIE ? new XDomainRequest() : new XMLHttpRequest(); 374 | xhr.open('GET', this.endpoint + path); 375 | var self = this; 376 | xhr.onload = function() { 377 | var json; 378 | try { 379 | json = JSON.parse(xhr.responseText); 380 | } catch (e) { 381 | console.error(e); 382 | } 383 | callback.call(self, json); 384 | }; 385 | xhr.send(); 386 | }, 387 | 388 | /* 389 | Insert the URL into the page as a script tag. Once it's loaded the spreadsheet data 390 | it triggers the callback. This helps you avoid cross-domain errors 391 | http://code.google.com/apis/gdata/samples/spreadsheet_sample.html 392 | 393 | Let's be plain-Jane and not use jQuery or anything. 394 | */ 395 | injectScript: function(path, callback) { 396 | var script = document.createElement('script'); 397 | var callbackName; 398 | 399 | if (this.singleton) { 400 | if (callback === this.loadSheets) { 401 | callbackName = 'Tabletop.singleton.loadSheets'; 402 | } else if (callback === this.loadSheet) { 403 | callbackName = 'Tabletop.singleton.loadSheet'; 404 | } 405 | } else { 406 | var self = this; 407 | callbackName = 'tt' + (+new Date()) + (Math.floor(Math.random()*100000)); 408 | // Create a temp callback which will get removed once it has executed, 409 | // this allows multiple instances of Tabletop to coexist. 410 | Tabletop.callbacks[ callbackName ] = function () { 411 | var args = Array.prototype.slice.call( arguments, 0 ); 412 | callback.apply(self, args); 413 | script.parentNode.removeChild(script); 414 | delete Tabletop.callbacks[callbackName]; 415 | }; 416 | callbackName = 'Tabletop.callbacks.' + callbackName; 417 | } 418 | 419 | var url = path + '&callback=' + callbackName; 420 | 421 | if (this.simpleUrl) { 422 | // We've gone down a rabbit hole of passing injectScript the path, so let's 423 | // just pull the sheet_id out of the path like the least efficient worker bees 424 | if(path.indexOf('/list/') !== -1) { 425 | script.src = this.endpoint + '/' + this.key + '-' + path.split('/')[4]; 426 | } else { 427 | script.src = this.endpoint + '/' + this.key; 428 | } 429 | } else { 430 | script.src = this.endpoint + url; 431 | } 432 | 433 | if (this.parameterize) { 434 | script.src = this.parameterize + encodeURIComponent(script.src); 435 | } 436 | 437 | this.log('Injecting', script.src); 438 | 439 | document.getElementsByTagName('script')[0].parentNode.appendChild(script); 440 | }, 441 | 442 | /* 443 | This will only run if tabletop is being run in node.js 444 | */ 445 | serverSideFetch: function(path, callback) { 446 | var self = this; 447 | 448 | this.log('Fetching', this.endpoint + path); 449 | request({url: this.endpoint + path, json: true}, function(err, resp, body) { 450 | if (err) { 451 | return console.error(err); 452 | } 453 | callback.call(self, body); 454 | }); 455 | }, 456 | 457 | /* 458 | Is this a sheet you want to pull? 459 | If { wanted: ["Sheet1"] } has been specified, only Sheet1 is imported 460 | Pulls all sheets if none are specified 461 | */ 462 | isWanted: function(sheetName) { 463 | if (this.wanted.length === 0) { 464 | return true; 465 | } else { 466 | return (ttIndexOf(this.wanted, sheetName) !== -1); 467 | } 468 | }, 469 | 470 | /* 471 | What gets send to the callback 472 | if simpleSheet === true, then don't return an array of Tabletop.this.models, 473 | only return the first one's elements 474 | */ 475 | data: function() { 476 | // If the instance is being queried before the data's been fetched 477 | // then return undefined. 478 | if (this.modelNames.length === 0) { 479 | return undefined; 480 | } 481 | if (this.simpleSheet) { 482 | if (this.modelNames.length > 1 && this.debug) { 483 | this.log('WARNING You have more than one sheet but are using simple sheet mode! Don\'t blame me when something goes wrong.'); 484 | } 485 | return this.models[this.modelNames[0]].all(); 486 | } else { 487 | return this.models; 488 | } 489 | }, 490 | 491 | /* 492 | Add another sheet to the wanted list 493 | */ 494 | addWanted: function(sheet) { 495 | if(ttIndexOf(this.wanted, sheet) === -1) { 496 | this.wanted.push(sheet); 497 | } 498 | }, 499 | 500 | /* 501 | Load all worksheets of the spreadsheet, turning each into a Tabletop Model. 502 | Need to use injectScript because the worksheet view that you're working from 503 | doesn't actually include the data. The list-based feed (/feeds/list/key..) does, though. 504 | Calls back to loadSheet in order to get the real work done. 505 | 506 | Used as a callback for the worksheet-based JSON 507 | */ 508 | loadSheets: function(data) { 509 | var i, ilen; 510 | var toLoad = []; 511 | this.googleSheetName = data.feed.title.$t; 512 | this.foundSheetNames = []; 513 | 514 | for (i = 0, ilen = data.feed.entry.length; i < ilen ; i++) { 515 | this.foundSheetNames.push(data.feed.entry[i].title.$t); 516 | // Only pull in desired sheets to reduce loading 517 | if (this.isWanted(data.feed.entry[i].content.$t)) { 518 | var linkIdx = data.feed.entry[i].link.length-1; 519 | var sheetId = data.feed.entry[i].link[linkIdx].href.split('/').pop(); 520 | var jsonPath = '/feeds/list/' + this.key + '/' + sheetId + '/public/values?alt='; 521 | if (inNodeJS || supportsCORS) { 522 | jsonPath += 'json'; 523 | } else { 524 | jsonPath += 'json-in-script'; 525 | } 526 | if (this.query) { 527 | // Query Language Reference (0.7) 528 | jsonPath += '&tq=' + this.query; 529 | } 530 | if (this.orderby) { 531 | jsonPath += '&orderby=column:' + this.orderby.toLowerCase(); 532 | } 533 | if (this.reverse) { 534 | jsonPath += '&reverse=true'; 535 | } 536 | toLoad.push(jsonPath); 537 | } 538 | } 539 | 540 | this.sheetsToLoad = toLoad.length; 541 | for(i = 0, ilen = toLoad.length; i < ilen; i++) { 542 | this.requestData(toLoad[i], this.loadSheet); 543 | } 544 | }, 545 | 546 | /* 547 | Access layer for the this.models 548 | .sheets() gets you all of the sheets 549 | .sheets('Sheet1') gets you the sheet named Sheet1 550 | */ 551 | sheets: function(sheetName) { 552 | if (typeof sheetName === 'undefined') { 553 | return this.models; 554 | } else { 555 | if (typeof(this.models[sheetName]) === 'undefined') { 556 | // alert( "Can't find " + sheetName ); 557 | return; 558 | } else { 559 | return this.models[sheetName]; 560 | } 561 | } 562 | }, 563 | 564 | sheetReady: function(model) { 565 | this.models[model.name] = model; 566 | if (ttIndexOf(this.modelNames, model.name) === -1) { 567 | this.modelNames.push(model.name); 568 | } 569 | 570 | this.sheetsToLoad--; 571 | if (this.sheetsToLoad === 0) { 572 | this.doCallback(); 573 | } 574 | }, 575 | 576 | /* 577 | Parse a single list-based worksheet, turning it into a Tabletop Model 578 | 579 | Used as a callback for the list-based JSON 580 | */ 581 | loadSheet: function(data) { 582 | var that = this; 583 | new Tabletop.Model({ 584 | data: data, 585 | parseNumbers: this.parseNumbers, 586 | postProcess: this.postProcess, 587 | tabletop: this, 588 | prettyColumnNames: this.prettyColumnNames, 589 | onReady: function() { 590 | that.sheetReady(this); 591 | } 592 | }); 593 | }, 594 | 595 | /* 596 | Execute the callback upon loading! Rely on this.data() because you might 597 | only request certain pieces of data (i.e. simpleSheet mode) 598 | Tests this.sheetsToLoad just in case a race condition happens to show up 599 | */ 600 | doCallback: function() { 601 | if(this.sheetsToLoad === 0) { 602 | this.callback.apply(this.callbackContext || this, [this.data(), this]); 603 | } 604 | }, 605 | 606 | log: function() { 607 | if(this.debug) { 608 | if(typeof console !== 'undefined' && typeof console.log !== 'undefined') { 609 | Function.prototype.apply.apply(console.log, [console, arguments]); 610 | } 611 | } 612 | } 613 | 614 | }; 615 | 616 | /* 617 | Tabletop.Model stores the attribute names and parses the worksheet data 618 | to turn it into something worthwhile 619 | 620 | Options should be in the format { data: XXX }, with XXX being the list-based worksheet 621 | */ 622 | Tabletop.Model = function(options) { 623 | var i, j, ilen, jlen; 624 | this.columnNames = []; 625 | this.column_names = this.columnNames; // jshint ignore:line 626 | this.name = options.data.feed.title.$t; 627 | this.tabletop = options.tabletop; 628 | this.elements = []; 629 | this.onReady = options.onReady; 630 | this.raw = options.data; // A copy of the sheet's raw data, for accessing minutiae 631 | 632 | if (typeof(options.data.feed.entry) === 'undefined') { 633 | options.tabletop.log('Missing data for ' + this.name + ', make sure you didn\'t forget column headers'); 634 | this.originalColumns = []; 635 | this.elements = []; 636 | this.onReady.call(this); 637 | return; 638 | } 639 | 640 | for (var key in options.data.feed.entry[0]){ 641 | if (/^gsx/.test(key)) { 642 | this.columnNames.push(key.replace('gsx$','')); 643 | } 644 | } 645 | 646 | this.originalColumns = this.columnNames; 647 | this.original_columns = this.originalColumns; // jshint ignore:line 648 | 649 | for (i = 0, ilen = options.data.feed.entry.length ; i < ilen; i++) { 650 | var source = options.data.feed.entry[i]; 651 | var element = {}; 652 | for (j = 0, jlen = this.columnNames.length; j < jlen ; j++) { 653 | var cell = source['gsx$' + this.columnNames[j]]; 654 | if (typeof(cell) !== 'undefined') { 655 | if (options.parseNumbers && cell.$t !== '' && !isNaN(cell.$t)) { 656 | element[this.columnNames[j]] = +cell.$t; 657 | } else { 658 | element[this.columnNames[j]] = cell.$t; 659 | } 660 | } else { 661 | element[this.columnNames[j]] = ''; 662 | } 663 | } 664 | if (element.rowNumber === undefined) { 665 | element.rowNumber = i + 1; 666 | } 667 | 668 | if (options.postProcess) { 669 | options.postProcess(element); 670 | } 671 | 672 | this.elements.push(element); 673 | } 674 | 675 | if (options.prettyColumnNames) { 676 | this.fetchPrettyColumns(); 677 | } else { 678 | this.onReady.call(this); 679 | } 680 | }; 681 | 682 | Tabletop.Model.prototype = { 683 | /* 684 | Returns all of the elements (rows) of the worksheet as objects 685 | */ 686 | all: function() { 687 | return this.elements; 688 | }, 689 | 690 | fetchPrettyColumns: function() { 691 | if (!this.raw.feed.link[3]) { 692 | return this.ready(); 693 | } 694 | 695 | var cellurl = this.raw.feed.link[3].href.replace('/feeds/list/', '/feeds/cells/').replace('https://spreadsheets.google.com', ''); 696 | var that = this; 697 | this.tabletop.requestData(cellurl, function(data) { 698 | that.loadPrettyColumns(data); 699 | }); 700 | }, 701 | 702 | ready: function() { 703 | this.onReady.call(this); 704 | }, 705 | 706 | /* 707 | * Store column names as an object 708 | * with keys of Google-formatted "columnName" 709 | * and values of human-readable "Column name" 710 | */ 711 | loadPrettyColumns: function(data) { 712 | var prettyColumns = {}; 713 | 714 | var columnNames = this.columnNames; 715 | 716 | var i = 0; 717 | var l = columnNames.length; 718 | 719 | for (; i < l; i++) { 720 | if (typeof data.feed.entry[i].content.$t !== 'undefined') { 721 | prettyColumns[columnNames[i]] = data.feed.entry[i].content.$t; 722 | } else { 723 | prettyColumns[columnNames[i]] = columnNames[i]; 724 | } 725 | } 726 | 727 | this.prettyColumns = prettyColumns; 728 | this.pretty_columns = this.prettyColumns; // jshint ignore:line 729 | this.prettifyElements(); 730 | this.ready(); 731 | }, 732 | 733 | /* 734 | * Go through each row, substitutiting 735 | * Google-formatted "columnName" 736 | * with human-readable "Column name" 737 | */ 738 | prettifyElements: function() { 739 | var prettyElements = [], 740 | orderedPrettyNames = [], 741 | i, j, ilen, jlen; 742 | 743 | for (j = 0, jlen = this.columnNames.length; j < jlen ; j++) { 744 | orderedPrettyNames.push(this.prettyColumns[this.columnNames[j]]); 745 | } 746 | 747 | for (i = 0, ilen = this.elements.length; i < ilen; i++) { 748 | var newElement = {}; 749 | for (j = 0, jlen = this.columnNames.length; j < jlen ; j++) { 750 | var newColumnName = this.prettyColumns[this.columnNames[j]]; 751 | newElement[newColumnName] = this.elements[i][this.columnNames[j]]; 752 | } 753 | prettyElements.push(newElement); 754 | } 755 | this.elements = prettyElements; 756 | this.columnNames = orderedPrettyNames; 757 | }, 758 | 759 | /* 760 | Return the elements as an array of arrays, instead of an array of objects 761 | */ 762 | toArray: function() { 763 | var array = [], 764 | i, j, ilen, jlen; 765 | for (i = 0, ilen = this.elements.length; i < ilen; i++) { 766 | var row = []; 767 | for (j = 0, jlen = this.columnNames.length; j < jlen ; j++) { 768 | row.push(this.elements[i][ this.columnNames[j]]); 769 | } 770 | array.push(row); 771 | } 772 | 773 | return array; 774 | } 775 | }; 776 | 777 | if(typeof module !== 'undefined' && module.exports) { //don't just use inNodeJS, we may be in Browserify 778 | module.exports = Tabletop; 779 | } else if (typeof define === 'function' && define.amd) { 780 | define(function () { 781 | return Tabletop; 782 | }); 783 | } else { 784 | window.Tabletop = Tabletop; 785 | } 786 | 787 | })(); 788 | }).call(this,require('_process')) 789 | 790 | },{"_process":1}],"npm":[function(require,module,exports){ 791 | exports.Tabletop = require('tabletop'); 792 | 793 | 794 | },{"tabletop":2}],"sheet":[function(require,module,exports){ 795 | var Tabletop; 796 | 797 | Tabletop = require('npm').Tabletop; 798 | 799 | exports.Sheet = (function() { 800 | function Sheet(options) { 801 | this._key = options.key; 802 | this.get = (function(_this) { 803 | return function(callback) { 804 | return Tabletop.init({ 805 | key: _this._key, 806 | simpleSheet: true, 807 | callback: function(data, sheet) { 808 | return callback(data, sheet); 809 | } 810 | }); 811 | }; 812 | })(this); 813 | } 814 | 815 | return Sheet; 816 | 817 | })(); 818 | 819 | 820 | },{"npm":"npm"}]},{},[]) 821 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"framer.modules.js","sources":["../../../../../Users/liebchen/Dropbox (Facebook)/Personal/Framer/Sheet/sheetExample.framer/modules/sheet.coffee","../../../../../Users/liebchen/Dropbox (Facebook)/Personal/Framer/Sheet/sheetExample.framer/modules/npm.coffee","../../../../../Users/liebchen/Dropbox (Facebook)/Personal/Framer/Sheet/sheetExample.framer/node_modules/tabletop/src/tabletop.js","node_modules/process/browser.js","node_modules/browser-pack/_prelude.js"],"sourcesContent":["{ Tabletop } = require 'npm'\n\n\nclass exports.Sheet\n  constructor: (options) ->\n    @_key = options.key\n\n    @get = (callback) =>\n      Tabletop.init {\n        key: @_key\n        simpleSheet: true\n        callback: (data, sheet) =>\n          return callback(data, sheet)\n      }\n","exports.Tabletop = require 'tabletop'\n","(function() {\n  'use strict';\n\n  var inNodeJS = false;\n  if (typeof process !== 'undefined' && !process.browser) {\n    inNodeJS = true;\n    var request = require('request'.trim()); //prevents browserify from bundling the module\n  }\n\n  var supportsCORS = false;\n  var inLegacyIE = false;\n  try {\n    var testXHR = new XMLHttpRequest();\n    if (typeof testXHR.withCredentials !== 'undefined') {\n      supportsCORS = true;\n    } else {\n      if ('XDomainRequest' in window) {\n        supportsCORS = true;\n        inLegacyIE = true;\n      }\n    }\n  } catch (e) { }\n\n  // Create a simple indexOf function for support\n  // of older browsers.  Uses native indexOf if \n  // available.  Code similar to underscores.\n  // By making a separate function, instead of adding\n  // to the prototype, we will not break bad for loops\n  // in older browsers\n  var indexOfProto = Array.prototype.indexOf;\n  var ttIndexOf = function(array, item) {\n    var i = 0, l = array.length;\n    \n    if (indexOfProto && array.indexOf === indexOfProto) {\n      return array.indexOf(item);\n    }\n    \n    for (; i < l; i++) {\n      if (array[i] === item) {\n        return i;\n      }\n    }\n    return -1;\n  };\n  \n  /*\n    Initialize with Tabletop.init( { key: '0AjAPaAU9MeLFdHUxTlJiVVRYNGRJQnRmSnQwTlpoUXc' } )\n      OR!\n    Initialize with Tabletop.init( { key: 'https://docs.google.com/spreadsheet/pub?hl=en_US&hl=en_US&key=0AjAPaAU9MeLFdHUxTlJiVVRYNGRJQnRmSnQwTlpoUXc&output=html&widget=true' } )\n      OR!\n    Initialize with Tabletop.init('0AjAPaAU9MeLFdHUxTlJiVVRYNGRJQnRmSnQwTlpoUXc')\n  */\n\n  var Tabletop = function(options) {\n    // Make sure Tabletop is being used as a constructor no matter what.\n    if(!this || !(this instanceof Tabletop)) {\n      return new Tabletop(options);\n    }\n    \n    if(typeof(options) === 'string') {\n      options = { key : options };\n    }\n\n    this.callback = options.callback;\n    this.wanted = options.wanted || [];\n    this.key = options.key;\n    this.simpleSheet = !!options.simpleSheet;\n    this.parseNumbers = !!options.parseNumbers;\n    this.wait = !!options.wait;\n    this.reverse = !!options.reverse;\n    this.postProcess = options.postProcess;\n    this.debug = !!options.debug;\n    this.query = options.query || '';\n    this.orderby = options.orderby;\n    this.endpoint = options.endpoint || 'https://spreadsheets.google.com';\n    this.singleton = !!options.singleton;\n    this.simpleUrl = !!(options.simpleUrl || options.simple_url); //jshint ignore:line\n    this.callbackContext = options.callbackContext;\n    // Default to on, unless there's a proxy, in which case it's default off\n    this.prettyColumnNames = typeof(options.prettyColumnNames) === 'undefined' ? !options.proxy : options.prettyColumnNames;\n    \n    if(typeof(options.proxy) !== 'undefined') {\n      // Remove trailing slash, it will break the app\n      this.endpoint = options.proxy.replace(/\\/$/,'');\n      this.simpleUrl = true;\n      this.singleton = true;\n      // Let's only use CORS (straight JSON request) when\n      // fetching straight from Google\n      supportsCORS = false;\n    }\n    \n    this.parameterize = options.parameterize || false;\n    \n    if (this.singleton) {\n      if (typeof(Tabletop.singleton) !== 'undefined') {\n        this.log('WARNING! Tabletop singleton already defined');\n      }\n      Tabletop.singleton = this;\n    }\n    \n    /* Be friendly about what you accept */\n    if (/key=/.test(this.key)) {\n      this.log('You passed an old Google Docs url as the key! Attempting to parse.');\n      this.key = this.key.match('key=(.*?)(&|#|$)')[1];\n    }\n\n    if (/pubhtml/.test(this.key)) {\n      this.log('You passed a new Google Spreadsheets url as the key! Attempting to parse.');\n      this.key = this.key.match('d\\\\/(.*?)\\\\/pubhtml')[1];\n    }\n    \n    if(/spreadsheets\\/d/.test(this.key)) {\n      this.log('You passed the most recent version of Google Spreadsheets url as the key! Attempting to parse.');\n      this.key = this.key.match('d\\\\/(.*?)\\/')[1];\n    }\n\n    if (!this.key) {\n      this.log('You need to pass Tabletop a key!');\n      return;\n    }\n\n    this.log('Initializing with key ' + this.key);\n\n    this.models = {};\n    this.modelNames = [];\n    this.model_names = this.modelNames; //jshint ignore:line\n\n    this.baseJsonPath = '/feeds/worksheets/' + this.key + '/public/basic?alt=';\n\n    if (inNodeJS || supportsCORS) {\n      this.baseJsonPath += 'json';\n    } else {\n      this.baseJsonPath += 'json-in-script';\n    }\n    \n    if(!this.wait) {\n      this.fetch();\n    }\n  };\n\n  // A global storage for callbacks.\n  Tabletop.callbacks = {};\n\n  // Backwards compatibility.\n  Tabletop.init = function(options) {\n    return new Tabletop(options);\n  };\n\n  Tabletop.sheets = function() {\n    this.log('Times have changed! You\\'ll want to use var tabletop = Tabletop.init(...); tabletop.sheets(...); instead of Tabletop.sheets(...)');\n  };\n\n  Tabletop.prototype = {\n\n    fetch: function(callback) {\n      if (typeof(callback) !== 'undefined') {\n        this.callback = callback;\n      }\n      this.requestData(this.baseJsonPath, this.loadSheets);\n    },\n    \n    /*\n      This will call the environment appropriate request method.\n      \n      In browser it will use JSON-P, in node it will use request()\n    */\n    requestData: function(path, callback) {\n      this.log('Requesting', path);\n\n      if (inNodeJS) {\n        this.serverSideFetch(path, callback);\n      } else {\n        //CORS only works in IE8/9 across the same protocol\n        //You must have your server on HTTPS to talk to Google, or it'll fall back on injection\n        var protocol = this.endpoint.split('//').shift() || 'http';\n        if (supportsCORS && (!inLegacyIE || protocol === location.protocol)) {\n          this.xhrFetch(path, callback);\n        } else {\n          this.injectScript(path, callback);\n        }\n      }\n    },\n\n    /*\n      Use Cross-Origin XMLHttpRequest to get the data in browsers that support it.\n    */\n    xhrFetch: function(path, callback) {\n      //support IE8's separate cross-domain object\n      var xhr = inLegacyIE ? new XDomainRequest() : new XMLHttpRequest();\n      xhr.open('GET', this.endpoint + path);\n      var self = this;\n      xhr.onload = function() {\n        var json;\n        try {\n          json = JSON.parse(xhr.responseText);\n        } catch (e) {\n          console.error(e);\n        }\n        callback.call(self, json);\n      };\n      xhr.send();\n    },\n    \n    /*\n      Insert the URL into the page as a script tag. Once it's loaded the spreadsheet data\n      it triggers the callback. This helps you avoid cross-domain errors\n      http://code.google.com/apis/gdata/samples/spreadsheet_sample.html\n\n      Let's be plain-Jane and not use jQuery or anything.\n    */\n    injectScript: function(path, callback) {\n      var script = document.createElement('script');\n      var callbackName;\n      \n      if (this.singleton) {\n        if (callback === this.loadSheets) {\n          callbackName = 'Tabletop.singleton.loadSheets';\n        } else if (callback === this.loadSheet) {\n          callbackName = 'Tabletop.singleton.loadSheet';\n        }\n      } else {\n        var self = this;\n        callbackName = 'tt' + (+new Date()) + (Math.floor(Math.random()*100000));\n        // Create a temp callback which will get removed once it has executed,\n        // this allows multiple instances of Tabletop to coexist.\n        Tabletop.callbacks[ callbackName ] = function () {\n          var args = Array.prototype.slice.call( arguments, 0 );\n          callback.apply(self, args);\n          script.parentNode.removeChild(script);\n          delete Tabletop.callbacks[callbackName];\n        };\n        callbackName = 'Tabletop.callbacks.' + callbackName;\n      }\n      \n      var url = path + '&callback=' + callbackName;\n      \n      if (this.simpleUrl) {\n        // We've gone down a rabbit hole of passing injectScript the path, so let's\n        // just pull the sheet_id out of the path like the least efficient worker bees\n        if(path.indexOf('/list/') !== -1) {\n          script.src = this.endpoint + '/' + this.key + '-' + path.split('/')[4];\n        } else {\n          script.src = this.endpoint + '/' + this.key;\n        }\n      } else {\n        script.src = this.endpoint + url;\n      }\n      \n      if (this.parameterize) {\n        script.src = this.parameterize + encodeURIComponent(script.src);\n      }\n\n      this.log('Injecting', script.src);\n\n      document.getElementsByTagName('script')[0].parentNode.appendChild(script);\n    },\n    \n    /* \n      This will only run if tabletop is being run in node.js\n    */\n    serverSideFetch: function(path, callback) {\n      var self = this;\n\n      this.log('Fetching', this.endpoint + path);\n      request({url: this.endpoint + path, json: true}, function(err, resp, body) {\n        if (err) {\n          return console.error(err);\n        }\n        callback.call(self, body);\n      });\n    },\n\n    /* \n      Is this a sheet you want to pull?\n      If { wanted: [\"Sheet1\"] } has been specified, only Sheet1 is imported\n      Pulls all sheets if none are specified\n    */\n    isWanted: function(sheetName) {\n      if (this.wanted.length === 0) {\n        return true;\n      } else {\n        return (ttIndexOf(this.wanted, sheetName) !== -1);\n      }\n    },\n    \n    /*\n      What gets send to the callback\n      if simpleSheet === true, then don't return an array of Tabletop.this.models,\n      only return the first one's elements\n    */\n    data: function() {\n      // If the instance is being queried before the data's been fetched\n      // then return undefined.\n      if (this.modelNames.length === 0) {\n        return undefined;\n      }\n      if (this.simpleSheet) {\n        if (this.modelNames.length > 1 && this.debug) {\n          this.log('WARNING You have more than one sheet but are using simple sheet mode! Don\\'t blame me when something goes wrong.');\n        }\n        return this.models[this.modelNames[0]].all();\n      } else {\n        return this.models;\n      }\n    },\n\n    /*\n      Add another sheet to the wanted list\n    */\n    addWanted: function(sheet) {\n      if(ttIndexOf(this.wanted, sheet) === -1) {\n        this.wanted.push(sheet);\n      }\n    },\n    \n    /*\n      Load all worksheets of the spreadsheet, turning each into a Tabletop Model.\n      Need to use injectScript because the worksheet view that you're working from\n      doesn't actually include the data. The list-based feed (/feeds/list/key..) does, though.\n      Calls back to loadSheet in order to get the real work done.\n\n      Used as a callback for the worksheet-based JSON\n    */\n    loadSheets: function(data) {\n      var i, ilen;\n      var toLoad = [];\n      this.googleSheetName = data.feed.title.$t;\n      this.foundSheetNames = [];\n\n      for (i = 0, ilen = data.feed.entry.length; i < ilen ; i++) {\n        this.foundSheetNames.push(data.feed.entry[i].title.$t);\n        // Only pull in desired sheets to reduce loading\n        if (this.isWanted(data.feed.entry[i].content.$t)) {\n          var linkIdx = data.feed.entry[i].link.length-1;\n          var sheetId = data.feed.entry[i].link[linkIdx].href.split('/').pop();\n          var jsonPath = '/feeds/list/' + this.key + '/' + sheetId + '/public/values?alt=';\n          if (inNodeJS || supportsCORS) {\n            jsonPath += 'json';\n          } else {\n            jsonPath += 'json-in-script';\n          }\n          if (this.query) {\n            // Query Language Reference (0.7)\n            jsonPath += '&tq=' + this.query;\n          }\n          if (this.orderby) {\n            jsonPath += '&orderby=column:' + this.orderby.toLowerCase();\n          }\n          if (this.reverse) {\n            jsonPath += '&reverse=true';\n          }\n          toLoad.push(jsonPath);\n        }\n      }\n\n      this.sheetsToLoad = toLoad.length;\n      for(i = 0, ilen = toLoad.length; i < ilen; i++) {\n        this.requestData(toLoad[i], this.loadSheet);\n      }\n    },\n\n    /*\n      Access layer for the this.models\n      .sheets() gets you all of the sheets\n      .sheets('Sheet1') gets you the sheet named Sheet1\n    */\n    sheets: function(sheetName) {\n      if (typeof sheetName === 'undefined') {\n        return this.models;\n      } else {\n        if (typeof(this.models[sheetName]) === 'undefined') {\n          // alert( \"Can't find \" + sheetName );\n          return;\n        } else {\n          return this.models[sheetName];\n        }\n      }\n    },\n\n    sheetReady: function(model) {\n      this.models[model.name] = model;\n      if (ttIndexOf(this.modelNames, model.name) === -1) {\n        this.modelNames.push(model.name);\n      }\n      \n      this.sheetsToLoad--;\n      if (this.sheetsToLoad === 0) {\n        this.doCallback();\n      }\n    },\n    \n    /*\n      Parse a single list-based worksheet, turning it into a Tabletop Model\n\n      Used as a callback for the list-based JSON\n    */\n    loadSheet: function(data) {\n      var that = this;\n      new Tabletop.Model({ \n        data: data, \n        parseNumbers: this.parseNumbers,\n        postProcess: this.postProcess,\n        tabletop: this,\n        prettyColumnNames: this.prettyColumnNames,\n        onReady: function() {\n          that.sheetReady(this);\n        } \n      });\n    },\n\n    /*\n      Execute the callback upon loading! Rely on this.data() because you might\n        only request certain pieces of data (i.e. simpleSheet mode)\n      Tests this.sheetsToLoad just in case a race condition happens to show up\n    */\n    doCallback: function() {\n      if(this.sheetsToLoad === 0) {\n        this.callback.apply(this.callbackContext || this, [this.data(), this]);\n      }\n    },\n\n    log: function() {\n      if(this.debug) {\n        if(typeof console !== 'undefined' && typeof console.log !== 'undefined') {\n          Function.prototype.apply.apply(console.log, [console, arguments]);\n        }\n      }\n    }\n\n  };\n\n  /*\n    Tabletop.Model stores the attribute names and parses the worksheet data\n      to turn it into something worthwhile\n\n    Options should be in the format { data: XXX }, with XXX being the list-based worksheet\n  */\n  Tabletop.Model = function(options) {\n    var i, j, ilen, jlen;\n    this.columnNames = [];\n    this.column_names = this.columnNames; // jshint ignore:line\n    this.name = options.data.feed.title.$t;\n    this.tabletop = options.tabletop;\n    this.elements = [];\n    this.onReady = options.onReady;\n    this.raw = options.data; // A copy of the sheet's raw data, for accessing minutiae\n\n    if (typeof(options.data.feed.entry) === 'undefined') {\n      options.tabletop.log('Missing data for ' + this.name + ', make sure you didn\\'t forget column headers');\n      this.originalColumns = [];\n      this.elements = [];\n      this.onReady.call(this);\n      return;\n    }\n    \n    for (var key in options.data.feed.entry[0]){\n      if (/^gsx/.test(key)) {\n        this.columnNames.push(key.replace('gsx$',''));\n      }\n    }\n\n    this.originalColumns = this.columnNames;\n    this.original_columns = this.originalColumns; // jshint ignore:line\n    \n    for (i = 0, ilen =  options.data.feed.entry.length ; i < ilen; i++) {\n      var source = options.data.feed.entry[i];\n      var element = {};\n      for (j = 0, jlen = this.columnNames.length; j < jlen ; j++) {\n        var cell = source['gsx$' + this.columnNames[j]];\n        if (typeof(cell) !== 'undefined') {\n          if (options.parseNumbers && cell.$t !== '' && !isNaN(cell.$t)) {\n            element[this.columnNames[j]] = +cell.$t;\n          } else {\n            element[this.columnNames[j]] = cell.$t;\n          }\n        } else {\n          element[this.columnNames[j]] = '';\n        }\n      }\n      if (element.rowNumber === undefined) {\n        element.rowNumber = i + 1;\n      }\n        \n      if (options.postProcess) {\n        options.postProcess(element);\n      }\n        \n      this.elements.push(element);\n    }\n    \n    if (options.prettyColumnNames) {\n      this.fetchPrettyColumns();\n    } else {\n      this.onReady.call(this);\n    }\n  };\n\n  Tabletop.Model.prototype = {\n    /*\n      Returns all of the elements (rows) of the worksheet as objects\n    */\n    all: function() {\n      return this.elements;\n    },\n    \n    fetchPrettyColumns: function() {\n      if (!this.raw.feed.link[3]) {\n        return this.ready();\n      }\n        \n      var cellurl = this.raw.feed.link[3].href.replace('/feeds/list/', '/feeds/cells/').replace('https://spreadsheets.google.com', '');\n      var that = this;\n      this.tabletop.requestData(cellurl, function(data) {\n        that.loadPrettyColumns(data);\n      });\n    },\n    \n    ready: function() {\n      this.onReady.call(this);\n    },\n    \n    /*\n     * Store column names as an object\n     * with keys of Google-formatted \"columnName\"\n     * and values of human-readable \"Column name\"\n     */\n    loadPrettyColumns: function(data) {\n      var prettyColumns = {};\n\n      var columnNames = this.columnNames;\n\n      var i = 0;\n      var l = columnNames.length;\n\n      for (; i < l; i++) {\n        if (typeof data.feed.entry[i].content.$t !== 'undefined') {\n          prettyColumns[columnNames[i]] = data.feed.entry[i].content.$t;\n        } else {\n          prettyColumns[columnNames[i]] = columnNames[i];\n        }\n      }\n\n      this.prettyColumns = prettyColumns;\n      this.pretty_columns = this.prettyColumns; // jshint ignore:line\n      this.prettifyElements();\n      this.ready();\n    },\n    \n    /*\n     * Go through each row, substitutiting\n     * Google-formatted \"columnName\"\n     * with human-readable \"Column name\"\n     */\n    prettifyElements: function() {\n      var prettyElements = [],\n          orderedPrettyNames = [],\n          i, j, ilen, jlen;\n\n      for (j = 0, jlen = this.columnNames.length; j < jlen ; j++) {\n        orderedPrettyNames.push(this.prettyColumns[this.columnNames[j]]);\n      }\n\n      for (i = 0, ilen = this.elements.length; i < ilen; i++) {\n        var newElement = {};\n        for (j = 0, jlen = this.columnNames.length; j < jlen ; j++) {\n          var newColumnName = this.prettyColumns[this.columnNames[j]];\n          newElement[newColumnName] = this.elements[i][this.columnNames[j]];\n        }\n        prettyElements.push(newElement);\n      }\n      this.elements = prettyElements;\n      this.columnNames = orderedPrettyNames;\n    },\n\n    /*\n      Return the elements as an array of arrays, instead of an array of objects\n    */\n    toArray: function() {\n      var array = [],\n          i, j, ilen, jlen;\n      for (i = 0, ilen = this.elements.length; i < ilen; i++) {\n        var row = [];\n        for (j = 0, jlen = this.columnNames.length; j < jlen ; j++) {\n          row.push(this.elements[i][ this.columnNames[j]]);\n        }\n        array.push(row);\n      }\n      \n      return array;\n    }\n  };\n\n  if(typeof module !== 'undefined' && module.exports) { //don't just use inNodeJS, we may be in Browserify\n    module.exports = Tabletop;\n  } else if (typeof define === 'function' && define.amd) {\n    define(function () {\n      return Tabletop;\n    });\n  } else {\n    window.Tabletop = Tabletop;\n  }\n\n})();","// shim for using process in browser\nvar process = module.exports = {};\n\n// cached from whatever global is present so that test runners that stub it\n// don't break things.  But we need to wrap it in a try catch in case it is\n// wrapped in strict mode code which doesn't define any globals.  It's inside a\n// function because try/catches deoptimize in certain engines.\n\nvar cachedSetTimeout;\nvar cachedClearTimeout;\n\nfunction defaultSetTimout() {\n    throw new Error('setTimeout has not been defined');\n}\nfunction defaultClearTimeout () {\n    throw new Error('clearTimeout has not been defined');\n}\n(function () {\n    try {\n        if (typeof setTimeout === 'function') {\n            cachedSetTimeout = setTimeout;\n        } else {\n            cachedSetTimeout = defaultSetTimout;\n        }\n    } catch (e) {\n        cachedSetTimeout = defaultSetTimout;\n    }\n    try {\n        if (typeof clearTimeout === 'function') {\n            cachedClearTimeout = clearTimeout;\n        } else {\n            cachedClearTimeout = defaultClearTimeout;\n        }\n    } catch (e) {\n        cachedClearTimeout = defaultClearTimeout;\n    }\n} ())\nfunction runTimeout(fun) {\n    if (cachedSetTimeout === setTimeout) {\n        //normal enviroments in sane situations\n        return setTimeout(fun, 0);\n    }\n    // if setTimeout wasn't available but was latter defined\n    if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {\n        cachedSetTimeout = setTimeout;\n        return setTimeout(fun, 0);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedSetTimeout(fun, 0);\n    } catch(e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally\n            return cachedSetTimeout.call(null, fun, 0);\n        } catch(e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error\n            return cachedSetTimeout.call(this, fun, 0);\n        }\n    }\n\n\n}\nfunction runClearTimeout(marker) {\n    if (cachedClearTimeout === clearTimeout) {\n        //normal enviroments in sane situations\n        return clearTimeout(marker);\n    }\n    // if clearTimeout wasn't available but was latter defined\n    if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {\n        cachedClearTimeout = clearTimeout;\n        return clearTimeout(marker);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedClearTimeout(marker);\n    } catch (e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't  trust the global object when called normally\n            return cachedClearTimeout.call(null, marker);\n        } catch (e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.\n            // Some versions of I.E. have different rules for clearTimeout vs setTimeout\n            return cachedClearTimeout.call(this, marker);\n        }\n    }\n\n\n\n}\nvar queue = [];\nvar draining = false;\nvar currentQueue;\nvar queueIndex = -1;\n\nfunction cleanUpNextTick() {\n    if (!draining || !currentQueue) {\n        return;\n    }\n    draining = false;\n    if (currentQueue.length) {\n        queue = currentQueue.concat(queue);\n    } else {\n        queueIndex = -1;\n    }\n    if (queue.length) {\n        drainQueue();\n    }\n}\n\nfunction drainQueue() {\n    if (draining) {\n        return;\n    }\n    var timeout = runTimeout(cleanUpNextTick);\n    draining = true;\n\n    var len = queue.length;\n    while(len) {\n        currentQueue = queue;\n        queue = [];\n        while (++queueIndex < len) {\n            if (currentQueue) {\n                currentQueue[queueIndex].run();\n            }\n        }\n        queueIndex = -1;\n        len = queue.length;\n    }\n    currentQueue = null;\n    draining = false;\n    runClearTimeout(timeout);\n}\n\nprocess.nextTick = function (fun) {\n    var args = new Array(arguments.length - 1);\n    if (arguments.length > 1) {\n        for (var i = 1; i < arguments.length; i++) {\n            args[i - 1] = arguments[i];\n        }\n    }\n    queue.push(new Item(fun, args));\n    if (queue.length === 1 && !draining) {\n        runTimeout(drainQueue);\n    }\n};\n\n// v8 likes predictible objects\nfunction Item(fun, array) {\n    this.fun = fun;\n    this.array = array;\n}\nItem.prototype.run = function () {\n    this.fun.apply(null, this.array);\n};\nprocess.title = 'browser';\nprocess.browser = true;\nprocess.env = {};\nprocess.argv = [];\nprocess.version = ''; // empty string to avoid regexp issues\nprocess.versions = {};\n\nfunction noop() {}\n\nprocess.on = noop;\nprocess.addListener = noop;\nprocess.once = noop;\nprocess.off = noop;\nprocess.removeListener = noop;\nprocess.removeAllListeners = noop;\nprocess.emit = noop;\n\nprocess.binding = function (name) {\n    throw new Error('process.binding is not supported');\n};\n\nprocess.cwd = function () { return '/' };\nprocess.chdir = function (dir) {\n    throw new Error('process.chdir is not supported');\n};\nprocess.umask = function() { return 0; };\n","(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})"],"names":[],"mappings":"AIAA;ADAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ADpLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AD1lBA,OAAO,CAAC,QAAR,GAAmB,OAAA,CAAQ,UAAR;;;;ADAnB,IAAA;;AAAE,WAAa,OAAA,CAAQ,KAAR;;AAGT,OAAO,CAAC;EACC,eAAC,OAAD;IACX,IAAC,CAAA,IAAD,GAAQ,OAAO,CAAC;IAEhB,IAAC,CAAA,GAAD,GAAO,CAAA,SAAA,KAAA;aAAA,SAAC,QAAD;eACL,QAAQ,CAAC,IAAT,CAAc;UACZ,GAAA,EAAK,KAAC,CAAA,IADM;UAEZ,WAAA,EAAa,IAFD;UAGZ,QAAA,EAAU,SAAC,IAAD,EAAO,KAAP;AACR,mBAAO,QAAA,CAAS,IAAT,EAAe,KAAf;UADC,CAHE;SAAd;MADK;IAAA,CAAA,CAAA,CAAA,IAAA;EAHI"} 822 | -------------------------------------------------------------------------------- /sheetExample.framer/framer/images/cursor-active.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/images/cursor-active.png -------------------------------------------------------------------------------- /sheetExample.framer/framer/images/cursor-active@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/images/cursor-active@2x.png -------------------------------------------------------------------------------- /sheetExample.framer/framer/images/cursor.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/images/cursor.png -------------------------------------------------------------------------------- /sheetExample.framer/framer/images/cursor@2x.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/images/cursor@2x.png -------------------------------------------------------------------------------- /sheetExample.framer/framer/images/icon-120.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/images/icon-120.png -------------------------------------------------------------------------------- /sheetExample.framer/framer/images/icon-152.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/images/icon-152.png -------------------------------------------------------------------------------- /sheetExample.framer/framer/images/icon-180.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/images/icon-180.png -------------------------------------------------------------------------------- /sheetExample.framer/framer/images/icon-192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/images/icon-192.png -------------------------------------------------------------------------------- /sheetExample.framer/framer/images/icon-76.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/framer/images/icon-76.png -------------------------------------------------------------------------------- /sheetExample.framer/framer/style.css: -------------------------------------------------------------------------------- 1 | * { 2 | margin: 0; 3 | padding: 0; 4 | border: none; 5 | -webkit-user-select: none; 6 | -webkit-tap-highlight-color: rgba(0,0,0,0); 7 | } 8 | 9 | body { 10 | background-color: #fff; 11 | font: 28px/1em "Helvetica"; 12 | color: gray; 13 | overflow: hidden; 14 | } 15 | 16 | a { 17 | color: gray; 18 | } 19 | 20 | body { 21 | cursor: url('images/cursor.png') 32 32, auto; 22 | cursor: -webkit-image-set( 23 | url('images/cursor.png') 1x, 24 | url('images/cursor@2x.png') 2x 25 | ) 32 32, auto; 26 | } 27 | 28 | body:active { 29 | cursor: url('images/cursor-active.png') 32 32, auto; 30 | cursor: -webkit-image-set( 31 | url('images/cursor-active.png') 1x, 32 | url('images/cursor-active@2x.png') 2x 33 | ) 32 32, auto; 34 | } 35 | 36 | .framerAlertBackground { 37 | position: absolute; top:0px; left:0px; right:0px; bottom:0px; 38 | z-index: 1000; 39 | background-color: #fff; 40 | } 41 | 42 | .framerAlert { 43 | font:400 14px/1.4 "Helvetica Neue", Helvetica, Arial, sans-serif; 44 | -webkit-font-smoothing:antialiased; 45 | color:#616367; text-align:center; 46 | position: absolute; top:40%; left:50%; width:260px; margin-left:-130px; 47 | } 48 | .framerAlert strong { font-weight:500; color:#000; margin-bottom:8px; display:block; } 49 | .framerAlert a { color:#28AFFA; } 50 | .framerAlert .btn { 51 | font-weight:500; text-decoration:none; line-height:1; 52 | display:inline-block; padding:6px 12px 7px 12px; 53 | border-radius:3px; margin-top:12px; 54 | background:#28AFFA; color:#fff; 55 | } 56 | 57 | ::-webkit-scrollbar { 58 | display: none; 59 | } -------------------------------------------------------------------------------- /sheetExample.framer/framer/version: -------------------------------------------------------------------------------- 1 | 5 -------------------------------------------------------------------------------- /sheetExample.framer/images/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/andrewliebchen/framer-sheet/973d519c5c6c74c143bf1175fee469c79c06777d/sheetExample.framer/images/.gitkeep -------------------------------------------------------------------------------- /sheetExample.framer/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | -------------------------------------------------------------------------------- /sheetExample.framer/modules/npm.coffee: -------------------------------------------------------------------------------- 1 | exports.Tabletop = require 'tabletop' 2 | -------------------------------------------------------------------------------- /sheetExample.framer/modules/sheet.coffee: -------------------------------------------------------------------------------- 1 | { Tabletop } = require 'npm' 2 | 3 | 4 | class exports.Sheet 5 | constructor: (options) -> 6 | @_key = options.key 7 | 8 | @get = (callback) => 9 | Tabletop.init { 10 | key: @_key 11 | simpleSheet: true 12 | callback: (data, sheet) => 13 | return callback(data, sheet) 14 | } 15 | --------------------------------------------------------------------------------