Flexible scrollview and layout-controller for famo.us.
Above anything, famous-flex is a concept in which renderables are seperated from how
they are layed-out. This makes it possible to change layouts on the fly and animate
the renderables from one layout to another. For instance, you can layout a collection
of renderables using a GridLayout
, and change that into a ListLayout
. When using
flow
-mode the renderables will smoothly transition from the old state to the new state using physics, particles and springs.
Install using bower or npm:
bower install famous-flex
npm install famous-flex
LayoutController lays out renderables based on:
- a layout-function
- a data-source containing renderables
- optional layout-options
Example of laying out renderables using a CollectionLayout:
var LayoutController = require('famous-flex/LayoutController');
var CollectionLayout = require('famous-flex/layouts/CollectionLayout'); // import standard layout
// create collection-layout
var layoutController = new LayoutController({
layout: CollectionLayout,
layoutOptions: {
itemSize: [100, 100],
gutter: [20, 20],
justify: true
},
flow: true, // smoothly animates renderables when changing the layout
direction: 1, // 0 = X, 1 = Y, undefined = use default from selected layout-function
dataSource: [
new Surface({content: 'surface1'}),
new Surface({content: 'surface2'}),
new Surface({content: 'surface3'})
]
});
this.add(layoutController); // add layout-controller to the render-tree
When the flow
option is enabled, renderables are animated smoothly between
layout states.
A layout is represented as a Function
, which takes a context
argument and
an optional options
argument. The purpose of the function is to lay-out the
renderables in the data-source by calling context.set()
on a renderable.
When context.set()
is not called on a renderable in the data-source then it is
not added to the render-tree.
Famous-flex comes shipped with various standard layouts, but it is also very easy to create your own layout-functions. View LayoutContext for more details on creating your own layout-functions.
/**
* @param {LayoutContext} context Context used for enumerating renderables and setting the layout
* @param {Object} [options] additional layout-options that are passed to the function
*/
function LayoutFunction(context, options) {
// simple layout-function that lays out renderables from top to bottom
var node = context.next();
var y = 0;
while (node) {
context.set(node, {
size: [context.size[0], 100],
translate: [0, y, 0]
});
y += 100;
node = context.next();
}
};
For optimal performance, the layout function is only executed when:
- A resize occurs
- An option is changed on the layout-controller
- When the content is scrolled
The data-source contains the renderables that are to be layed-out. It can be one of three things:
- An
Array
- A
ViewSequence
- An
Object
with key/value pairs
In case of an Array
or ViewSequence
, use context.next()
in your
layout-function to enumerate all the renderables in the data-source:
var layoutController = new LayoutController({
layout: function (context, options) {
var y = 0;
var node = context.next();
while (node) {
context.set(node, {
size: [context.size[0], 100],
translate: [0, y, 0]
});
y += 100;
node = context.next();
}
},
dataSource: [
new Surface({content: 'surface1'}),
new Surface({content: 'surface2'}),
new Surface({content: 'surface3'})
]
});
Sometimes it is easier to identify renderables by an id, rather than a
sequence. In that case use context.get()
or directly pass the data-source id
to the context.set()
function:
var layoutController = new LayoutController({
layout: function (context, options) {
context.set('one', {
size: [100, 100],
translate: [0, 0, 0]
});
context.set('two', {
size: [100, 100],
translate: [100, 0, 0]
});
context.set('three', {
size: [100, 100],
translate: [200, 0, 0]
});
},
dataSource: {
'one': new Surface({content: 'one'}),
'two': new Surface({content: 'two'}),
'three': new Surface({content: 'three'})
}
});
Layout literals are objects which describe layouts through a definition rather
than a function. The following example describes the use of a layout literal
using dock
semantics:
var layoutController = new LayoutController({
layout: {dock: [
['top', 'header', 50],
['bottom', 'footer', 50],
['fill', 'content']
]},
dataSource: {
header: new Surface({content: 'Header'}),
footer: new Surface({content: 'Footer'}),
content: new Surface({content: 'Content'})
}
});
Layout literals are implemented through LayoutHelpers. To create your own layout literals, perform the following steps:
- Create a LayoutHelper (see LayoutDockHelper for an example).
- Implement the
parse
function on the LayoutHelper. - Register the helper using
LayoutUtility.registerHelper
.
Layout helpers are special classes that simplify writing layout functions.
Helper | Literal | Description |
---|---|---|
LayoutDockHelper | dock |
Layout renderables using docking semantics. |
FlexScrollView is a flexible and highly performant scroll-view for famo.us supporting features such as:
- pull to refresh
- sticky-headers
- smooth flowing when inserting/removing re-sizing
- multi-cell layouts (CollectionLayout)
- margins & spacing
- bottom/right alignment
- all the good stuff you expect from a scrollview and more ;)
It is based on ScrollController which implements the core functionality of the scroll-view, which is turn is inherits from LayoutController.
Take the FlexScrollView Tutorial
var FlexScrollView = require('famous-flex/ScrollView');
//var CollectionLayout = require('famous-flex/layouts/CollectionLayout');
var RefreshLoader = require('famous-refresh-loader/RefreshLoader');
var scrollView = new FlexScrollView({
//layout: CollectionLayout, // uncomment to enable collection-layouts
layoutOptions: {
margins: [10, 20, 10, 20], // margins supported by ListLayout
spacing: [5, 20], // spacing between list-items
isHeaderCallback: function(node) {
return node.isHeader; // see Tutorial for details!
}
},
flow: true, // true enables smooth flowing
direction: 0, // 0: horizontal, 1: vertical
alignment: 0, // set to 1 for bottom/right alignment
useContainer: false, // set to true to auto-embed in a ContainerSurface
mouseMove: true, // allow hold and move using the mouse
autoPipeEvents: true, // automatically call .pipe when renderable is inserted
pullToRefreshHeader: new RefreshLoader() // enable pull to refresh
});
this.add(scrollView);
// add renderables
for (var i = 0; i < 50; i++) {
scrollView.push(new Surface({content: 'my surface'}));
}
Layout | DataSource | Scrollable | Description |
---|---|---|---|
GridLayout | ViewSequence / Array | No | Grid-layout with fixed number of rows & columns. |
ListLayout | ViewSequence / Array | Yes | List layout with margins, spacing and optionally sticky headers. |
CollectionLayout | ViewSequence / Array | Yes | Lays out renderables with a specific width & height. |
HeaderFooterLayout | Id-based | No | Layout containing a top-header, bottom- footer and content. |
NavBarLayout | Id-based | No | Layout containing one or more left and right items and a title. |
Class | Description |
---|---|
LayoutController | Lays out renderables and optionally animates between layout states. |
ScrollController | Scrollable LayoutController (base class for FlexScrollView). |
FlexScrollView | Customizable scroll-view with pull-to-refrsh and scroll-view linking. |
LayoutContext | Context used for writing layout-functions. |
LayoutUtility | Utility class containing helper functions. |
Famous-flex is still in its infancy. I am commited in creating a first-class layout-solution for famo.us that is as performant, pluggable and awesome as can be. But to do this, I need your support and feedback. Let me know which of features below are most important to you, by leaving a comment in the corresponding issue.
- Effects (Apply after-effects on the renderables)
- AutoLayout (Cassowary constraints)
- Drag & drop (Drag & drop renderables in a layout)
If you like this project and want to support it, show some love and give it a star.
- @IjzerenHein
- http://www.gloey.nl
- [email protected]
© 2014 - Hein Rutjes