Commit e053b35c authored by Eric Seidel's avatar Eric Seidel

Merge pull request #566 from eseidelGoogle/fitness

Make Fitness tiles nicer
parents 1e0ceaf1 b2212785
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Forked from https://github.com/dart-lang/sdk/blob/master/samples-dev/swarm/swarm_ui_lib/util/DateUtils.dart
class DateUtils {
static const WEEKDAYS = const ['Monday', 'Tuesday', 'Wednesday', 'Thursday',
'Friday', 'Saturday', 'Sunday'];
static const YESTERDAY = 'Yesterday';
static const MS_IN_WEEK =
DateTime.DAYS_PER_WEEK * Duration.MILLISECONDS_PER_DAY;
/** Formats a time in H:MM A format */
static String toHourMinutesString(Duration duration) {
assert(duration.inDays == 0);
int hours = duration.inHours;
String a;
if (hours >= 12) {
a = 'pm';
if (hours != 12) {
hours -= 12;
}
} else {
a = 'am';
if (hours == 0) {
hours += 12;
}
}
String twoDigits(int n) {
if (n >= 10) return "${n}";
return "0${n}";
}
String mm =
twoDigits(duration.inMinutes.remainder(Duration.MINUTES_PER_HOUR));
return "${hours}:${mm} ${a}";
}
/**
* A date/time formatter that takes into account the current date/time:
* - if it's from today, just show the time
* - if it's from yesterday, just show 'Yesterday'
* - if it's from the same week, just show the weekday
* - otherwise, show just the date
*/
static String toRecentTimeString(DateTime then) {
bool datesAreEqual(DateTime d1, DateTime d2) {
return (d1.year == d2.year) && (d1.month == d2.month) &&
(d1.day == d2.day);
}
final now = new DateTime.now();
if (datesAreEqual(then, now)) {
return toHourMinutesString(new Duration(
days: 0,
hours: then.hour,
minutes: then.minute,
seconds: then.second,
milliseconds: then.millisecond));
}
final today = new DateTime(now.year, now.month, now.day, 0, 0, 0, 0);
Duration delta = today.difference(then);
if (delta.inMilliseconds < Duration.MILLISECONDS_PER_DAY) {
return YESTERDAY;
} else if (delta.inMilliseconds < MS_IN_WEEK) {
return WEEKDAYS[then.weekday];
} else {
// TODO(jmesserly): locale specific date format
String twoDigits(int n) {
if (n >= 10) return "${n}";
return "0${n}";
}
String twoDigitMonth = twoDigits(then.month);
String twoDigitDay = twoDigits(then.day);
return "${then.year}-${twoDigitMonth}-${twoDigitDay}";
}
}
}
...@@ -6,7 +6,9 @@ part of fitness; ...@@ -6,7 +6,9 @@ part of fitness;
typedef void FitnessItemHandler(FitnessItem item); typedef void FitnessItemHandler(FitnessItem item);
const double kFitnessItemHeight = 79.0; // TODO(eseidel): This should be a constant on a SingleLineTile class
// https://www.google.com/design/spec/components/lists.html#lists-specs
const double kFitnessItemHeight = 48.0;
abstract class FitnessItem { abstract class FitnessItem {
FitnessItem.fromJson(Map json) : when = DateTime.parse(json['when']); FitnessItem.fromJson(Map json) : when = DateTime.parse(json['when']);
...@@ -19,7 +21,7 @@ abstract class FitnessItem { ...@@ -19,7 +21,7 @@ abstract class FitnessItem {
Map toJson() => { 'when' : when.toIso8601String() }; Map toJson() => { 'when' : when.toIso8601String() };
// TODO(jackson): Internationalize // TODO(jackson): Internationalize
String get displayDate => "${when.year.toString()}-${when.month.toString().padLeft(2,'0')}-${when.day.toString().padLeft(2,'0')}"; String get displayDate => DateUtils.toRecentTimeString(when);
FitnessItemRow toRow({ FitnessItemHandler onDismissed }); FitnessItemRow toRow({ FitnessItemHandler onDismissed });
} }
...@@ -40,12 +42,19 @@ abstract class FitnessItemRow extends Component { ...@@ -40,12 +42,19 @@ abstract class FitnessItemRow extends Component {
Widget build() { Widget build() {
return new Dismissable( return new Dismissable(
onDismissed: () => onDismissed(item), onDismissed: () => onDismissed(item),
child: new Card( child: new Container(
child: new Container( height: kFitnessItemHeight,
height: kFitnessItemHeight, // TODO(eseidel): Padding top should be 16px for a single-line tile:
padding: const EdgeDims.all(8.0), // https://www.google.com/design/spec/components/lists.html#lists-specs
child: buildContent() padding: const EdgeDims.all(10.0),
) // TODO(eseidel): This line should be drawn by the list as it should
// stay put even when the tile is dismissed!
decoration: new BoxDecoration(
border: new Border(
bottom: new BorderSide(color: Theme.of(this).dividerColor)
)
),
child: buildContent()
) )
); );
} }
......
...@@ -9,6 +9,8 @@ import 'package:sky/painting/text_style.dart'; ...@@ -9,6 +9,8 @@ import 'package:sky/painting/text_style.dart';
import 'package:sky/theme/colors.dart' as colors; import 'package:sky/theme/colors.dart' as colors;
import 'package:sky/widgets.dart'; import 'package:sky/widgets.dart';
import 'user_data.dart'; import 'user_data.dart';
import 'date_utils.dart';
import 'dart:async';
part 'feed.dart'; part 'feed.dart';
part 'fitness_item.dart'; part 'fitness_item.dart';
...@@ -17,23 +19,47 @@ part 'meal.dart'; ...@@ -17,23 +19,47 @@ part 'meal.dart';
part 'measurement.dart'; part 'measurement.dart';
part 'settings.dart'; part 'settings.dart';
class FitnessApp extends App { class UserData {
List<FitnessItem> _items = [];
List<FitnessItem> get items => _items;
void set items(List<FitnessItem> newItems) {
_items = [];
_items.addAll(newItems);
sort();
}
void sort() {
_items.sort((a, b) => -a.when.compareTo(b.when));
}
void addAndSave(FitnessItem item) {
_items.add(item);
sort();
save();
}
void removeAndSave(FitnessItem item) {
_items.remove(item);
save();
}
Future save() => saveFitnessData(_items);
}
class FitnessApp extends App {
NavigationState _navigationState; NavigationState _navigationState;
final List<FitnessItem> _userData = []; final UserData _userData = new UserData();
void didMount() { void didMount() {
super.didMount(); super.didMount();
loadFitnessData().then((List<Measurement> list) { loadFitnessData().then((List<Measurement> list) {
setState(() { setState(() => _userData.items = list);
_userData.addAll(list);
});
}).catchError((e) => print("Failed to load data: $e")); }).catchError((e) => print("Failed to load data: $e"));
} }
void save() { void save() {
saveFitnessData(_userData) _userData.save().catchError((e) => print("Failed to load data: $e"));
.catchError((e) => print("Failed to load data: $e"));
} }
void initState() { void initState() {
...@@ -42,7 +68,7 @@ class FitnessApp extends App { ...@@ -42,7 +68,7 @@ class FitnessApp extends App {
name: '/', name: '/',
builder: (navigator, route) => new FeedFragment( builder: (navigator, route) => new FeedFragment(
navigator: navigator, navigator: navigator,
userData: _userData, userData: _userData.items,
onItemCreated: _handleItemCreated, onItemCreated: _handleItemCreated,
onItemDeleted: _handleItemDeleted onItemDeleted: _handleItemDeleted
) )
...@@ -71,27 +97,18 @@ class FitnessApp extends App { ...@@ -71,27 +97,18 @@ class FitnessApp extends App {
void onBack() { void onBack() {
if (_navigationState.hasPrevious()) { if (_navigationState.hasPrevious()) {
setState(() { setState(() => _navigationState.pop());
_navigationState.pop();
});
} else { } else {
super.onBack(); super.onBack();
} }
} }
void _handleItemCreated(FitnessItem item) { void _handleItemCreated(FitnessItem item) {
setState(() { setState(() => _userData.addAndSave(item));
_userData.add(item);
_userData.sort((a, b) => a.when.compareTo(b.when));
save();
});
} }
void _handleItemDeleted(FitnessItem item) { void _handleItemDeleted(FitnessItem item) {
setState(() { setState(() => _userData.removeAndSave(item));
_userData.remove(item);
saveFitnessData(_userData);
});
} }
BackupMode backupSetting = BackupMode.disabled; BackupMode backupSetting = BackupMode.disabled;
......
...@@ -35,7 +35,7 @@ class MeasurementRow extends FitnessItemRow { ...@@ -35,7 +35,7 @@ class MeasurementRow extends FitnessItemRow {
new Flexible( new Flexible(
child: new Text( child: new Text(
measurement.displayWeight, measurement.displayWeight,
style: const TextStyle(textAlign: TextAlign.right) style: Theme.of(this).text.subhead
) )
), ),
new Flexible( new Flexible(
......
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