Commit c3a8df1d authored by Hans Muller's avatar Hans Muller

Merge pull request #122 from HansMuller/shrinking-card

Card Collection dismiss animation
parents 94a4b972 b954e020
...@@ -15,6 +15,7 @@ import 'package:sky/widgets/widget.dart'; ...@@ -15,6 +15,7 @@ import 'package:sky/widgets/widget.dart';
class BlockViewportApp extends App { class BlockViewportApp extends App {
BlockViewportLayoutState layoutState = new BlockViewportLayoutState();
List<double> lengths = <double>[]; List<double> lengths = <double>[];
double offset = 0.0; double offset = 0.0;
...@@ -96,7 +97,8 @@ class BlockViewportApp extends App { ...@@ -96,7 +97,8 @@ class BlockViewportApp extends App {
child: new BlockViewport( child: new BlockViewport(
builder: builder, builder: builder,
startOffset: offset, startOffset: offset,
token: lengths.length token: lengths.length,
layoutState: layoutState
) )
) )
), ),
......
...@@ -2,62 +2,145 @@ ...@@ -2,62 +2,145 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
import 'package:sky/animation/animation_performance.dart';
import 'package:sky/animation/curves.dart';
import 'package:sky/base/lerp.dart'; import 'package:sky/base/lerp.dart';
import 'package:sky/painting/text_style.dart'; import 'package:sky/painting/text_style.dart';
import 'package:sky/theme/colors.dart'; import 'package:sky/theme/colors.dart';
import 'package:sky/widgets/animated_component.dart';
import 'package:sky/widgets/basic.dart'; import 'package:sky/widgets/basic.dart';
import 'package:sky/widgets/block_viewport.dart';
import 'package:sky/widgets/card.dart'; import 'package:sky/widgets/card.dart';
import 'package:sky/widgets/dismissable.dart'; import 'package:sky/widgets/dismissable.dart';
import 'package:sky/widgets/scaffold.dart';
import 'package:sky/widgets/variable_height_scrollable.dart'; import 'package:sky/widgets/variable_height_scrollable.dart';
import 'package:sky/widgets/scaffold.dart';
import 'package:sky/widgets/theme.dart'; import 'package:sky/widgets/theme.dart';
import 'package:sky/widgets/tool_bar.dart'; import 'package:sky/widgets/tool_bar.dart';
import 'package:sky/widgets/widget.dart'; import 'package:sky/widgets/widget.dart';
import 'package:sky/theme/colors.dart' as colors;
import 'package:sky/widgets/task_description.dart'; import 'package:sky/widgets/task_description.dart';
class CardModel {
CardModel(this.value, this.height, this.color);
int value;
double height;
Color color;
AnimationPerformance performance;
String get label => "Item $value";
String get key => value.toString();
bool operator ==(other) => other is CardModel && other.value == value;
int get hashCode => 373 * 37 * value.hashCode;
}
class ShrinkingCard extends AnimatedComponent {
ShrinkingCard({
String key,
CardModel this.card,
Function this.onUpdated,
Function this.onCompleted
}) : super(key: key);
CardModel card;
Function onUpdated;
Function onCompleted;
double get currentHeight => card.performance.variable.value;
void initState() {
assert(card.performance != null);
card.performance.addListener(handleAnimationProgress);
watch(card.performance);
}
void handleAnimationProgress() {
if (card.performance.isCompleted) {
if (onCompleted != null)
onCompleted();
} else if (onUpdated != null) {
onUpdated();
}
}
void syncFields(ShrinkingCard source) {
card = source.card;
onCompleted = source.onCompleted;
onUpdated = source.onUpdated;
super.syncFields(source);
}
Widget build() => new Container(height: currentHeight);
}
class CardCollectionApp extends App { class CardCollectionApp extends App {
final TextStyle cardLabelStyle = final TextStyle cardLabelStyle =
new TextStyle(color: white, fontSize: 18.0, fontWeight: bold); new TextStyle(color: white, fontSize: 18.0, fontWeight: bold);
final List<double> cardHeights = [ BlockViewportLayoutState layoutState = new BlockViewportLayoutState();
48.0, 64.0, 82.0, 46.0, 60.0, 55.0, 84.0, 96.0, 50.0, List<CardModel> cardModels;
48.0, 64.0, 82.0, 46.0, 60.0, 55.0, 84.0, 96.0, 50.0,
48.0, 64.0, 82.0, 46.0, 60.0, 55.0, 84.0, 96.0, 50.0,
48.0, 64.0, 82.0, 46.0, 60.0, 55.0, 84.0, 96.0, 50.0
];
List<int> visibleCardIndices;
void initState() { void initState() {
visibleCardIndices = new List.generate(cardHeights.length, (i) => i); List<double> cardHeights = <double>[
48.0, 63.0, 82.0, 146.0, 60.0, 55.0, 84.0, 96.0, 50.0,
48.0, 63.0, 82.0, 146.0, 60.0, 55.0, 84.0, 96.0, 50.0,
48.0, 63.0, 82.0, 146.0, 60.0, 55.0, 84.0, 96.0, 50.0
];
cardModels = new List.generate(cardHeights.length, (i) {
Color color = lerpColor(Red[300], Blue[900], i / cardHeights.length);
return new CardModel(i, cardHeights[i], color);
});
super.initState(); super.initState();
} }
void dismissCard(int cardIndex) { void shrinkCard(CardModel card, int index) {
if (card.performance != null)
return;
layoutState.invalidate([index]);
setState(() { setState(() {
visibleCardIndices.remove(cardIndex); assert(card.performance == null);
card.performance = new AnimationPerformance()
..duration = const Duration(milliseconds: 300)
..variable = new AnimatedType<double>(
card.height + kCardMargins.top + kCardMargins.bottom,
end: 0.0,
curve: ease,
interval: new Interval(0.5, 1.0)
)
..play();
}); });
} }
Widget _builder(int index) { void dismissCard(CardModel card) {
if (index >= visibleCardIndices.length) if (cardModels.contains(card)) {
setState(() {
cardModels.remove(card);
});
}
}
Widget builder(int index) {
if (index >= cardModels.length)
return null; return null;
CardModel card = cardModels[index];
if (card.performance != null) {
return new ShrinkingCard(
key: card.key,
card: card,
onUpdated: () { layoutState.invalidate([index]); },
onCompleted: () { dismissCard(card); }
);
}
int cardIndex = visibleCardIndices[index];
Color color = lerpColor(Red[500], Blue[500], cardIndex / cardHeights.length);
Widget label = new Text("Item ${cardIndex}", style: cardLabelStyle);
return new Dismissable( return new Dismissable(
key: cardIndex.toString(), key: card.key,
onDismissed: () { dismissCard(cardIndex); }, onDismissed: () { shrinkCard(card, index); },
child: new Card( child: new Card(
color: color, color: card.color,
child: new Container( child: new Container(
height: cardHeights[cardIndex], height: card.height,
padding: const EdgeDims.all(8.0), padding: const EdgeDims.all(8.0),
child: new Center(child: label) child: new Center(child: new Text(card.label, style: cardLabelStyle))
) )
) )
); );
...@@ -68,16 +151,17 @@ class CardCollectionApp extends App { ...@@ -68,16 +151,17 @@ class CardCollectionApp extends App {
padding: const EdgeDims.symmetric(vertical: 12.0, horizontal: 8.0), padding: const EdgeDims.symmetric(vertical: 12.0, horizontal: 8.0),
decoration: new BoxDecoration(backgroundColor: Theme.of(this).primarySwatch[50]), decoration: new BoxDecoration(backgroundColor: Theme.of(this).primarySwatch[50]),
child: new VariableHeightScrollable( child: new VariableHeightScrollable(
builder: _builder, builder: builder,
token: visibleCardIndices.length token: cardModels.length,
layoutState: layoutState
) )
); );
return new Theme( return new Theme(
data: new ThemeData( data: new ThemeData(
brightness: ThemeBrightness.light, brightness: ThemeBrightness.light,
primarySwatch: colors.Blue, primarySwatch: Blue,
accentColor: colors.RedAccent[200] accentColor: RedAccent[200]
), ),
child: new TaskDescription( child: new TaskDescription(
label: 'Cards', label: 'Cards',
......
...@@ -5,6 +5,8 @@ ...@@ -5,6 +5,8 @@
import 'package:sky/widgets/basic.dart'; import 'package:sky/widgets/basic.dart';
import 'package:sky/widgets/material.dart'; import 'package:sky/widgets/material.dart';
const EdgeDims kCardMargins = const EdgeDims.all(4.0);
/// A material design card /// A material design card
/// ///
/// <https://www.google.com/design/spec/components/cards.html> /// <https://www.google.com/design/spec/components/cards.html>
...@@ -16,7 +18,7 @@ class Card extends Component { ...@@ -16,7 +18,7 @@ class Card extends Component {
Widget build() { Widget build() {
return new Container( return new Container(
margin: const EdgeDims.all(4.0), margin: kCardMargins,
child: new Material( child: new Material(
color: color, color: color,
type: MaterialType.card, type: MaterialType.card,
......
...@@ -2,8 +2,6 @@ ...@@ -2,8 +2,6 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
import 'dart:collection';
import 'package:sky/animation/scroll_behavior.dart'; import 'package:sky/animation/scroll_behavior.dart';
import 'package:sky/widgets/basic.dart'; import 'package:sky/widgets/basic.dart';
import 'package:sky/widgets/block_viewport.dart'; import 'package:sky/widgets/block_viewport.dart';
...@@ -14,18 +12,39 @@ class VariableHeightScrollable extends Scrollable { ...@@ -14,18 +12,39 @@ class VariableHeightScrollable extends Scrollable {
VariableHeightScrollable({ VariableHeightScrollable({
String key, String key,
this.builder, this.builder,
this.token this.token,
this.layoutState
}) : super(key: key); }) : super(key: key);
IndexedBuilder builder; IndexedBuilder builder;
Object token; Object token;
BlockViewportLayoutState layoutState;
// When the token changes the scrollable's contents may have
// changed. Remember as much so that after the new contents
// have been laid out we can adjust the scrollOffset so that
// the last page of content is still visible.
bool _contentsChanged = true; bool _contentsChanged = true;
void initState() {
assert(layoutState != null);
layoutState.removeListener(_handleLayoutChanged);
layoutState.addListener(_handleLayoutChanged);
super.initState();
}
void syncFields(VariableHeightScrollable source) { void syncFields(VariableHeightScrollable source) {
builder = source.builder; builder = source.builder;
if (token != source.token) if (token != source.token)
_contentsChanged = true; _contentsChanged = true;
token = source.token; token = source.token;
if (layoutState != source.layoutState) {
// Warning: this is unlikely to be what you intended.
assert(source.layoutState != null);
layoutState == source.layoutState;
layoutState.removeListener(_handleLayoutChanged);
layoutState.addListener(_handleLayoutChanged);
}
super.syncFields(source); super.syncFields(source);
} }
...@@ -36,15 +55,9 @@ class VariableHeightScrollable extends Scrollable { ...@@ -36,15 +55,9 @@ class VariableHeightScrollable extends Scrollable {
scrollBehavior.containerSize = newSize.height; scrollBehavior.containerSize = newSize.height;
} }
void _handleLayoutChanged( void _handleLayoutChanged() {
int firstVisibleChildIndex, if (layoutState.didReachLastChild) {
int visibleChildCount, scrollBehavior.contentsSize = layoutState.contentsSize;
UnmodifiableListView<double> childOffsets,
bool didReachLastChild
) {
assert(childOffsets.length > 0);
if (didReachLastChild) {
scrollBehavior.contentsSize = childOffsets.last;
if (_contentsChanged && scrollOffset > scrollBehavior.maxScrollOffset) { if (_contentsChanged && scrollOffset > scrollBehavior.maxScrollOffset) {
_contentsChanged = false; _contentsChanged = false;
settleScrollOffset(); settleScrollOffset();
...@@ -59,7 +72,7 @@ class VariableHeightScrollable extends Scrollable { ...@@ -59,7 +72,7 @@ class VariableHeightScrollable extends Scrollable {
callback: _handleSizeChanged, callback: _handleSizeChanged,
child: new BlockViewport( child: new BlockViewport(
builder: builder, builder: builder,
onLayoutChanged: _handleLayoutChanged, layoutState: layoutState,
startOffset: scrollOffset, startOffset: scrollOffset,
token: token token: token
) )
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment