2019-09-14 11:19:33 +02:00
|
|
|
import 'dart:io';
|
2019-01-28 14:32:01 +01:00
|
|
|
import 'package:flutter/material.dart';
|
2019-01-29 12:41:24 +01:00
|
|
|
import 'package:flutter/cupertino.dart';
|
|
|
|
import 'package:flutter/gestures.dart';
|
2019-09-23 14:33:27 +02:00
|
|
|
import 'package:git_touch/models/theme.dart';
|
2019-09-07 11:48:59 +02:00
|
|
|
import 'package:git_touch/screens/user.dart';
|
2019-10-02 08:58:11 +02:00
|
|
|
import 'package:git_touch/widgets/border_view.dart';
|
2019-10-06 09:50:22 +02:00
|
|
|
import 'package:git_touch/widgets/repository_item.dart';
|
|
|
|
import 'package:git_touch/widgets/table_view.dart';
|
2019-09-13 17:34:19 +02:00
|
|
|
import 'package:intl/intl.dart';
|
2019-08-31 15:37:29 +02:00
|
|
|
import 'package:primer/primer.dart';
|
2019-09-23 14:33:27 +02:00
|
|
|
import 'package:provider/provider.dart';
|
2019-11-02 16:56:10 +01:00
|
|
|
import 'package:tuple/tuple.dart';
|
2019-09-28 18:25:14 +02:00
|
|
|
import 'package:url_launcher/url_launcher.dart';
|
2019-09-25 07:34:13 +02:00
|
|
|
export 'package:flutter_vector_icons/flutter_vector_icons.dart';
|
2019-01-28 14:32:01 +01:00
|
|
|
|
2019-03-10 09:09:26 +01:00
|
|
|
class StorageKeys {
|
2019-09-26 16:14:14 +02:00
|
|
|
static const accounts = 'accounts';
|
2019-03-10 09:09:26 +01:00
|
|
|
static const account = 'account';
|
|
|
|
static const github = 'github';
|
|
|
|
static const theme = 'theme';
|
|
|
|
static const newsFilter = 'news.filter';
|
|
|
|
}
|
|
|
|
|
2019-10-02 10:09:54 +02:00
|
|
|
class CommonStyle {
|
|
|
|
static const padding = EdgeInsets.all(12);
|
2019-11-05 14:22:41 +01:00
|
|
|
static final border = BorderView();
|
2019-10-02 10:09:54 +02:00
|
|
|
static const verticalGap = SizedBox(height: 18);
|
|
|
|
static final monospace = Platform.isIOS ? 'Menlo' : 'monospace'; // FIXME:
|
|
|
|
}
|
|
|
|
|
2019-02-04 11:32:39 +01:00
|
|
|
Color convertColor(String cssHex) {
|
2019-09-02 12:14:17 +02:00
|
|
|
if (cssHex == null) {
|
|
|
|
return Color(0xffcccccc); // Default color
|
|
|
|
}
|
|
|
|
|
2019-02-04 11:32:39 +01:00
|
|
|
if (cssHex.startsWith('#')) {
|
|
|
|
cssHex = cssHex.substring(1);
|
|
|
|
}
|
2019-03-12 12:00:31 +01:00
|
|
|
if (cssHex.length == 3) {
|
|
|
|
cssHex = cssHex.split('').map((char) => char + char).join('');
|
|
|
|
}
|
2019-02-04 11:32:39 +01:00
|
|
|
return Color(int.parse('ff' + cssHex, radix: 16));
|
|
|
|
}
|
|
|
|
|
2019-09-14 09:39:38 +02:00
|
|
|
Color getFontColorByBrightness(Color color) {
|
|
|
|
var grayscale = color.red * 0.3 + color.green * 0.59 + color.blue * 0.11;
|
2019-10-13 05:01:29 +02:00
|
|
|
// Fimber.d('color: $color, $grayscale');
|
2019-09-14 09:39:38 +02:00
|
|
|
|
|
|
|
var showWhite = grayscale < 128;
|
|
|
|
return showWhite ? Colors.white : Colors.black;
|
|
|
|
}
|
|
|
|
|
2019-02-10 05:16:52 +01:00
|
|
|
void nextTick(Function callback, [int milliseconds = 0]) {
|
2019-02-08 16:20:28 +01:00
|
|
|
// FIXME:
|
2019-02-10 05:16:52 +01:00
|
|
|
Future.delayed(Duration(milliseconds: 0)).then((_) {
|
2019-02-08 03:48:25 +01:00
|
|
|
callback();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-09-23 14:33:27 +02:00
|
|
|
TextSpan createLinkSpan(
|
|
|
|
BuildContext context,
|
|
|
|
String text,
|
|
|
|
Widget Function(BuildContext) builder,
|
|
|
|
) {
|
2019-01-29 12:41:24 +01:00
|
|
|
return TextSpan(
|
|
|
|
text: text,
|
2019-09-24 13:58:34 +02:00
|
|
|
style: TextStyle(
|
|
|
|
color: PrimerColors.blue500,
|
|
|
|
fontWeight: FontWeight.w600,
|
|
|
|
),
|
2019-02-11 17:31:06 +01:00
|
|
|
recognizer: TapGestureRecognizer()
|
|
|
|
..onTap = () {
|
2019-09-23 14:33:27 +02:00
|
|
|
Provider.of<ThemeModel>(context).pushRoute(context, builder);
|
2019-02-11 17:31:06 +01:00
|
|
|
},
|
2019-01-29 12:41:24 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-09-07 11:48:59 +02:00
|
|
|
TextSpan createUserSpan(BuildContext context, String login) {
|
2019-09-23 14:33:27 +02:00
|
|
|
return createLinkSpan(context, login, (_) => UserScreen(login));
|
2019-09-07 11:48:59 +02:00
|
|
|
}
|
|
|
|
|
2019-11-02 16:56:10 +01:00
|
|
|
Tuple2<String, String> parseRepositoryFullName(String fullName) {
|
|
|
|
final ls = fullName.split('/');
|
|
|
|
assert(ls.length == 2);
|
|
|
|
return Tuple2(ls[0], ls[1]);
|
|
|
|
}
|
|
|
|
|
2019-11-05 14:22:41 +01:00
|
|
|
class GithubPalette {
|
2019-11-08 13:23:09 +01:00
|
|
|
static const open = Color(0xff2cbe4e);
|
|
|
|
static const closed = PrimerColors.red600;
|
|
|
|
static const merged = PrimerColors.purple500;
|
2019-01-28 14:32:01 +01:00
|
|
|
}
|
|
|
|
|
2019-02-19 08:25:15 +01:00
|
|
|
// final pageSize = 5;
|
2019-02-09 07:20:21 +01:00
|
|
|
final pageSize = 30;
|
2019-01-28 14:32:01 +01:00
|
|
|
|
2019-01-29 06:34:52 +01:00
|
|
|
var createWarning =
|
|
|
|
(String text) => Text(text, style: TextStyle(color: Colors.redAccent));
|
2019-01-28 14:32:01 +01:00
|
|
|
var warningSpan =
|
|
|
|
TextSpan(text: 'xxx', style: TextStyle(color: Colors.redAccent));
|
2019-02-09 06:36:15 +01:00
|
|
|
|
2019-09-03 15:39:00 +02:00
|
|
|
List<T> join<T>(T seperator, List<T> xs) {
|
|
|
|
List<T> result = [];
|
|
|
|
xs.asMap().forEach((index, x) {
|
|
|
|
if (x == null) return;
|
|
|
|
|
|
|
|
result.add(x);
|
|
|
|
if (index < xs.length - 1) {
|
|
|
|
result.add(seperator);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
List<T> joinAll<T>(T seperator, List<List<T>> xss) {
|
|
|
|
List<T> result = [];
|
|
|
|
xss.asMap().forEach((index, x) {
|
|
|
|
if (x == null || x.isEmpty) return;
|
|
|
|
|
|
|
|
result.addAll(x);
|
|
|
|
if (index < xss.length - 1) {
|
|
|
|
result.add(seperator);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2019-09-13 17:34:19 +02:00
|
|
|
final numberFormat = NumberFormat();
|
|
|
|
|
2019-09-14 17:48:01 +02:00
|
|
|
bool isNotNullOrEmpty(String text) {
|
|
|
|
return text != null && text.isNotEmpty;
|
|
|
|
}
|
|
|
|
|
2019-09-23 14:04:53 +02:00
|
|
|
String getBranchQueryKey(String branch, {bool withParams = false}) {
|
|
|
|
if (branch == null) return 'defaultBranchRef';
|
|
|
|
return 'ref' + (withParams ? '(qualifiedName: "$branch")' : '');
|
|
|
|
}
|
2019-09-25 07:34:13 +02:00
|
|
|
|
|
|
|
// TODO: Primer
|
|
|
|
class PrimerBranchName extends StatelessWidget {
|
|
|
|
final String name;
|
|
|
|
|
|
|
|
PrimerBranchName(this.name);
|
|
|
|
|
|
|
|
static const branchBgColor = Color(0xffeaf5ff);
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Container(
|
|
|
|
padding: EdgeInsets.symmetric(vertical: 2, horizontal: 4),
|
|
|
|
height: 16,
|
|
|
|
decoration: BoxDecoration(
|
|
|
|
color: branchBgColor,
|
|
|
|
borderRadius: BorderRadius.all(Radius.circular(3)),
|
|
|
|
),
|
|
|
|
child: Text(
|
|
|
|
name,
|
|
|
|
style: TextStyle(
|
|
|
|
color: PrimerColors.blue500,
|
|
|
|
fontSize: 12,
|
|
|
|
height: 1,
|
2019-10-02 10:09:54 +02:00
|
|
|
fontFamily: CommonStyle.monospace,
|
2019-09-25 07:34:13 +02:00
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2019-09-29 07:32:53 +02:00
|
|
|
|
|
|
|
launchUrl(String url) async {
|
|
|
|
if (url == null) return;
|
|
|
|
|
|
|
|
if (await canLaunch(url)) {
|
|
|
|
await launch(url);
|
|
|
|
} else {
|
|
|
|
// TODO: fallback
|
|
|
|
}
|
|
|
|
}
|
2019-10-06 09:50:22 +02:00
|
|
|
|
|
|
|
Iterable<Widget> buildPinnedItems(List pinnedItems, List repositories) {
|
|
|
|
String title;
|
|
|
|
List items = [];
|
|
|
|
|
|
|
|
if (pinnedItems.isNotEmpty) {
|
|
|
|
title = 'pinned repositories';
|
|
|
|
items = pinnedItems;
|
|
|
|
} else if (repositories.isNotEmpty) {
|
|
|
|
title = 'popular repositories';
|
|
|
|
items = repositories;
|
|
|
|
}
|
|
|
|
|
|
|
|
items = items
|
|
|
|
.where((x) => x.isNotEmpty)
|
|
|
|
.toList(); // TODO: Pinned items may include Gist
|
|
|
|
if (items.isEmpty) return [];
|
|
|
|
|
|
|
|
return [
|
|
|
|
CommonStyle.verticalGap,
|
|
|
|
if (title != null) TableViewHeader(title),
|
|
|
|
...join(
|
|
|
|
CommonStyle.border,
|
|
|
|
items.map((item) {
|
|
|
|
return RepositoryItem(item);
|
|
|
|
}).toList(),
|
|
|
|
),
|
|
|
|
];
|
|
|
|
}
|