UNPKG

blossom

Version:

Modern, Cross-Platform Application Framework

1,073 lines (863 loc) 36.2 kB
// ========================================================================== // Project: Blossom - Modern, Cross-Platform Application Framework // Copyright: ©2012 Fohr Motion Picture Studios. All rights reserved. // License: Licensed under the GPLv3 license (see BLOSSOM-LICENSE). // ========================================================================== /*globals sc_assert */ sc_require('surfaces/view'); SC.LOG_ILIST_UPDATES = false; var base03 = "#002b36"; var base02 = "#073642"; var base01 = "#586e75"; var base00 = "#657b83"; var base0 = "#839496"; var base1 = "#93a1a1"; var base2 = "#eee8d5"; var base3 = "#fdf6e3"; var yellow = "#b58900"; var orange = "#cb4b16"; var red = "#dc322f"; var magenta = "#d33682"; var violet = "#6c71c4"; var blue = "#268bd2"; var cyan = "#2aa198"; var green = "#859900"; var white = "white"; /** @class `SC.ListView` implements a scrollable view. You can use it interchangeably with `SC.View`, the only difference is the scrolling. Setting the bounds of the scroll is important. @extends SC.ScrollView @since Blossom 1.0 */ SC.IListView = SC.View.extend({ __tagName__: 'div', __useContentSize__: false, isCompositeSurface: true, // Walk like a duck. subsurfaces: function() { return [this._sc_scrollingSurface]; }.property(), /** true if the view should maintain a horizontal scroller. This property must be set when the view is created. @property {Boolean} */ hasHorizontalScroller: true, /** true if the view should maintain a horizontal scroller. This property must be set when the view is created. @property {Boolean} */ hasVerticalScroller: true, // .......................................................... // PSURFACE SUPPORT (Private) // updatePsurface: function(psurface, surfaces) { // console.log('SC.IListView#updatePsurface()'); sc_assert(this === SC.surfaces[this.__id__], "SC.Surface#updatePsurface() can only be called on active surfaces."); // Sanity check the Psurface. sc_assert(psurface); sc_assert(psurface instanceof SC.Psurface); sc_assert(psurface.__element__); sc_assert(psurface.__element__ === document.getElementById(this.__id__)); var surface = this._sc_scrollingSurface; if (surfaces) surfaces[surface.__id__] = surface; var cur = psurface.push(surface); surface.updatePsurface(cur, surfaces); cur.pop(); }, updateLayout: function() { // console.log('SC.IListView#updateLayout()'); arguments.callee.base.apply(this, arguments); this.adjustLayout(); }, content: null, contentBindingDefault: SC.Binding.multiple(), _sc_hasScrollListener: false, didCreateElement: function(div) { // We don't want SC.View's implementation; don't call it. div.style.overflowX = this.get('hasHorizontalScroller')? 'scroll' : 'hidden'; div.style.overflowY = this.get('hasVerticalScroller')? 'scroll' : 'hidden'; // FIXME: This should be done dynamically, per scrollview. I'm not doing // it now because the CSS has pseudo-selectors, so I have to generate // stylesheet code specially. (Here and a few other places, actually.) // // For now, I'll specially customize the CSS to work with Postbooks' UI // correctly. div.className = 'frame'; // This should probably only be set on mobile Safari/Google Chrome for // Android. // // See http://stackoverflow.com/questions/7763458/ios5-webkit-overflow-scrolling-causes-touch-events-stopping-work // for a fix I haven't yet implemented, too. div.style.webkitOverflowScrolling = 'touch'; // We have to establish this stuff every time we set up our DOM. SC.Event.add(div, 'scroll', this, this._sc_didScroll); this._sc_hasScrollListener = true; this._sc_scrollTopPrev = 0; this._sc_didNotRender = true; this._sc_scrollDelay = 0; }, _sc_scrollTopPrev: 0, _sc_didNotRender: true, _sc_scrollDelay: 0, _sc_rowIndex: 0, _sc_rowLength: 0, _sc_didScroll: function(evt) { // console.log('did scroll'); var now = new Date().getTime(), delay = this._sc_scrollDelay, height = this.get('rowHeight'), canvas = this._sc_context.__sc_canvas__, didNotDraw = true, rect, top, prev, div = SC.psurfaces[this.__id__].__element__; rect = SC.psurfaces[this._sc_scrollingSurface.__id__].__element__.getBoundingClientRect(); top = -rect.top; prev = this._sc_scrollTopPrev; if (this._sc_hasScrollListener) { SC.Event.remove(div, 'scroll', this, this._sc_didScroll); this._sc_hasScrollListener = false; } if (Math.abs(top - prev) < rect.height*0.10) { if (now - delay > 150) { // console.log('drawing'); didNotDraw = false; // var canvasTop = Math.min(Math.floor((top/(height*3)))*(height*3), 300000); var scrollframeHeight = this.get('frame').height, listHeight = this._sc_scrollingSurface.get('frame').height, canvasHeight = this._sc_scrollingSurface._sc_scrollingCanvas.get('frame').height, offset = rect.top - div.getBoundingClientRect().top, rowHeight = this.get('rowHeight'); // 1. var x = -offset + (scrollframeHeight/2) - (canvasHeight/2); // 2. var y = x - (x % rowHeight); // 3. var z = Math.min(y, listHeight - canvasHeight); // 4. offset = z < 0 ? 0 : z; // console.log(x, y, z, offset); // sc_assert(offset % rowHeight === 0); // sc_assert(offset >= 0 && offset <= listHeight - canvasHeight); if (offset === (listHeight - canvasHeight)) { console.log('requesting more records'); this._sc_tellSurrogateToFetchMoreRecords = true; } canvas.style.top = offset + 'px'; this.triggerRendering(); this._sc_rowIndex = offset/rowHeight; } } else { delay = new Date().getTime(); } if (didNotDraw || top !== prev) { // SC.requestAnimationFrame = true; var that = this; SC.RequestAnimationFrame(function() { SC.RunLoop.begin(); that._sc_didScroll(); SC.RunLoop.end(); }); this._sc_scrollTopPrev = top; } else { SC.Event.add(div, 'scroll', this, this._sc_didScroll); this._sc_hasScrollListener = true; } }, // .......................................................... // SELECTION SUPPORT // selection: null, _sc_selection: null, _sc_selectionDidChange: function() { var old = this._sc_selection, cur = this.get('selection'), func = this.triggerRendering; if (old === cur) return; // nothing to do if (old) old.removeObserver('[]', this, func); this._sc_selection = cur; if (cur) cur.addObserver('[]', this, func); this.triggerRendering(); }.observes('selection'), rowHeight: 30, renderRow: function(context, width, height, index, object, isSelected, isLast) { context.fillStyle = 'grey'; context.fillRect(0, 0, width, height); context.font = "12pt Helvetica"; context.fillStyle = 'black'; context.textAlign = 'center'; context.textBaseline = 'middle'; context.fillText(String(index), width/2, height/2); }, renderSurrogate: function(context, width, height, index, object, isSelected, isLast) { console.log('rendering surrogate', object.status); context.fillStyle = 'clear'; context.fillRect(0, 0, width, height); context.strokeStyle = 'grey'; context.lineWidth = 1; context.beginPath(); context.moveTo(0, height - 0.5); context.lineTo(width, height - 0.5); context.stroke(); context.font = "10pt Helvetica"; context.fillStyle = 'black'; context.textAlign = 'center'; context.textBaseline = 'middle'; var status = object.get('status'); if (status === 'FULL') { context.fillText("All "+index+" records are loaded.", width/2, height/2+1); context.fillStyle = 'white'; context.fillText("All "+index+" records are loaded.", width/2, height/2); } else { context.fillText(object.get('status'), width/2, height/2+1); context.fillStyle = 'white'; context.fillText(object.get('status'), width/2, height/2); } }, hasHorizontalScroller: false, clearBackground: true, _sc_plistItems: null, __forceFullRender__: false, updateDisplay: function() { // console.log('SC.IListView#updateDisplay()', SC.guidFor(this)); var benchKey = 'SC.IListView#updateDisplay()'; SC.Benchmark.start(benchKey); var LOG = SC.LOG_ILIST_UPDATES; var ctx = this._sc_context; sc_assert(ctx); sc_assert(document.getElementById(ctx.__sc_canvas__.id)); // FIXME: Clear the background if requested. We don't really want to do this. // if (this.get('clearBackground')) ctx.clearRect(0, 0, ctx.w, ctx.h); // // if (this.willRenderLayers) { // ctx.save(); // this.willRenderLayers(ctx); // ctx.restore(); // } var content = this.get('content'), selection = this.get('selection'), rowIndex = this._sc_rowIndex, len = this._sc_rowLength, w = ctx.w, h = this.get('rowHeight'), plistItems = this._sc_plistItems, newPlistItems = {}, forceFullRender = this.__forceFullRender__; this.__forceFullRender__ = false; // If we have fewer rows (no scrolling), don't try and render too much. len = Math.min(len, (content? content.get('length') : 0) - rowIndex); // console.log(idx, len); var changedStoreKeys = SC.changedStoreKeys; var needsRendering = false; // Create this here so we don't have to make so many. var processLayer = function(layer) { if (layer.__needsRendering__) needsRendering = true; else layer.get('sublayers').forEach(processLayer); }; if (content && len > 0) { for (var idx = 0; idx<len; ++idx) { var obj = content.objectAt(idx + rowIndex), storeKey = obj.storeKey, isSurrogate = obj.isIRecordArray; if (isSurrogate) storeKey = -1; // Won't collide with actual records. sc_assert(storeKey); var plistItem = plistItems[storeKey]; if (plistItem) { delete plistItems[storeKey]; // We've processed it. newPlistItems[storeKey] = plistItem; // We need to update this plist item and determine if it should be // re-rendered. First we check the properties dependent on the // list view itself. if (plistItem.index !== idx + rowIndex) { plistItem.index = idx + rowIndex; if (LOG) console.log(storeKey, 'needs rendering because', 'its index changed'); needsRendering = true; } if (plistItem.offset !== idx*h) { plistItem.offset = idx*h; if (LOG) console.log(storeKey, 'needs rendering because', 'its offset changed'); needsRendering = true; } var isSelected = selection.contains(obj); if (plistItem.isSelected !== isSelected) { plistItem.isSelected = isSelected; if (LOG) console.log(storeKey, 'needs rendering because', 'its isSelected value changed'); needsRendering = true; } var isLast = (idx + rowIndex) === len - 1; if (plistItem.isLast !== isLast) { plistItem.isLast = isSelected; if (LOG) console.log(storeKey, 'needs rendering because', 'its isLast value changed'); needsRendering = true; } if (isSurrogate) { var status = obj.status; if (plistItem.status !== obj.status) { plistItem.status = status; if (LOG) console.log(storeKey, 'needs rendering because', 'its status changed'); needsRendering = true; } if (this._sc_tellSurrogateToFetchMoreRecords) { this._sc_tellSurrogateToFetchMoreRecords = false; obj.fetchRecords(); } // Early exit for speed. if (needsRendering) plistItem.needsRendering = true; needsRendering = false; // Reset break; // We're done examining the surrogate. } // Next we check the storeKeys for changes *if* if (!needsRendering && !plistItem.needsRendering) { if (changedStoreKeys[storeKey]) { if (LOG) console.log(storeKey, 'needs rendering because', 'its storeKey change'); needsRendering = true; } // Okay, check our dependent keys. if (!needsRendering) { var dependentKeys = plistItem.dependentKeys; if (dependentKeys) { for (var dependentKey in dependentKey) { if (!dependentKeys.hasOwnProperty(dependentKey)) continue; if (changedStoreKeys[dependentKey]) { if (LOG) console.log(storeKey, 'needs rendering because', 'a dependent key changed', dependentKey); needsRendering = true; break; } } } } } // Check the layer tree, if it exists if (!needsRendering) { var layerTree = plistItem.editableLayerTree; if (!layerTree) layerTree = plistItem.mouseLayerTree; if (!layerTree) layerTree = plistItem.renderLayerTree; if (layerTree) { if (layerTree.__needsRendering__) { if (LOG) console.log(storeKey, 'needs rendering because', 'its layer tree needs rendering'); needsRendering = true; } else layerTree.get('sublayers').forEach(processLayer); } } if (needsRendering) plistItem.needsRendering = true; // Create a new plistItem } else { plistItem = newPlistItems[storeKey] = new SC.PListItem(idx + rowIndex, obj, idx*h); plistItem.isSelected = selection.contains(obj); plistItem.isLast = (idx + rowIndex) === len - 1; // item is already marked as needing rendering on init if (isSurrogate) { plistItem.isSurrogate = true; plistItem.status = obj.status; } } needsRendering = false; // Reset } } // Handle plist items that are no longer with us. var isInputSurface = SC.app.get('inputSurface') === this; for (var oldStoreKey in plistItems) { if (!plistItems.hasOwnProperty(oldStoreKey)) continue; plistItem = plistItems[oldStoreKey]; var editableLayerTree = plistItem.editableLayerTree; if (editableLayerTree && isInputSurface) SC.CloseFieldEditor(); // Release stuff. delete plistItem.object; delete plistItem.renderLayerTree; delete plistItem.mouseLayerTree; delete plistItem.editableLayerTree; } // Keep for next round. this._sc_plistItems = newPlistItems; if (LOG) console.log(newPlistItems); var clearBackground = this.get('clearBackground'), backgroundColor = this.get('backgroundColor'); // Draw plist items as needed. for (storeKey in newPlistItems) { if (!newPlistItems.hasOwnProperty(storeKey)) continue; plistItem = newPlistItems[storeKey]; if (forceFullRender || plistItem.needsRendering) { if (LOG) console.log('rendering storeKey', storeKey); plistItem.needsRendering = false; ctx.save(); ctx.translate(0, plistItem.offset); if (!plistItem.isSurrogate) { // We render the most complicated option. layerTree = plistItem.editableLayerTree; if (!layerTree) layerTree = plistItem.mouseLayerTree; if (!layerTree) layerTree = plistItem.renderLayerTree; var renderFunction = plistItem.renderFunction; if (!layerTree && !renderFunction) { if (plistItem.isSurrogate) { // We need to find or create the correct one. } else if (this.createRenderLayerTree) { // console.log('creating render tree'); layerTree = plistItem.renderLayerTree = this.createRenderLayerTree(); if (layerTree) { // sc_assert(layerTree); // sc_assert(layerTree.kindOf(SC.Layer)); layerTree.__forceWidthHeight__ = true; layerTree.set('width', w); layerTree.set('height', h); layerTree.__needsLayout__ = true; } } else { // FIXME: Seems like this could be done better... if (this.renderRow) { renderFunction = plistItem.renderFunction = this.renderRow; } } } } else { renderFunction = this.renderSurrogate; } // Render with either the layer tree or the render function. ctx.beginPath(); ctx.rect(0,0,w,h); ctx.clip(); if (layerTree) { // Set the properties for the layer tree. layerTree.set('rowIndex', plistItem.index); layerTree.set('content', plistItem.object); layerTree.set('isSelected', plistItem.isSelected); layerTree.set('isLast', plistItem.isLast); if (layerTree.__needsLayout__) { // Update the layout. var textLayersNeedingLayout = []; layerTree.updateLayout(textLayersNeedingLayout); // Update any text layouts. for (idx=0, len=textLayersNeedingLayout.length; idx<len; ++idx) { textLayersNeedingLayout[idx].updateTextLayout(ctx); } } if (clearBackground) ctx.clearRect(0, 0, w, h); else { // We need to draw the background color, even though it is also // applied with CSS, in order to get correct anti-aliasing. ctx.fillStyle = backgroundColor; ctx.fillRect(0, 0, w, h); } layerTree.renderIntoContext(ctx); } else if (renderFunction) { if (clearBackground) ctx.clearRect(0, 0, w, h); else { // We need to draw the background color, even though it is also // applied with CSS, in order to get correct anti-aliasing. ctx.fillStyle = backgroundColor; ctx.fillRect(0, 0, w, h); } renderFunction(ctx, w, h, plistItem.index, plistItem.object, plistItem.isSelected, plistItem.isLast); } ctx.restore(); } } // if (this.didRenderLayers) { // ctx.save(); // this.didRenderLayers(ctx); // ctx.restore(); // } SC.Benchmark.end(benchKey); }, /** Finds the layer that is hit by this event, and returns its behavior if it exists. Otherwise, returns the receiver. */ targetResponderForEvent: function(evt) { // console.log('ListDemo.LayerListView#targetResponderForEvent(', evt.type, ')'); var context = this._sc_hitTestCanvas.getContext('2d'), hitLayer = null, zIndex = -1, boundingRect, x, y; if (evt.pageX === undefined) return this; // debugger; boundingRect = evt.target.getBoundingClientRect(); x = evt.clientX - boundingRect.left + window.pageXOffset; y = evt.clientY - boundingRect.top + window.pageYOffset; // console.log('*****', evt.type, x, y, '*****'); // FIXME: calculate the correct rowIndex! var rowIndex = Math.floor((evt.pageY - boundingRect.top) / this.get('rowHeight')) + this._sc_rowIndex; sc_assert(!isNaN(rowIndex)); // console.log('rowIndex', rowIndex); function spaces(depth) { console.log(depth); var ret = "", idx, len; for (idx = 0, len = depth; idx<len; ++idx) ret += "--"; return ret; } function hitTestLayer(layer, point, depth) { // debugger; // console.log(spaces(depth), "on entry:", point.x, point.y); if (!layer.get('isVisible')) return; context.save(); // Prevent this layer and any sublayer from drawing paths outside our // bounds. layer.renderBoundsPath(context); context.clip(); // Make sure the layer's transform is current. if (layer._sc_transformFromSuperlayerToLayerIsDirty) { layer._sc_computeTransformFromSuperlayerToLayer(); } // Apply the sublayer's transform from our layer (it's superlayer). var t = layer._sc_transformFromSuperlayerToLayer; context.transform(t[0], t[1], t[2], t[3], t[4], t[5]); SC.PointApplyAffineTransformTo(point, t, point); var frame = layer.get('frame'); // console.log(spaces(depth), 'frame:', frame.x, frame.y, frame.width, frame.height); // console.log(spaces(depth), 'transformed:', point.x, point.y); // First, test our sublayers. var sublayers = layer.get('sublayers'), idx = sublayers.length; depth++; while (idx--) { hitTestLayer(sublayers[idx], SC.MakePoint(point), depth); } // Only test ourself if (a) no hit has been found, or (b) our zIndex is // higher than whatever hit has been found so far. var layerZ = layer.get('zIndex'); if (!hitLayer || zIndex < layerZ) { if (layer.hitsPointInContext(x, y, context)) { evt.hitPoint = SC.MakePoint(x - point[0]/*x*/, y - point[1]/*y*/); hitLayer = layer; zIndex = layerZ; } } context.restore(); } // Next, begin the hit testing process. When this completes, hitLayer // will contain the layer that was hit with the highest zIndex. var plistItems = this._sc_plistItems, rowHeight = this.get('rowHeight'), mouseY = evt.pageY - boundingRect.top, layerTree, offset; for (var storeKey in plistItems) { if (!plistItems.hasOwnProperty(storeKey)) continue; var plistItem = plistItems[storeKey]; offset = plistItem.offset; // this is relative to the canvas if (offset < mouseY && mouseY < (offset + rowHeight)) { layerTree = plistItem.editableLayerTree; if (!layerTree) layerTree = plistItem.mouseLayerTree; if (!layerTree) layerTree = plistItem.renderLayerTree; if (!layerTree) { return this; // row is using a render function } break; } } if (!layerTree) return this; context.save(); context.translate(0, (rowIndex - this._sc_rowIndex)*this.get('rowHeight')); hitTestLayer(layerTree, SC.MakePoint(), 0); context.restore(); // We don't need to test `layer`, because we already know it was hit when // this method is called by SC.RootResponder. if (!hitLayer) return this; else { // this.triggerRendering(); // if (evt.type === 'mousedown') console.log('rowIndex', rowIndex); // If we hit a layer, remember it so our view knows. evt.layer = hitLayer; evt.hitPoint.y = evt.hitPoint.y - offset; hitLayer.set('surface', this._sc_scrollingSurface._sc_scrollingCanvas); var content = this.get('content'); hitLayer.set('content', content.objectAt(rowIndex)); // if (evt.type === 'mousedown') console.log('content.objectAt(rowIndex).get(\'index\')', content.objectAt(rowIndex).get('index')); // Try and find the behavior attached to this layer. var behavior = hitLayer.get('behavior'); while (!behavior && hitLayer) { hitLayer = hitLayer.get('superlayer'); if (hitLayer) behavior = hitLayer.get('behavior'); } return behavior || this; } }, mouseDown: function(evt) { // console.log('SC.ListView#mouseDown()', SC.guidFor(this)); var top = evt.target.getBoundingClientRect().top; this._scrollTop = top; this._scrollTarget = evt.target; this._rowIndex = Math.floor((evt.pageY - top) / this.get('rowHeight')) + this._sc_rowIndex; console.log('this._rowIndex', this._rowIndex); evt.allowDefault(); return true; }, // FIXME: This behavior is only needed on touch devices! mouseUp: function(evt) { var idx = this._rowIndex, content = this._sc_content, selection = this._sc_selection, scrollTarget = this._scrollTarget; this._scrollTarget = null; if (Math.abs(this._scrollTop - scrollTarget.getBoundingClientRect().top) > 15) { return; // We're scrolling... } if (content && selection) { var obj = content.objectAt(idx); if (obj && !selection.contains(obj)) { var sel = SC.SelectionSet.create(); sel.addObject(obj); this.set('selection', sel.freeze()); var action = this.get('action'); if (action && typeof action === 'function') { action.call(this, obj, idx); } // TODO: Support the usual target/action paradigm. } } }, // performLayoutIfNeeded: function(timestamp) { // console.log('SC.IListView#performLayoutIfNeeded()', SC.guidFor(this)); // arguments.callee.base.apply(this, arguments); // }, performRenderingIfNeeded: function(timestamp) { // console.log('SC.IListView#performRenderingIfNeeded()', SC.guidFor(this)); this.__needsRendering__ = true; // We do all our work in updateDisplay() arguments.callee.base.apply(this, arguments); }, adjustLayout: function() { console.log('SC.IListView#adjustLayout()', SC.guidFor(this)); var benchKey = 'SC.IListView#adjustLayout()'; SC.Benchmark.start(benchKey); var frame = SC.MakeRect(this.get('frame')), content = this.get('content'), rowHeight = this.get('rowHeight'); var rows = content? content.get('length') : 0; frame[0]/*x*/ = 0; frame[1]/*y*/ = 0; frame[2]/*w*/ = frame[2]/*w*/ ; // - 15; // account for scroller frame[3]/*h*/ = Math.max(rows*rowHeight, frame[3]/*h*/); // We never have to offset in this manner. var scrollTranslation = this._sc_scrollTranslation; scrollTranslation[0]/*x*/ = 0; scrollTranslation[1]/*y*/ = 0; this._sc_scrollingSurface.set('frame', frame); this._sc_scrollingSurface.__needsLayout__ = true; SC.Benchmark.end(benchKey); }, _sc_content: null, _sc_contentPropertyDidChange: function() { // console.log('SC.ListView#_sc_contentPropertyDidChange()', SC.guidFor(this)); var func = this._sc_contentLengthDidChange, old = this._sc_content, cur = this.get('content'); if (old === cur) return; if (old) { this._sc_removeContentRangeObserver(); old.removeObserver('length', this, func); } this._sc_content = cur; if (cur) { cur.addObserver('length', this, func); this._sc_updateContentRangeObserver(); } this._sc_contentLengthDidChange(); }.observes('content'), _sc_contentLengthDidChange: function() { // console.log('SC.ListView#_sc_contentLengthDidChange()', SC.guidFor(this)); this._sc_updateContentRangeObserver(); this.triggerLayoutAndRendering(); }, _sc_contentRangeObserver: null, _sc_updateContentRangeObserver: function() { // console.log('SC.ListView#_sc_updateContentRangeObserver()', SC.guidFor(this)); var observer = this._sc_contentRangeObserver, content = this.get('content'); if (!content) return ; // nothing to do var nowShowing = SC.IndexSet.create(0, content.get('length')); if (observer) { content.updateRangeObserver(observer, nowShowing); } else { var func = this._sc_contentRangeDidChange; observer = content.addRangeObserver(nowShowing, this, func, null, true); this._sc_contentRangeObserver = observer; } }, _sc_contentRangeDidChange: function() { // console.log('SC.ListView#_sc_contentRangeDidChange()', SC.guidFor(this)); this.triggerRendering(); }, _sc_removeContentRangeObserver: function() { // console.log('SC.ListView#_sc_removeContentRangeObserver()', SC.guidFor(this)); var content = this.get('content'), observer = this._sc_contentRangeObserver ; if (observer) { if (content) content.removeRangeObserver(observer); this._sc_contentRangeObserver = null ; } }, _sc_scrollingSurface: null, _sc_compositeIsPresentInViewportDidChange: function() { // console.log("SC.IListViewSurface#_sc_compositeIsPresentInViewportDidChange()"); var isPresentInViewport = this.get('isPresentInViewport'); this._sc_scrollingSurface.set('isPresentInViewport', isPresentInViewport); }.observes('isPresentInViewport'), init: function() { arguments.callee.base.apply(this, arguments); var scrollingSurface; scrollingSurface = this._sc_scrollingSurface = SC.InternalIListViewSurface.create({ supersurface: this, __scrollView__: this }); this._sc_scrollTranslation = SC.MakePoint(); this._sc_contentPropertyDidChange(); this._sc_selectionDidChange(); this._sc_plistItems = {}; }, rowOffsetForLayerTree: function(layerTree) { console.log("SC.IListViewSurface#rowOffsetForLayerTree()"); var plistItems = this._sc_plistItems; for (var storeKey in plistItems) { if (!plistItems.hasOwnProperty(storeKey)) continue; var plistItem = plistItems[storeKey]; var plistLayerTree = plistItem.editableLayerTree; if (!plistLayerTree) plistLayerTree = plistItem.mouseLayerTree; if (!plistLayerTree) plistLayerTree = plistItem.renderLayerTree; if (plistLayerTree === layerTree) { return plistItem.offset + parseInt(this._sc_context.__sc_canvas__.style.top.slice(0,-2), 10); } } debugger; console.log('Could not find row offset for layer tree. This is a bug.'); return 0; // Don't know! } }); /** @private */ SC.InternalIListViewSurface = SC.LeafSurface.extend({ __tagName__: 'div', isLeafSurface: false, isCompositeSurface: true, // Walk like a duck. subsurfaces: function() { return [this._sc_scrollingCanvas]; }.property(), rowOffsetForLayerTree: function(layerTree) { return this.__scrollView__.rowOffsetForLayerTree(layerTree); }, // .......................................................... // SURFACE TREE SUPPORT // _sc_compositeIsPresentInViewportDidChange: function() { // console.log("SC.InternalIListViewSurface#_sc_compositeIsPresentInViewportDidChange()"); var isPresentInViewport = this.get('isPresentInViewport'); this._sc_scrollingCanvas.set('isPresentInViewport', isPresentInViewport); }.observes('isPresentInViewport'), __scrollView__: null, surface: function() { // console.log('SC.InternalIListViewSurface@surface'); return this.__scrollView__; }.property().cacheable(), didCreateElement: function(div) { // console.log('SC.InternalIListViewSurface#didCreateElement()', SC.guidFor(this)); arguments.callee.base.apply(this, arguments); div.style.overflow = 'hidden'; div.style.backgroundImage = 'url()'; div.style.backgroundPosition = 'right top'; div.style.backgroundRepeat = 'repeat-y'; }, targetResponderForEvent: function(evt) { return this.get('surface').targetResponderForEvent(evt); }, // performLayoutIfNeeded: function(timestamp) { // console.log('SC.InternalIListViewSurface#performLayoutIfNeeded()', SC.guidFor(this)); // arguments.callee.base.apply(this, arguments); // // this._sc_scrollingCanvas.performLayoutIfNeeded(timestamp); // }, // .......................................................... // PSURFACE SUPPORT (Private) // updatePsurface: function(psurface, surfaces) { // console.log('SC.InternalIListViewSurface#updatePsurface()'); sc_assert(this === SC.surfaces[this.__id__], "SC.Surface#updatePsurface() can only be called on active surfaces."); // Sanity check the Psurface. sc_assert(psurface); sc_assert(psurface instanceof SC.Psurface); sc_assert(psurface.__element__); sc_assert(psurface.__element__ === document.getElementById(this.__id__)); var surface = this._sc_scrollingCanvas; if (surfaces) surfaces[surface.__id__] = surface; psurface.push(surface); }, updateLayout: function() { // console.log('SC.InternalIListViewSurface#updateLayout()', SC.guidFor(this)); arguments.callee.base.apply(this, arguments); this.adjustLayout(); }, _sc_scrollingCanvas: null, init: function() { arguments.callee.base.apply(this, arguments); var scrollingCanvas; scrollingCanvas = this._sc_scrollingCanvas = SC.InternalListViewCanvas.create({ supersurface: this, __scrollSurface__: this }); }, adjustLayout: function() { console.log('SC.InternalIListViewSurface#adjustLayout()', SC.guidFor(this)); var frame = SC.MakeRect(this.__scrollView__.get('frame')), myFrame = SC.MakeRect(this.get('frame')), rowHeight = this.__scrollView__.get('rowHeight'); frame[3] = frame[3] * 2; // We need to be even with the rowHeight frame[3] = frame[3] + (rowHeight - (frame[3] % rowHeight)); // Record how many rows we can render. this.__scrollView__._sc_rowLength = frame[3] / rowHeight; frame.x = myFrame.x-12; frame.y = myFrame.y; this._sc_scrollingCanvas.set('frame', frame); } }); /** @private */ SC.InternalListViewCanvas = SC.LeafSurface.extend({ __tagName__: 'canvas', __useContentSize__: true, // we need our width and height attributes set __neverAnimate__: true, __scrollSurface__: null, triggerContentSizeUpdate: function() { arguments.callee.base.apply(this, arguments); this.__scrollSurface__.__scrollView__.__forceFullRender__ = true; }, surface: function() { // console.log('SC.InternalListViewCanvas@surface'); return this.__scrollSurface__; }.property().cacheable(), didCreateElement: function(canvas) { // console.log('SC.InternalListViewCanvas#didCreateElement()', SC.guidFor(this)); arguments.callee.base.apply(this, arguments); var ctx = canvas.getContext('2d'); // Enables ctx.width and ctx.height to work. ctx.__sc_canvas__ = canvas; this.__scrollSurface__.__scrollView__._sc_context = ctx; this.__scrollSurface__.__scrollView__.__forceFullRender__ = true; this.__scrollSurface__.__scrollView__.triggerRendering(); }, rowOffsetForLayerTree: function(layerTree) { return this.__scrollSurface__.rowOffsetForLayerTree(layerTree); }, // performLayoutIfNeeded: function(timestamp) { // console.log('SC.InternalListViewCanvas#performLayoutIfNeeded()', SC.guidFor(this)); // arguments.callee.base.apply(this, arguments); // }, // updateLayout: function() { // console.log('SC.InternalListViewCanvas#updateLayout()', SC.guidFor(this)); // }, targetResponderForEvent: function(evt) { return this.get('surface').targetResponderForEvent(evt); } }); // Constructor SC.PListItem = function(index, object, offset) { // Set all properties for speed with hidden classes. this.index = index; this.object = object; this.storeKey = object.storeKey; this.offset = offset; this.isSelected = false; this.isLast = false; this.needsRendering = true; this.dependentStoreKeys = null; // A PListItem can only have one render function or layer tree, not both. this.renderFunction = null; // Shared with other PListItems this.renderLayerTree = null; // Shared with other PListItems // A PListItem can only have mouse or editable, not both. this.mouseLayerTree = null; // Exclusive to this PListItem this.editableLayerTree = null; // Exclusive to this PListItem this.status = null; this.isSurrogate = false; return this; };