git-touch-android-ios-app/lib/screens/gl_project.dart

250 lines
9.4 KiB
Dart

import 'package:antd_mobile/antd_mobile.dart';
import 'package:filesize/filesize.dart';
import 'package:flutter/cupertino.dart';
import 'package:flutter/widgets.dart';
import 'package:flutter_gen/gen_l10n/S.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:git_touch/models/auth.dart';
import 'package:git_touch/models/gitlab.dart';
import 'package:git_touch/models/theme.dart';
import 'package:git_touch/scaffolds/refresh_stateful.dart';
import 'package:git_touch/utils/utils.dart';
import 'package:git_touch/widgets/action_button.dart';
import 'package:git_touch/widgets/app_bar_title.dart';
import 'package:git_touch/widgets/entry_item.dart';
import 'package:git_touch/widgets/language_bar.dart';
import 'package:git_touch/widgets/markdown_view.dart';
import 'package:git_touch/widgets/repo_header.dart';
import 'package:go_router/go_router.dart';
import 'package:provider/provider.dart';
import 'package:tuple/tuple.dart';
class GlProjectScreen extends StatelessWidget {
const GlProjectScreen(this.id, {this.branch});
final int id;
final String? branch;
@override
Widget build(BuildContext context) {
return RefreshStatefulScaffold<
Tuple5<GitlabProject, Future<Map<String, double>>, Future<int>,
MarkdownViewData?, List<GitlabBranch>>>(
title: AppBarTitle(AppLocalizations.of(context)!.project),
fetch: () async {
final auth = context.read<AuthModel>();
final p =
await auth.fetchGitlab('/projects/$id?statistics=1').then((res) {
return GitlabProject.fromJson(res);
});
final langFuture =
auth.fetchGitlab('/projects/$id/languages').then((v) {
return Map<String, double>.from(v);
});
// auth.fetchGitlab('/projects/$id/badges') // badges
final memberCountFuture = auth
.fetchGitlabWithPage('/projects/$id/members?per_page=1')
.then((v) => v.total);
MarkdownViewData? readmeData;
if (p.readmeUrl != null) {
md() => auth.fetchWithGitlabToken(
p.readmeUrl!.replaceFirst(r'/blob/', '/raw/'));
readmeData = MarkdownViewData(
context,
md: md,
html: () => md().then((md) async {
// we should get the markdown content, then render it
// https://gitlab.com/gitlab-org/gitlab/-/issues/16335
final res = await auth.fetchGitlab('/markdown',
isPost: true,
body: {
'text': md,
'gfm': true,
'project': '${p.namespace!.name}/${p.name}'
});
return (res['html'] as String).normalizedHtml;
}),
);
}
final branches = await auth
.fetchGitlab('/projects/$id/repository/branches')
.then((v) {
return [for (var branch in v) GitlabBranch.fromJson(branch)];
});
return Tuple5(p, langFuture, memberCountFuture, readmeData, branches);
},
actionBuilder: (t, _) {
return ActionButton(
title: AppLocalizations.of(context)!.projectActions,
items: [
...ActionItem.getUrlActions(t.item1.webUrl),
],
);
},
bodyBuilder: (t, _) {
final p = t.item1;
final langFuture = t.item2;
final memberCountFuture = t.item3;
final readmeData = t.item4;
final branches = t.item5;
final theme = Provider.of<ThemeModel>(context);
final auth = Provider.of<AuthModel>(context);
final prefix =
'${auth.activeAccount!.domain}/${p.namespace!.path}/${p.name}'
.urlencode;
return Column(
crossAxisAlignment: CrossAxisAlignment.stretch,
children: <Widget>[
RepoHeader(
avatarUrl: p.avatarUrl,
avatarLink: p.namespace!.kind == 'group'
? '/gitlab/group/${p.namespace!.id}'
: '/gitlab/user/${p.namespace!.id}',
owner: p.namespace!.name,
name: p.name,
description: p.description,
// trailings: <Widget>[
// if (badges.isNotEmpty)
// Wrap(spacing: 4, runSpacing: 4, children: [
// for (var label in badges)
// SvgPicture.network(label.renderedImageUrl, height: 20),
// ])
// ],
),
CommonStyle.border,
Row(
children: <Widget>[
FutureBuilder<int>(
future: memberCountFuture,
builder: (context, snapshot) {
return EntryItem(
count: snapshot.data!,
text: AppLocalizations.of(context)!.members,
url: '/gitlab/projects/$id/members',
);
},
),
EntryItem(
count: p.starCount!,
text: AppLocalizations.of(context)!.stars,
url: '/gitlab/projects/$id/starrers',
),
EntryItem(
count: p.forksCount!,
text: AppLocalizations.of(context)!.forks, // TODO:
),
],
),
CommonStyle.border,
FutureBuilder<Map<String, double>>(
future: langFuture,
builder: (context, snapshot) {
final data = snapshot.data;
if (data == null) {
return LanguageBar([
LanguageBarItem(name: '', ratio: 1),
]);
} else {
return LanguageBar([
for (var e in data.entries)
LanguageBarItem(name: e.key, ratio: e.value / 100)
]);
}
},
),
CommonStyle.border,
AntList(
children: [
AntListItem(
prefix: const Icon(Octicons.code),
extra: p.statistics == null
? null
: Text(filesize(p.statistics!.repositorySize)),
onClick: () {
context.push(
'/gitlab/projects/$id/tree/${branch ?? p.defaultBranch}');
},
child: FutureBuilder<Map<String, double>>(
future: langFuture,
builder: (context, snapshot) {
if (snapshot.data == null) {
return const Text('');
} else {
final langs = snapshot.data!.keys;
return Text(langs.isEmpty
? AppLocalizations.of(context)!.code
: langs.first);
}
},
),
),
if (p.issuesEnabled!)
AntListItem(
prefix: const Icon(Octicons.issue_opened),
extra: Text(numberFormat.format(p.openIssuesCount)),
onClick: () {
context
.push('/gitlab/projects/$id/issues?prefix=$prefix');
},
child: Text(AppLocalizations.of(context)!.issues),
),
if (p.mergeRequestsEnabled!)
AntListItem(
prefix: const Icon(Octicons.git_pull_request),
child: Text(AppLocalizations.of(context)!.mergeRequests),
onClick: () {
context.push(
'/gitlab/projects/$id/merge_requests?prefix=$prefix');
},
),
AntListItem(
prefix: const Icon(Octicons.history),
extra: p.statistics == null
? null
: Text(p.statistics!.commitCount.toString()),
onClick: () {
context.push(
'/gitlab/projects/$id/commits?prefix=$prefix&branch=${branch ?? p.defaultBranch}');
},
child: Text(AppLocalizations.of(context)!.commits), // EDIT
),
AntListItem(
prefix: const Icon(Octicons.git_branch),
extra: Text(
'${(branch ?? p.defaultBranch) ?? ''}${branches.length}'),
onClick: () async {
if (branches.length < 2) return;
await theme.showPicker(
context,
PickerGroupItem(
value: branch,
items: branches
.map((b) => PickerItem(b.name, text: b.name))
.toList(),
onClose: (ref) {
if (ref != branch) {
context.pushUrl('/gitlab/projects/$id?branch=$ref',
replace: true);
}
},
),
);
},
child: Text(AppLocalizations.of(context)!.branches),
),
],
),
CommonStyle.verticalGap,
MarkdownView(readmeData),
],
);
},
);
}
}