diff --git a/ui/ImageGrid.data.js b/ui/ImageGrid.data.js
new file mode 100755
index 00000000..86080400
--- /dev/null
+++ b/ui/ImageGrid.data.js
@@ -0,0 +1,515 @@
+/**********************************************************************
+*
+*
+*
+**********************************************************************/
+
+//var DEBUG = DEBUG != null ? DEBUG : true
+//
+// XXX STUB
+// Data format...
+var DATA = {
+ current: 0,
+ // the ribbon cache...
+ // in the simplest form this is a list of lists of GIDs
+ ribbons: [
+ $(new Array(100)).map(function(i){return i}).toArray()
+ ],
+ // flat ordered list of images in current context...
+ // in the simplest form this is a list of GIDs.
+ order: $(new Array(100)).map(function(i){return i}).toArray(),
+ // the images object, this is indexed by image GID and contains all
+ // the needed data...
+ images: {
+ // sub image, for testing load mechanics...
+ SIZE: {
+ id: 'SIZE',
+ ctime: 0,
+ path: './images/sizes/900px/SIZE.jpg',
+ preview: {
+ '150px': './images/sizes/150px/SIZE.jpg',
+ '350px': './images/sizes/350px/SIZE.jpg',
+ '900px': './images/sizes/900px/SIZE.jpg',
+ },
+ classes: '',
+ },
+ }
+}
+
+
+
+/**********************************************************************
+* Helpers
+*/
+
+// A predicate returning:
+// - 0 if a is equal at position i in lst or is between i and i+1
+// - -1 if a is "below" position i
+// - +1 if a is "above" position i
+//
+// NOTE: this is here mostly to make debuging easy...
+function isBetween(a, i, lst){
+ console.log('>>>', a, i, lst)
+ var b = lst[i]
+ var c = lst[i+1]
+ // hit...
+ if(a == b || (a > b && a < c)){
+ return 0
+ // before...
+ } else if(a < b){
+ return -1
+ // later...
+ } else {
+ return 1
+ }
+}
+
+
+// Basic liner search...
+function linSearch(target, lst, check, return_position, disable_direct_indexing){
+ // XXX is this the correct default?
+ check = check == null ? isBetween : check
+ // special case: target in the list directly...
+ if(disable_direct_indexing
+ && check(target, lst.indexOf(target), lst) == 0){
+ return target
+ }
+ // special case: tail...
+ if(check(target, lst.length-1, lst) >= 0){
+ return lst[lst.length-1]
+ }
+
+ for(var i=0; i < lst.length; i++){
+ if(check(target, i, lst) == 0){
+ return return_position ? i : lst[i]
+ }
+ }
+
+ // no hit...
+ return return_position ? -1 : null
+}
+
+
+// Basic binary search implementation...
+//
+// NOTE: this will return the object by default, to return position set
+// return_position to true.
+// NOTE: by default this will use isBetween as a predicate.
+// NOTE: this still depends on .indexOf(...), to disable set
+// disable_direct_indexing to true
+// XXX BUGGY
+// XXX this is a mess, needs revision...
+function binSearch(target, lst, check, return_position, disable_direct_indexing){
+ // XXX is this the correct default?
+ check = check == null ? isBetween : check
+ // special case: target in the list directly...
+ if(disable_direct_indexing
+ && check(target, lst.indexOf(target), lst) == 0){
+ return target
+ }
+ // special case: tail...
+ if(check(target, lst.length-1, lst) >= 0){
+ return lst[lst.length-1]
+ }
+ // special case: head...
+ var res = check(target, 0, lst)
+ if(res == 0){
+ return lst[0]
+ } else if(res < 0){
+ // no hit...
+ return return_position ? -1 : null
+ }
+
+ var l = Math.ceil(lst.length/2)
+ var i = l
+
+ while(l > 0){
+ // XXX this is a hack -- should we reach 0 using floor(..) instead?
+ l = l <= 1 ? 0 : Math.ceil(l/2)
+ res = check(target, i, lst)
+ // right branch...
+ if(res > 0){
+ i += l
+ // left branch...
+ } else if(res < 0){
+ i -= l
+ // hit...
+ } else {
+ return return_position ? i : lst[i]
+ }
+ }
+ // no hit...
+ return return_position ? -1 : null
+}
+
+
+// Same as getImageBefore, but uses gids and searches in DATA...
+function getGIDBefore(gid, ribbon, search){
+ search = search == null ? linSearch : search
+ ribbon = DATA.ribbons[ribbon]
+ var order = DATA.order
+
+ var target = order.indexOf(gid)
+
+ return search(target, ribbon, function (a, i, lst){
+ var b = order.indexOf(lst[i])
+ var c = order.indexOf(lst[i+1])
+ // hit...
+ if(a == b || (a > b && a < c)){
+ return 0
+ // before...
+ } else if(a < b){
+ return -1
+ // later...
+ } else {
+ return 1
+ }
+ })
+}
+
+
+
+/**********************************************************************
+* Loaders
+*/
+
+// NOTE: count can be either negative or positive, this will indicate
+// load direction...
+// NOTE: this will not include the 'from' GID in the resulting list...
+// NOTE: this can calculate the ribbon number if an image can be only
+// in one ribbon...
+// NOTE: if an image can be in more than one ribbon, one MUST suply the
+// correct ribbon number...
+// XXX do we need more checking???
+// XXX inclusive can not be false, only null or true...
+function getImageGIDs(from, count, ribbon, inclusive){
+ if(count == 0){
+ return []
+ }
+ // ribbon default value...
+ if(ribbon == null){
+ $(DATA.ribbons).each(function(i, e){
+ if(e.indexOf(from) >= 0){
+ ribbon = i
+ return false
+ }
+ })
+ }
+ // XXX check if this is empty...
+ ribbon = DATA.ribbons[ribbon]
+
+
+ if(count > 0){
+ var c = inclusive == null ? 1 : 0
+ var start = ribbon.indexOf(from) + c
+ return ribbon.slice(start, start + count)
+ } else {
+ // XXX
+ var c = inclusive == null ? 0 : 1
+ var end = ribbon.indexOf(from)
+ return ribbon.slice((Math.abs(count) >= end ? 0 : end + count + c), end + c)
+ }
+}
+
+
+function updateImage(image, gid, size){
+ image = $(image)
+ if(gid == null){
+ gid = JSON.parse(image.attr('gid'))
+ } else {
+ image.attr('gid', JSON.stringify(gid))
+ }
+ size = size == null ? getVisibleImageSize() : size
+
+ // update classes and other indicators...
+ image
+ .attr({
+ //order: JSON.stringify(DATA.order.indexOf(gid)),
+ order: JSON.stringify(gid)
+ // XXX update other attrs...
+ })
+
+ // XXX STUB
+ image.text(gid)
+
+ // XXX STUB, use real image GID...
+ gid = 'SIZE'
+
+ var img_data = DATA.images[gid]
+
+ // select best preview by size...
+ var url, s
+ for(var k in img_data.preview){
+ s = parseInt(k)
+ if(s > size){
+ url = 'url('+ img_data.preview[k] +')'
+ break
+ }
+ }
+ // if no preview found use the original...
+ if(url == null){
+ url = 'url('+DATA.images[gid].path+')'
+ }
+ image.css({
+ 'background-image': url,
+ })
+
+
+
+ // XXX STUB
+ //image.text(image.text() + ' ('+ s +'px)')
+
+}
+
+// shorthand...
+function updateImages(size){
+ size = size == null ? getVisibleImageSize() : size
+ return $('.image').each(function(){
+ updateImage($(this), null, size)
+ })
+}
+
+
+// Load count images around a given image/gid into the given ribbon.
+//
+// NOTE: this will reload the current image elements...
+// NOTE: this is similar to extendRibbon(...) but different in interface...
+function loadImages(ref_gid, count, ribbon){
+ ribbon = $(ribbon)
+ var images = ribbon.find('.image')
+ var ribbon_i = getRibbonIndex(ribbon)
+ var gid = getGIDBefore(ref_gid, ribbon_i)
+ gid = gid == null ? DATA.ribbons[ribbon_i][0] : gid
+
+ // start/end points...
+ var l = DATA.ribbons[ribbon_i].length
+ if(l <= count){
+ var from_i = 0
+ } else {
+ var from_i = DATA.ribbons[ribbon_i].indexOf(gid) - Math.floor(count/2)
+ }
+ // special case: head...
+ from_i = from_i < 0 ? 0 : from_i
+ // special case: tail...
+ from_i = l - from_i < count ? l - count : from_i
+ var from_gid = DATA.ribbons[ribbon_i][from_i]
+
+ // XXX load only what is needed instead of reloading everything...
+ // XXX
+
+ var size = getVisibleImageSize()
+ var gids = getImageGIDs(from_gid, count, ribbon_i, true)
+
+ //console.log('>>>', ribbon_i, gids)
+
+ // do nothing...
+ // XXX this is still wrong, need to check what's loaded...
+ if(count > gids.length){
+ return images
+
+ } else if(count != images.length){
+ var l = images.length
+ var ext = count - l
+ var ext_l = Math.floor(ext/2)
+ var ext_r = ext - ext_l
+ // NOTE: this avoids reattaching images that are already there...
+ extendRibbon(ext_l, ext_r, ribbon)
+ images = ribbon.find('.image')
+ }
+
+ return images.each(function(i, e){
+ updateImage(e, gids[i], size)
+ })
+}
+
+// XXX here for testing...
+function loadImagesAround(ref_gid, count, ribbon){
+ var ribbon_i = getRibbonIndex(ribbon)
+ var gid = getGIDBefore(ref_gid, ribbon_i)
+ return loadImages(ref_gid, count, ribbon).filter('[gid='+JSON.stringify(gid)+']').click()
+}
+
+
+var LOAD_SCREENS = 2
+var LOAD_THRESHOLD = 1
+var DEFAULT_SCREEN_IMAGES = 5
+
+function loadData(data, images_per_screen){
+ var ribbons_set = $('.ribbon-set')
+ var current = data.current
+ // if no width is given, use the current or default...
+ var w = images_per_screen == null ? getScreenWidthInImages() : images_per_screen
+ w = w > MAX_SCREEN_IMAGES ? DEFAULT_SCREEN_IMAGES : w
+
+ // clear data...
+ $('.ribbon').remove()
+
+ // create ribbons...
+ $.each(data.ribbons, function(i, e){
+ createRibbon().appendTo(ribbons_set)
+ })
+
+ // create images...
+ $('.ribbon').each(function(i, e){
+ loadImages(current, Math.min(w * LOAD_SCREENS * 1.5, data.ribbons[i].length), $(this))
+ })
+
+ focusImage($('.image').filter('[gid='+JSON.stringify(current)+']'))
+
+ fitNImages(w)
+ centerRibbons('css')
+}
+
+
+// NOTE: this is signature-compatible with rollRibbon...
+// NOTE: this will load data ONLY if it is available, otherwise this
+// will have no effect...
+// NOTE: this can roll past the currently loaded images (n > images.length)
+function rollImages(n, ribbon, extend){
+ if(n == 0){
+ return $([])
+ }
+ ribbon = ribbon == null ? getRibbon() : $(ribbon)
+ var images = ribbon.find('.image')
+
+ var from = n > 0 ? JSON.parse(ribbon.find('.image').last().attr('gid'))
+ : JSON.parse(ribbon.find('.image').first().attr('gid'))
+ var gids = getImageGIDs(from, n)
+ if(gids.length == 0){
+ return $([])
+ }
+ // truncate the results to the length of images...
+ if(n > images.length){
+ gids.reverse().splice(images.length)
+ gids.reverse()
+ } else if(Math.abs(n) > images.length){
+ gids.splice(images.length)
+ }
+
+ if(n < images.length){
+ images = rollRibbon(gids.length * (n > 0 ? 1 : -1), ribbon)
+ }
+
+ var size = getVisibleImageSize()
+ images.each(function(i, e){
+ updateImage($(e), gids[i], size)
+ })
+
+ return images
+}
+
+
+
+/**********************************************************************
+* Setup
+*/
+
+function setupDataBindings(){
+ $('.viewer')
+ // XXX this always reloads everything...
+ // XXX this causes miss-aligns after shifting and/or zooming...
+ .on('preCenteringRibbon', function(evt, ribbon, image){
+ // NOTE: we do not need to worry about centering the ribbon
+ // here, just ball-park-load the correct batch...
+
+ // check if we are in the right range...
+ var gid = getImageGID(image)
+ var r = getRibbonIndex(ribbon)
+ var gr = DATA.ribbons[r]
+ var img_before = getImageBefore(image, ribbon)
+ var gid_before = getGIDBefore(gid, r)
+
+ // load...
+ if(gid_before == null || gid_before != getImageGID(img_before)){
+ loadImages(gid, Math.round((LOAD_SCREENS * 1.5) * getScreenWidthInImages()), ribbon)
+ // XXX compensate for the changing number of images...
+ // XXX
+ }
+ })
+ /*
+ // XXX BUGGY...
+ .on('centeringRibbon', function(evt, ribbon, image){
+ // check if we are in the right range...
+ var gid = getImageGID(image)
+ var r = getRibbonIndex(ribbon)
+ var img_before = getImageBefore(image, ribbon)
+ var gid_before = getGIDBefore(gid, r)
+
+ if(img_before.length == 0){
+ img_before = ribbon.find('.image').first()
+ }
+
+ var head = img_before.prevAll('.image')
+ var tail = img_before.nextAll('.image')
+
+ // get the frame size to load...
+ var screen_size = getScreenWidthInImages()
+ // NOTE: if this is greater than the number of images currently
+ // loaded, it might lead to odd effects...
+ // XXX need to load additional images and keep track of the
+ // loaded chunk size...
+ //var frame_size = screen_size * LOAD_SCREENS
+ var frame_size = 4
+ //var threshold = screen_size * LOAD_THRESHOLD
+ var threshold = 2
+
+ // do the loading...
+ // XXX need to expand/contract the ribbon depending on zoom and speed...
+ // XXX use extendRibbon, to both roll and expand/contract...
+ if(tail.length < threshold){
+ var rolled = rollImages(frame_size, ribbon)
+ }
+ if(head.length < threshold){
+ var rolled = rollImages(-frame_size, ribbon)
+ }
+ })
+ */
+ .on('shiftedImage', function(evt, image, from, to){
+ from = getRibbonIndex(from)
+ var ribbon = to
+ to = getRibbonIndex(to)
+
+ var gid = getImageGID(image)
+
+ var index = DATA.ribbons[from].indexOf(gid)
+ var img = DATA.ribbons[from].splice(index, 1)
+
+ // XXX a bit ugly, revise...
+ index = ribbon.find('.image')
+ .index(ribbon.find('[gid='+JSON.stringify(gid)+']'))
+ DATA.ribbons[to].splice(index, 0, gid)
+ })
+
+ .on('createdRibbon', function(evt, index){
+ index = getRibbonIndex(index)
+ DATA.ribbons.splice(index, 0, [])
+ })
+ .on('removedRibbon', function(evt, index){
+ DATA.ribbons.splice(index, 1)
+ })
+
+ .on('requestedFirstImage', function(evt, ribbon){
+ var r = getRibbonIndex(ribbon)
+ var gr = DATA.ribbons[r]
+ rollImages(-gr.length, ribbon)
+ })
+ .on('requestedLastImage', function(evt, ribbon){
+ var r = getRibbonIndex(ribbon)
+ var gr = DATA.ribbons[r]
+ rollImages(gr.length, ribbon)
+ })
+
+ // XXX do we need to make this less global?
+ .on('fittingImages', function(evt, n){
+ updateImages()
+ })
+
+ .on('focusingImage', function(evt, image){
+ DATA.current = getImageGID($(image))
+ })
+}
+
+
+
+/**********************************************************************
+* vim:set ts=4 sw=4 : */
diff --git a/ui/ImageGrid.js b/ui/ImageGrid.js
index d0684ac8..855055d7 100755
--- a/ui/ImageGrid.js
+++ b/ui/ImageGrid.js
@@ -20,36 +20,6 @@
*
**********************************************************************/
-// XXX STUB
-// Data format...
-var DATA = {
- current: 0,
- // the ribbon cache...
- // in the simplest form this is a list of lists of GIDs
- ribbons: [
- $(new Array(100)).map(function(i){return i}).toArray()
- ],
- // flat ordered list of images in current context...
- // in the simplest form this is a list of GIDs.
- order: $(new Array(100)).map(function(i){return i}).toArray(),
- // the images object, this is indexed by image GID and contains all
- // the needed data...
- images: {
- // sub image, for testing load mechanics...
- SIZE: {
- id: 'SIZE',
- ctime: 0,
- path: './images/sizes/900px/SIZE.jpg',
- preview: {
- '150px': './images/sizes/150px/SIZE.jpg',
- '350px': './images/sizes/350px/SIZE.jpg',
- '900px': './images/sizes/900px/SIZE.jpg',
- },
- classes: '',
- },
- }
-}
-
/**********************************************************************
* Helpers
@@ -166,131 +136,6 @@ function getImageBefore(image, ribbon, mode){
}
-// A predicate returning:
-// - 0 if a is equal at position i in lst or is between i and i+1
-// - -1 if a is "below" position i
-// - +1 if a is "above" position i
-//
-// NOTE: this is here mostly to make debuging easy...
-function isBetween(a, i, lst){
- console.log('>>>', a, i, lst)
- var b = lst[i]
- var c = lst[i+1]
- // hit...
- if(a == b || (a > b && a < c)){
- return 0
- // before...
- } else if(a < b){
- return -1
- // later...
- } else {
- return 1
- }
-}
-
-
-// Basic liner search...
-function linSearch(target, lst, check, return_position, disable_direct_indexing){
- // XXX is this the correct default?
- check = check == null ? isBetween : check
- // special case: target in the list directly...
- if(disable_direct_indexing
- && check(target, lst.indexOf(target), lst) == 0){
- return target
- }
- // special case: tail...
- if(check(target, lst.length-1, lst) >= 0){
- return lst[lst.length-1]
- }
-
- for(var i=0; i < lst.length; i++){
- if(check(target, i, lst) == 0){
- return return_position ? i : lst[i]
- }
- }
-
- // no hit...
- return return_position ? -1 : null
-}
-
-// Basic binary search implementation...
-//
-// NOTE: this will return the object by default, to return position set
-// return_position to true.
-// NOTE: by default this will use isBetween as a predicate.
-// NOTE: this still depends on .indexOf(...), to disable set
-// disable_direct_indexing to true
-// XXX BUGGY
-// XXX this is a mess, needs revision...
-function binSearch(target, lst, check, return_position, disable_direct_indexing){
- // XXX is this the correct default?
- check = check == null ? isBetween : check
- // special case: target in the list directly...
- if(disable_direct_indexing
- && check(target, lst.indexOf(target), lst) == 0){
- return target
- }
- // special case: tail...
- if(check(target, lst.length-1, lst) >= 0){
- return lst[lst.length-1]
- }
- // special case: head...
- var res = check(target, 0, lst)
- if(res == 0){
- return lst[0]
- } else if(res < 0){
- // no hit...
- return return_position ? -1 : null
- }
-
- var l = Math.ceil(lst.length/2)
- var i = l
-
- while(l > 0){
- // XXX this is a hack -- should we reach 0 using floor(..) instead?
- l = l <= 1 ? 0 : Math.ceil(l/2)
- res = check(target, i, lst)
- // right branch...
- if(res > 0){
- i += l
- // left branch...
- } else if(res < 0){
- i -= l
- // hit...
- } else {
- return return_position ? i : lst[i]
- }
- }
- // no hit...
- return return_position ? -1 : null
-}
-
-
-// Same as getImageBefore, but uses gids and searches in DATA...
-function getGIDBefore(gid, ribbon, search){
- search = search == null ? linSearch : search
- ribbon = DATA.ribbons[ribbon]
- var order = DATA.order
-
- var target = order.indexOf(gid)
-
- return search(target, ribbon, function (a, i, lst){
- var b = order.indexOf(lst[i])
- var c = order.indexOf(lst[i+1])
- // hit...
- if(a == b || (a > b && a < c)){
- return 0
- // before...
- } else if(a < b){
- return -1
- // later...
- } else {
- return 1
- }
- })
-}
-
-
function shiftTo(image, ribbon){
var target = getImageBefore(image, ribbon, NAV_ALL)
var cur_ribbon = getRibbon(image)
@@ -505,237 +350,6 @@ function rollRibbon(n, ribbon){
-/**********************************************************************
-* Loaders
-*/
-
-// NOTE: count can be either negative or positive, this will indicate
-// load direction...
-// NOTE: this will not include the 'from' GID in the resulting list...
-// NOTE: this can calculate the ribbon number if an image can be only
-// in one ribbon...
-// NOTE: if an image can be in more than one ribbon, one MUST suply the
-// correct ribbon number...
-// XXX do we need more checking???
-// XXX inclusive can not be false, only null or true...
-function getImageGIDs(from, count, ribbon, inclusive){
- if(count == 0){
- return []
- }
- // ribbon default value...
- if(ribbon == null){
- $(DATA.ribbons).each(function(i, e){
- if(e.indexOf(from) >= 0){
- ribbon = i
- return false
- }
- })
- }
- // XXX check if this is empty...
- ribbon = DATA.ribbons[ribbon]
-
-
- if(count > 0){
- var c = inclusive == null ? 1 : 0
- var start = ribbon.indexOf(from) + c
- return ribbon.slice(start, start + count)
- } else {
- // XXX
- var c = inclusive == null ? 0 : 1
- var end = ribbon.indexOf(from)
- return ribbon.slice((Math.abs(count) >= end ? 0 : end + count + c), end + c)
- }
-}
-
-
-function updateImage(image, gid, size){
- image = $(image)
- if(gid == null){
- gid = JSON.parse(image.attr('gid'))
- } else {
- image.attr('gid', JSON.stringify(gid))
- }
- size = size == null ? getVisibleImageSize() : size
-
- // update classes and other indicators...
- image
- .attr({
- //order: JSON.stringify(DATA.order.indexOf(gid)),
- order: JSON.stringify(gid)
- // XXX update other attrs...
- })
-
- // XXX STUB
- image.text(gid)
-
- // XXX STUB, use real image GID...
- gid = 'SIZE'
-
- var img_data = DATA.images[gid]
-
- // select best preview by size...
- var url, s
- for(var k in img_data.preview){
- s = parseInt(k)
- if(s > size){
- url = 'url('+ img_data.preview[k] +')'
- break
- }
- }
- // if no preview found use the original...
- if(url == null){
- url = 'url('+DATA.images[gid].path+')'
- }
- image.css({
- 'background-image': url,
- })
-
-
-
- // XXX STUB
- //image.text(image.text() + ' ('+ s +'px)')
-
-}
-
-// shorthand...
-function updateImages(size){
- size = size == null ? getVisibleImageSize() : size
- return $('.image').each(function(){
- updateImage($(this), null, size)
- })
-}
-
-
-// Load count images around a given image/gid into the given ribbon.
-//
-// NOTE: this will reload the current image elements...
-// NOTE: this is similar to extendRibbon(...) but different in interface...
-function loadImages(ref_gid, count, ribbon){
- ribbon = $(ribbon)
- var images = ribbon.find('.image')
- var ribbon_i = getRibbonIndex(ribbon)
- var gid = getGIDBefore(ref_gid, ribbon_i)
- gid = gid == null ? DATA.ribbons[ribbon_i][0] : gid
-
- // start/end points...
- var l = DATA.ribbons[ribbon_i].length
- if(l <= count){
- var from_i = 0
- } else {
- var from_i = DATA.ribbons[ribbon_i].indexOf(gid) - Math.floor(count/2)
- }
- // special case: head...
- from_i = from_i < 0 ? 0 : from_i
- // special case: tail...
- from_i = l - from_i < count ? l - count : from_i
- var from_gid = DATA.ribbons[ribbon_i][from_i]
-
- // XXX load only what is needed instead of reloading everything...
- // XXX
-
- var size = getVisibleImageSize()
- var gids = getImageGIDs(from_gid, count, ribbon_i, true)
-
- //console.log('>>>', ribbon_i, gids)
-
- // do nothing...
- // XXX this is still wrong, need to check what's loaded...
- if(count > gids.length){
- return images
-
- } else if(count != images.length){
- var l = images.length
- var ext = count - l
- var ext_l = Math.floor(ext/2)
- var ext_r = ext - ext_l
- // NOTE: this avoids reattaching images that are already there...
- extendRibbon(ext_l, ext_r, ribbon)
- images = ribbon.find('.image')
- }
-
- return images.each(function(i, e){
- updateImage(e, gids[i], size)
- })
-}
-
-// XXX here for testing...
-function loadImagesAround(ref_gid, count, ribbon){
- var ribbon_i = getRibbonIndex(ribbon)
- var gid = getGIDBefore(ref_gid, ribbon_i)
- return loadImages(ref_gid, count, ribbon).filter('[gid='+JSON.stringify(gid)+']').click()
-}
-
-
-var LOAD_SCREENS = 2
-var LOAD_THRESHOLD = 1
-var DEFAULT_SCREEN_IMAGES = 5
-
-function loadData(data, images_per_screen){
- var ribbons_set = $('.ribbon-set')
- var current = data.current
- // if no width is given, use the current or default...
- var w = images_per_screen == null ? getScreenWidthInImages() : images_per_screen
- w = w > MAX_SCREEN_IMAGES ? DEFAULT_SCREEN_IMAGES : w
-
- // clear data...
- $('.ribbon').remove()
-
- // create ribbons...
- $.each(data.ribbons, function(i, e){
- createRibbon().appendTo(ribbons_set)
- })
-
- // create images...
- $('.ribbon').each(function(i, e){
- loadImages(current, Math.min(w * LOAD_SCREENS * 1.5, data.ribbons[i].length), $(this))
- })
-
- focusImage($('.image').filter('[gid='+JSON.stringify(current)+']'))
-
- fitNImages(w)
- centerRibbons('css')
-}
-
-
-// NOTE: this is signature-compatible with rollRibbon...
-// NOTE: this will load data ONLY if it is available, otherwise this
-// will have no effect...
-// NOTE: this can roll past the currently loaded images (n > images.length)
-function rollImages(n, ribbon, extend){
- if(n == 0){
- return $([])
- }
- ribbon = ribbon == null ? getRibbon() : $(ribbon)
- var images = ribbon.find('.image')
-
- var from = n > 0 ? JSON.parse(ribbon.find('.image').last().attr('gid'))
- : JSON.parse(ribbon.find('.image').first().attr('gid'))
- var gids = getImageGIDs(from, n)
- if(gids.length == 0){
- return $([])
- }
- // truncate the results to the length of images...
- if(n > images.length){
- gids.reverse().splice(images.length)
- gids.reverse()
- } else if(Math.abs(n) > images.length){
- gids.splice(images.length)
- }
-
- if(n < images.length){
- images = rollRibbon(gids.length * (n > 0 ? 1 : -1), ribbon)
- }
-
- var size = getVisibleImageSize()
- images.each(function(i, e){
- updateImage($(e), gids[i], size)
- })
-
- return images
-}
-
-
-
/**********************************************************************
* Modes
*/
diff --git a/ui/index.html b/ui/index.html
index fd4b3e0e..02875b4a 100755
--- a/ui/index.html
+++ b/ui/index.html
@@ -224,6 +224,7 @@
+