203 lines
7.8 KiB
Dart
203 lines
7.8 KiB
Dart
import 'dart:convert';
|
|
import 'dart:io';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:git_touch/models/auth.dart';
|
|
import 'package:git_touch/models/gitee.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/app_bar_title.dart';
|
|
import 'package:git_touch/widgets/entry_item.dart';
|
|
import 'package:git_touch/widgets/markdown_view.dart';
|
|
import 'package:git_touch/widgets/mutation_button.dart';
|
|
import 'package:git_touch/widgets/repo_header.dart';
|
|
import 'package:git_touch/widgets/table_view.dart';
|
|
import 'package:provider/provider.dart';
|
|
import 'package:tuple/tuple.dart';
|
|
import 'package:http/http.dart' as http;
|
|
import 'package:flutter_gen/gen_l10n/S.dart';
|
|
|
|
class StatusPayload {
|
|
bool isWatching;
|
|
bool isStarred;
|
|
StatusPayload(this.isWatching, this.isStarred);
|
|
}
|
|
|
|
class GeRepoScreen extends StatelessWidget {
|
|
final String owner;
|
|
final String name;
|
|
final String? branch;
|
|
GeRepoScreen(this.owner, this.name, {this.branch});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return RefreshStatefulScaffold<
|
|
Tuple4<GiteeRepo, MarkdownViewData, List<GiteeBranch>, StatusPayload>>(
|
|
title: AppBarTitle(AppLocalizations.of(context)!.repository),
|
|
fetch: () async {
|
|
final auth = context.read<AuthModel>();
|
|
final repo = await auth.fetchGitee('/repos/$owner/$name').then((v) {
|
|
return GiteeRepo.fromJson(v);
|
|
});
|
|
|
|
final md =
|
|
() => auth.fetchGitee('/repos/$owner/$name/readme').then((v) {
|
|
return (v['content'] as String?)?.base64ToUtf8 ?? '';
|
|
});
|
|
final html = () => md().then((v) async {
|
|
final res = await http.post(
|
|
Uri.parse('${auth.activeAccount!.domain}/api/v5/markdown'),
|
|
headers: {'Authorization': 'token ${auth.token}'},
|
|
body: {'text': v},
|
|
);
|
|
return utf8.decode(res.bodyBytes).normalizedHtml;
|
|
});
|
|
final readmeData = MarkdownViewData(context, md: md, html: html);
|
|
final branches =
|
|
await auth.fetchGitee('/repos/$owner/$name/branches').then((v) {
|
|
return [for (var branch in v) GiteeBranch.fromJson(branch)];
|
|
});
|
|
bool isStarred = await auth
|
|
.fetchGitee('/user/starred/$owner/$name', requestType: 'NO CONTENT')
|
|
.then((v) => v.statusCode == HttpStatus.noContent);
|
|
bool isWatching = await auth
|
|
.fetchGitee('/user/subscriptions/$owner/$name',
|
|
requestType: 'NO CONTENT')
|
|
.then((v) => v.statusCode == HttpStatus.noContent);
|
|
StatusPayload statusPayload = StatusPayload(isWatching, isStarred);
|
|
return Tuple4(repo, readmeData, branches, statusPayload);
|
|
},
|
|
bodyBuilder: (t, setData) {
|
|
final p = t.item1;
|
|
final branches = t.item3;
|
|
final theme = context.read<ThemeModel>();
|
|
return Column(
|
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
|
children: <Widget>[
|
|
RepoHeader(
|
|
avatarUrl: p.owner!.avatarUrl,
|
|
avatarLink: '/gitee/${p.namespace!.path}',
|
|
owner: p.namespace!.path,
|
|
name: p.path,
|
|
description: p.description,
|
|
homepageUrl: p.homepage,
|
|
actions: [
|
|
Row(children: <Widget>[
|
|
MutationButton(
|
|
active: t.item4.isWatching,
|
|
text: t.item4.isWatching ? 'Ignore' : 'Watch',
|
|
onTap: () async {
|
|
final String watchType =
|
|
t.item4.isWatching ? 'ignoring' : 'watching';
|
|
await context.read<AuthModel>().fetchGitee(
|
|
'/user/subscriptions/$owner/$name?watch_type=$watchType',
|
|
requestType: t.item4.isWatching ? 'DELETE' : 'PUT');
|
|
|
|
t.item4.isWatching = !t.item4.isWatching;
|
|
setData(t);
|
|
},
|
|
),
|
|
SizedBox(width: 8),
|
|
MutationButton(
|
|
active: t.item4.isStarred,
|
|
text: t.item4.isStarred ? 'Unstar' : 'Star',
|
|
onTap: () async {
|
|
await context.read<AuthModel>().fetchGitee(
|
|
'/user/starred/$owner/$name',
|
|
requestType: t.item4.isStarred ? 'DELETE' : 'PUT');
|
|
|
|
t.item4.isStarred = !t.item4.isStarred;
|
|
setData(t);
|
|
},
|
|
)
|
|
])
|
|
]),
|
|
Row(
|
|
children: <Widget>[
|
|
EntryItem(
|
|
count: p.watchersCount,
|
|
text: 'Watchers',
|
|
url: '/gitee/$owner/$name/watchers',
|
|
),
|
|
EntryItem(
|
|
count: p.stargazersCount,
|
|
text: 'Stars',
|
|
url: '/gitee/$owner/$name/stargazers',
|
|
),
|
|
EntryItem(
|
|
count: p.forksCount,
|
|
text: 'Forks',
|
|
url: '/gitee/$owner/$name/forks',
|
|
),
|
|
],
|
|
),
|
|
CommonStyle.border,
|
|
TableView(
|
|
hasIcon: true,
|
|
items: [
|
|
TableViewItem(
|
|
leftIconData: Octicons.code,
|
|
text: Text('Code'),
|
|
rightWidget: Text(p.license ?? ''),
|
|
url: '/gitee/$owner/$name/tree/${branch ?? p.defaultBranch}',
|
|
),
|
|
TableViewItem(
|
|
leftIconData: Octicons.issue_opened,
|
|
text: Text('Issues'),
|
|
rightWidget: Text(numberFormat.format(p.openIssuesCount)),
|
|
url: '/gitee/$owner/$name/issues',
|
|
),
|
|
if (p.pullRequestsEnabled!)
|
|
TableViewItem(
|
|
leftIconData: Octicons.git_pull_request,
|
|
text: Text('Pull requests'),
|
|
url: '/gitee/$owner/$name/pulls',
|
|
),
|
|
TableViewItem(
|
|
leftIconData: Octicons.history,
|
|
text: Text('Commits'),
|
|
url:
|
|
'/gitee/$owner/$name/commits?branch=${branch ?? p.defaultBranch}',
|
|
),
|
|
TableViewItem(
|
|
leftIconData: Octicons.git_branch,
|
|
text: Text(AppLocalizations.of(context)!.branches),
|
|
rightWidget: Text((branch ?? p.defaultBranch)! +
|
|
' • ' +
|
|
branches.length.toString()),
|
|
onTap: () 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) {
|
|
theme.push(
|
|
context, '/gitee/$owner/$name?branch=$ref',
|
|
replace: true);
|
|
}
|
|
},
|
|
),
|
|
);
|
|
},
|
|
),
|
|
TableViewItem(
|
|
leftIconData: Octicons.organization,
|
|
text: Text('Contributors'),
|
|
url: '/gitee/$owner/$name/contributors'),
|
|
],
|
|
),
|
|
CommonStyle.verticalGap,
|
|
MarkdownView(t.item2)
|
|
],
|
|
);
|
|
},
|
|
);
|
|
}
|
|
}
|