git-touch-android-ios-app/lib/widgets/timeline_item.dart

310 lines
9.3 KiB
Dart
Raw Normal View History

2019-01-29 06:34:52 +01:00
import 'dart:core';
import 'package:flutter/material.dart';
import 'package:flutter/cupertino.dart';
2019-05-12 09:14:28 +02:00
import 'package:primer/primer.dart';
2019-01-31 07:37:25 +01:00
import '../utils/utils.dart';
2019-02-07 07:35:19 +01:00
import 'comment_item.dart';
import 'user_name.dart';
2019-09-07 11:48:59 +02:00
class TimelineEventItem extends StatelessWidget {
final String actor;
final IconData iconData;
final Color iconColor;
final TextSpan textSpan;
final item;
2019-09-07 11:48:59 +02:00
TimelineEventItem({
this.actor,
this.iconData = Octicons.octoface,
this.iconColor = PrimerColors.gray400,
this.textSpan,
this.item,
});
2019-09-07 11:48:59 +02:00
@override
Widget build(BuildContext context) {
return Row(
children: <Widget>[
2019-09-07 11:48:59 +02:00
SizedBox(width: 6),
Icon(iconData, color: iconColor, size: 20),
2019-09-07 11:48:59 +02:00
SizedBox(width: 12),
Expanded(
child: RichText(
text: TextSpan(style: TextStyle(color: Colors.black), children: [
2019-02-18 11:54:06 +01:00
// TODO: actor is null
2019-09-07 11:48:59 +02:00
createUserSpan(context, actor),
textSpan,
// TextSpan(text: ' ' + TimeAgo.formatFromString(item['createdAt']))
]),
),
),
],
);
}
2019-09-07 11:48:59 +02:00
}
class TimelineItem extends StatelessWidget {
final Map<String, dynamic> payload;
final Function(String emojiKey, bool isRemove) onReaction;
TimelineItem(this.payload, {@required this.onReaction});
TextSpan _buildReviewText(BuildContext context, item) {
switch (item['state']) {
case 'APPROVED':
return TextSpan(text: ' approved these changes');
case 'COMMENTED':
return TextSpan(text: ' commented ');
default:
return warningSpan;
}
}
2019-01-29 06:34:52 +01:00
TextSpan _buildLabel(item) {
2019-09-14 09:39:38 +02:00
// FIXME:
var color = convertColor(item['label']['color']);
2019-01-29 06:34:52 +01:00
return TextSpan(
text: item['label']['name'],
style: TextStyle(
2019-09-14 09:39:38 +02:00
color: getFontColorByBrightness(color),
backgroundColor: color,
2019-02-04 11:32:39 +01:00
// https://stackoverflow.com/a/52592679
// ..strokeWidth = 16.5
// ..style = PaintingStyle.stroke
),
2019-01-29 06:34:52 +01:00
);
}
Widget _buildByType(BuildContext context) {
String type = payload['__typename'];
2019-09-07 11:48:59 +02:00
var defaultItem = TimelineEventItem(
2019-02-08 07:30:22 +01:00
actor: '',
iconData: Octicons.octoface,
textSpan: TextSpan(children: [
TextSpan(text: 'Woops, $type type not implemented yet'),
]),
item: payload,
2019-02-08 07:30:22 +01:00
);
switch (type) {
// common types
case 'Commit':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['author']['user'] == null
2019-02-18 11:54:06 +01:00
? null
: payload['author']['user']['login'],
2019-02-08 07:30:22 +01:00
iconData: Octicons.git_commit,
textSpan: TextSpan(children: [
2019-02-08 07:30:22 +01:00
TextSpan(text: ' added commit '),
TextSpan(text: payload['oid'].substring(0, 8))
]),
item: payload,
);
2019-02-08 07:30:22 +01:00
case 'IssueComment':
2019-04-05 15:19:00 +02:00
return CommentItem(payload, onReaction: onReaction);
2019-02-08 07:30:22 +01:00
case 'CrossReferencedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
2019-02-08 07:30:22 +01:00
iconData: Octicons.primitive_dot,
iconColor: Palette.green,
textSpan: TextSpan(
text: ' referenced this on #' +
payload['source']['number'].toString()),
item: payload,
);
case 'ClosedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
iconData: Octicons.circle_slash,
2019-05-12 09:14:28 +02:00
iconColor: PrimerColors.red600,
textSpan: TextSpan(text: ' closed this '),
item: payload,
);
2019-02-08 07:30:22 +01:00
case 'ReopenedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
iconData: Octicons.primitive_dot,
iconColor: Palette.green,
textSpan: TextSpan(text: ' reopened this '),
item: payload,
);
2019-02-08 07:30:22 +01:00
case 'SubscribedEvent':
case 'UnsubscribedEvent':
return defaultItem; // TODO:
case 'ReferencedEvent':
// TODO: isCrossRepository
if (payload['commit'] == null) {
2019-02-08 07:30:22 +01:00
return Container();
}
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
2019-02-08 07:30:22 +01:00
iconData: Octicons.bookmark,
textSpan: TextSpan(children: [
TextSpan(text: ' referenced this pull request from commit '),
TextSpan(text: payload['commit']['oid'].substring(0, 8)),
2019-02-08 07:30:22 +01:00
]),
item: payload,
2019-02-08 07:30:22 +01:00
);
case 'AssignedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
2019-02-08 07:30:22 +01:00
iconData: Octicons.key,
textSpan: TextSpan(children: [
TextSpan(text: ' assigned this to '),
TextSpan(text: payload['user']['login'])
2019-02-08 07:30:22 +01:00
]),
item: payload,
);
2019-02-08 07:30:22 +01:00
case 'UnassignedEvent':
return defaultItem; // TODO:
2019-01-29 06:34:52 +01:00
case 'LabeledEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
2019-01-29 06:34:52 +01:00
iconData: Octicons.tag,
textSpan: TextSpan(children: [
TextSpan(text: ' added '),
_buildLabel(payload),
2019-01-29 06:34:52 +01:00
TextSpan(text: ' label'),
]),
item: payload,
2019-01-29 06:34:52 +01:00
);
case 'UnlabeledEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
2019-01-29 06:34:52 +01:00
iconData: Octicons.tag,
textSpan: TextSpan(children: [
TextSpan(text: ' removed '),
_buildLabel(payload),
2019-01-29 06:34:52 +01:00
TextSpan(text: ' label'),
]),
item: payload,
2019-01-29 06:34:52 +01:00
);
2019-02-08 07:30:22 +01:00
2019-01-29 06:34:52 +01:00
case 'MilestonedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
2019-01-29 06:34:52 +01:00
iconData: Octicons.milestone,
textSpan: TextSpan(children: [
TextSpan(text: ' added this to '),
TextSpan(text: payload['milestoneTitle']),
2019-01-29 06:34:52 +01:00
TextSpan(text: ' milestone'),
]),
item: payload,
2019-01-29 06:34:52 +01:00
);
2019-02-08 07:30:22 +01:00
case 'DemilestonedEvent':
return defaultItem; // TODO:
case 'RenamedTitleEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
2019-02-08 07:30:22 +01:00
iconData: Octicons.pencil,
2019-01-29 06:34:52 +01:00
textSpan: TextSpan(children: [
2019-02-08 07:30:22 +01:00
TextSpan(text: ' changed the title '),
TextSpan(
text: payload['previousTitle'],
2019-02-08 07:30:22 +01:00
style: TextStyle(decoration: TextDecoration.lineThrough),
),
TextSpan(text: ' to '),
TextSpan(text: payload['currentTitle'])
2019-01-29 06:34:52 +01:00
]),
item: payload,
2019-01-29 06:34:52 +01:00
);
2019-02-08 07:30:22 +01:00
case 'LockedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
2019-02-08 07:30:22 +01:00
iconData: Octicons.lock,
2019-01-29 06:34:52 +01:00
textSpan: TextSpan(children: [
2019-02-08 07:30:22 +01:00
TextSpan(text: ' locked this conversation '),
2019-01-29 06:34:52 +01:00
]),
item: payload,
2019-01-29 06:34:52 +01:00
);
2019-02-08 07:30:22 +01:00
case 'UnlockedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
2019-01-29 06:34:52 +01:00
iconData: Octicons.key,
textSpan: TextSpan(children: [
2019-02-08 07:30:22 +01:00
TextSpan(text: ' unlocked this conversation '),
2019-01-29 06:34:52 +01:00
]),
item: payload,
2019-01-29 06:34:52 +01:00
);
2019-02-08 07:30:22 +01:00
// issue only types
case 'TransferredEvent':
return defaultItem; // TODO:
// pull request only types
case 'CommitCommentThread':
return defaultItem; // TODO:
case 'PullRequestReview':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['author']['login'],
2019-02-06 12:14:11 +01:00
iconColor: Color(0xff28a745),
iconData: Octicons.check,
textSpan: _buildReviewText(context, payload),
item: payload,
);
2019-02-08 07:30:22 +01:00
case 'PullRequestReviewThread':
case 'PullRequestReviewComment':
return defaultItem; // TODO:
case 'MergedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
iconData: Octicons.git_merge,
2019-02-06 12:14:11 +01:00
iconColor: Color(0xff6f42c1),
textSpan: TextSpan(children: [
TextSpan(text: ' merged commit '),
TextSpan(text: payload['commit']['oid'].substring(0, 8)),
TextSpan(text: ' into '),
TextSpan(text: payload['mergeRefName']),
]),
item: payload,
);
2019-02-08 07:30:22 +01:00
case 'DeployedEvent':
case 'DeploymentEnvironmentChangedEvent':
return defaultItem; // TODO:
case 'HeadRefDeletedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
actor: payload['actor']['login'],
iconData: Octicons.git_branch,
textSpan: TextSpan(children: [
TextSpan(text: ' deleted the '),
TextSpan(text: payload['headRefName']),
TextSpan(text: ' branch'),
]),
item: payload,
);
2019-02-08 07:30:22 +01:00
case 'HeadRefRestoredEvent':
case 'HeadRefForcePushedEvent':
case 'BaseRefForcePushedEvent':
return defaultItem; // TODO:
case 'ReviewRequestedEvent':
2019-09-07 11:48:59 +02:00
return TimelineEventItem(
2019-02-08 07:30:22 +01:00
iconData: Octicons.eye,
// actor: payload['author']['login'],
// TODO:
actor: 'test',
textSpan: TextSpan(children: [
2019-02-08 07:30:22 +01:00
TextSpan(text: ' requested a review from '),
2019-09-07 11:48:59 +02:00
createUserSpan(context, payload['requestedReviewer']['login']),
]),
item: payload,
);
2019-02-08 07:30:22 +01:00
case 'ReviewRequestRemovedEvent':
case 'ReviewDismissedEvent':
return defaultItem; // TODO:
default:
2019-02-08 07:30:22 +01:00
return defaultItem;
}
}
@override
Widget build(BuildContext context) {
return Container(
2019-04-05 15:19:00 +02:00
padding: EdgeInsets.all(12),
child: _buildByType(context),
);
}
}