import 'package:flutter/material.dart'; import 'package:flutter/cupertino.dart'; import 'package:git_touch/models/theme.dart'; import 'package:git_touch/scaffolds/refresh_stateful.dart'; import 'package:git_touch/screens/repositories.dart'; import 'package:git_touch/widgets/action_entry.dart'; import 'package:git_touch/widgets/app_bar_title.dart'; import 'package:git_touch/widgets/table_view.dart'; import 'package:git_touch/widgets/text_contains_organization.dart'; import 'package:git_touch/widgets/user_item.dart'; import 'package:primer/primer.dart'; import 'package:github_contributions/github_contributions.dart'; import 'package:git_touch/models/auth.dart'; import 'package:provider/provider.dart'; import '../widgets/entry_item.dart'; import 'package:git_touch/widgets/repository_item.dart'; import 'package:git_touch/widgets/action_button.dart'; import '../screens/users.dart'; import '../screens/settings.dart'; import '../utils/utils.dart'; class UserScreen extends StatelessWidget { final String login; final bool isMe; UserScreen(this.login, {this.isMe = false}); Future query(BuildContext context) async { var data = await Provider.of(context).query(''' { user(login: "$login") { $userGqlChunk company location email websiteUrl starredRepositories { totalCount } followers { totalCount } following { totalCount } repositories(first: 6, ownerAffiliations: OWNER, orderBy: {field: STARGAZERS, direction: DESC}) { totalCount nodes { $repoChunk } } pinnedItems(first: 6) { nodes { ... on Repository { $repoChunk } } } viewerCanFollow viewerIsFollowing url } } '''); return data['user']; } Iterable _buildRepos(payload) { String title; List items = []; var pinnedItems = payload['pinnedItems']['nodes'] as List; var repositories = payload['repositories']['nodes'] as List; if (pinnedItems.isNotEmpty) { title = 'pinned repositories'; items = pinnedItems; } else if (repositories.isNotEmpty) { title = 'popular repositories'; items = repositories; } items = items .where((x) => x != null) .toList(); // TODO: Pinned items may include somethings other than repo if (items.isEmpty) return []; return [ verticalGap, if (title != null) TableViewHeader(title), ...join( borderView, items.map((item) { return RepositoryItem(item); }).toList(), ), ]; } Widget _buildContributions(List contributions) { final row = Row( children: [], crossAxisAlignment: CrossAxisAlignment.start, ); Column column; contributions.asMap().forEach((i, v) { var rect = SizedBox( width: 10, height: 10, child: DecoratedBox( decoration: BoxDecoration( color: convertColor(v.color), ), ), ); if (i % 7 == 0) { column = Column(children: [rect]); row.children.add(column); row.children.add(SizedBox(width: 3)); } else { column.children.add(SizedBox(height: 3)); column.children.add(rect); } }); return Container( color: Colors.white, padding: EdgeInsets.all(10), child: SingleChildScrollView( scrollDirection: Axis.horizontal, reverse: true, child: row, ), ); } @override Widget build(BuildContext context) { return RefreshStatefulScaffold( fetchData: () { return Future.wait( [query(context), getContributions(login)], ); }, title: AppBarTitle('User'), actionBuilder: (payload) { var data = payload.data; if (isMe) { return ActionEntry( iconData: Icons.settings, onTap: () { Provider.of(context).pushRoute( context, (_) => SettingsScreen(), fullscreenDialog: true); }, ); } else { return ActionButton( title: 'User Actions', items: [ if (data != null && data[0]['viewerCanFollow']) ActionItem( text: data[0]['viewerIsFollowing'] ? 'Unfollow' : 'Follow', onPress: () async { if (data[0]['viewerIsFollowing']) { await Provider.of(context) .deleteWithCredentials('/user/following/$login'); data[0]['viewerIsFollowing'] = false; } else { Provider.of(context) .putWithCredentials('/user/following/$login'); data[0]['viewerIsFollowing'] = true; } }, ), if (data != null) ...[ ActionItem.share(data[0]['url']), ActionItem.launch(data[0]['url']), ], ], ); } }, bodyBuilder: (payload) { var data = payload.data[0]; var contributions = payload.data[1] as List; return Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ UserItem.fromData(data, inUserScreen: true), borderView, Row(children: [ EntryItem( count: data['repositories']['totalCount'], text: 'Repositories', screenBuilder: (context) => RepositoriesScreen(login), ), EntryItem( count: data['starredRepositories']['totalCount'], text: 'Stars', screenBuilder: (context) => RepositoriesScreen.stars(login), ), EntryItem( count: data['followers']['totalCount'], text: 'Followers', screenBuilder: (context) => UsersScreen.followers(login), ), EntryItem( count: data['following']['totalCount'], text: 'Following', screenBuilder: (context) => UsersScreen.following(login), ), ]), verticalGap, _buildContributions(contributions), verticalGap, TableView( hasIcon: true, items: [ if (isNotNullOrEmpty(data['company'])) TableViewItem( leftIconData: Octicons.organization, text: TextContainsOrganization(data['company'], style: TextStyle( fontSize: 16, color: PrimerColors.gray900), overflow: TextOverflow.ellipsis), ), if (isNotNullOrEmpty(data['location'])) TableViewItem( leftIconData: Octicons.location, text: Text(data['location']), onTap: () { launchUrl('https://www.google.com/maps/place/' + (data['location'] as String) .replaceAll(RegExp(r'\s+'), '')); }, ), if (isNotNullOrEmpty(data['email'])) TableViewItem( leftIconData: Octicons.mail, text: Text(data['email']), onTap: () { launchUrl('mailto:' + data['email']); }, ), if (isNotNullOrEmpty(data['websiteUrl'])) TableViewItem( leftIconData: Octicons.link, text: Text(data['websiteUrl']), onTap: () { var url = data['websiteUrl'] as String; if (!url.startsWith('http')) { url = 'http://$url'; } launchUrl(url); }, ), ], ), ..._buildRepos(data), verticalGap, ], ); }, ); } }