mirror of
				https://github.com/flynx/ImageGrid.git
				synced 2025-10-31 19:30:07 +00:00 
			
		
		
		
	some refactoring...
Signed-off-by: Alex A. Naanou <alex.nanou@gmail.com>
This commit is contained in:
		
							parent
							
								
									439001f654
								
							
						
					
					
						commit
						4c64a041e3
					
				
							
								
								
									
										515
									
								
								ui/ImageGrid.data.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										515
									
								
								ui/ImageGrid.data.js
									
									
									
									
									
										Executable file
									
								
							| @ -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 :                                                */ | ||||
							
								
								
									
										386
									
								
								ui/ImageGrid.js
									
									
									
									
									
								
							
							
						
						
									
										386
									
								
								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 | ||||
| */ | ||||
|  | ||||
							
								
								
									
										112
									
								
								ui/index.html
									
									
									
									
									
								
							
							
						
						
									
										112
									
								
								ui/index.html
									
									
									
									
									
								
							| @ -224,6 +224,7 @@ | ||||
| <script src="lib/keyboard.js"></script> | ||||
| 
 | ||||
| <script src="ImageGrid.js"></script> | ||||
| <script src="ImageGrid.data.js"></script> | ||||
| <script src="keybindings3.js"></script> | ||||
| 
 | ||||
| <script> | ||||
| @ -243,116 +244,7 @@ $(function(){ | ||||
| 			KEYBOARD_CONFIG, | ||||
| 			function(k){console.log(k)})) | ||||
| 
 | ||||
| 	// dynamic loading... | ||||
| 	DYNAMIC_LOADING = true | ||||
| 
 | ||||
| 	if(DYNAMIC_LOADING){ | ||||
| 		// XXX move to a setup function in the lib... | ||||
| 		$('.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)) | ||||
| 			}) | ||||
| 	} | ||||
| 
 | ||||
| 
 | ||||
| 	setupDataBindings() | ||||
| 
 | ||||
| 	// XXX stub... | ||||
| 	centerView(focusImage($('.image').first()), 'css') | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user