完成登录和本地登出

main
落雨楓 2 years ago
parent 28d9d832c3
commit a9e70fccac

@ -3,20 +3,21 @@ import 'dart:convert';
import 'package:dio/dio.dart'; import 'package:dio/dio.dart';
import 'package:dio_cache_interceptor/dio_cache_interceptor.dart'; import 'package:dio_cache_interceptor/dio_cache_interceptor.dart';
import 'package:dio_cookie_manager/dio_cookie_manager.dart'; import 'package:dio_cookie_manager/dio_cookie_manager.dart';
import 'package:cookie_jar/cookie_jar.dart';
import 'package:dio_http2_adapter/dio_http2_adapter.dart'; import 'package:dio_http2_adapter/dio_http2_adapter.dart';
import 'package:cookie_jar/cookie_jar.dart';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:isekai_wiki/global.dart'; import 'package:isekai_wiki/global.dart';
import 'package:isekai_wiki/utils/api_utils.dart'; import 'package:path_provider/path_provider.dart';
class HttpResponseCodeError extends Error { class HttpResponseException implements Exception {
int? statusCode; int statusCode;
String? statusText;
HttpResponseCodeError(this.statusCode); HttpResponseException(this.statusCode, {this.statusText});
@override @override
String toString() { String toString() {
return "Http error: $statusCode"; return "Http error: $statusCode $statusText";
} }
} }
@ -24,26 +25,26 @@ class BaseApi {
static Dio? _dioInstance; static Dio? _dioInstance;
static CookieJar? cookieJar; static CookieJar? cookieJar;
static Dio createClient() { static Future<Dio> createClient() async {
var dio = Dio(); var dio = Dio();
return dio;
}
static Dio getClient() {
_dioInstance ??= createClient();
if (!kIsWeb) { if (!kIsWeb) {
// HTTP2 // HTTP2
_dioInstance!.httpClientAdapter = Http2Adapter( /*
dio.httpClientAdapter = Http2Adapter(
ConnectionManager( ConnectionManager(
idleTimeout: 10000, idleTimeout: 10000,
), ),
); );
*/
// Cookie // Cookie
cookieJar = PersistCookieJar(); var tempDir = await getTemporaryDirectory();
_dioInstance!.interceptors.add(CookieManager(cookieJar!));
cookieJar = PersistCookieJar(
storage: FileStorage("${tempDir.path}/cookies"),
);
dio.interceptors.add(CookieManager(cookieJar!));
// //
final cacheOptions = CacheOptions( final cacheOptions = CacheOptions(
@ -55,11 +56,10 @@ class BaseApi {
keyBuilder: CacheOptions.defaultCacheKeyBuilder, keyBuilder: CacheOptions.defaultCacheKeyBuilder,
allowPostMethod: false, allowPostMethod: false,
); );
_dioInstance!.interceptors dio.interceptors.add(DioCacheInterceptor(options: cacheOptions));
.add(DioCacheInterceptor(options: cacheOptions));
// Header // Header
_dioInstance!.interceptors.add( dio.interceptors.add(
InterceptorsWrapper(onRequest: (options, handler) { InterceptorsWrapper(onRequest: (options, handler) {
options.headers["X-IsekaiWikiApp-Version"] = options.headers["X-IsekaiWikiApp-Version"] =
Global.packageInfo?.version ?? "unknow"; Global.packageInfo?.version ?? "unknow";
@ -69,24 +69,37 @@ class BaseApi {
); );
} }
return dio;
}
static Future<Dio> getClient() async {
_dioInstance ??= await createClient();
return _dioInstance!; return _dioInstance!;
} }
static Future<void> clearCookie() async {
await cookieJar?.deleteAll();
}
static Future<String> get(Uri uri, {Map<String, dynamic>? search}) async { static Future<String> get(Uri uri, {Map<String, dynamic>? search}) async {
var res = await getClient().get<String>( var client = await getClient();
var res = await client.get<String>(
uri.toString(), uri.toString(),
queryParameters: search,
options: Options(responseType: ResponseType.plain), options: Options(responseType: ResponseType.plain),
); );
if (res.statusCode != 200) { if (res.statusCode != null && res.statusCode != 200) {
throw HttpResponseCodeError(res.statusCode); throw HttpResponseException(res.statusCode!,
statusText: res.statusMessage!);
} }
return res.data ?? ""; return res.data ?? "";
} }
static Future<Map> getJson(Uri uri) async { static Future<Map> getJson(Uri uri, {Map<String, dynamic>? search}) async {
var resText = await get(uri); var resText = await get(uri, search: search);
var resData = jsonDecode(resText); var resData = jsonDecode(resText);
if (resData is Map) { if (resData is Map) {

@ -19,21 +19,9 @@ class MWApiList {
} }
var mwRes = await MWApi.get("query", query: query); var mwRes = await MWApi.get("query", query: query);
var rcRes = RecentChangesResponse.fromJson(mwRes.data);
if (!mwRes.ok) { return mwRes.replaceData(rcRes.recentchanges);
return MWResponse(errorList: mwRes.errorList);
}
if (mwRes.data != null) {
var rcRes = RecentChangesResponse.fromJson(mwRes.data!);
return MWResponse(
ok: true,
data: rcRes.recentchanges,
continueInfo: mwRes.continueInfo);
} else {
return MWResponse(
errorList: [MWError(code: 'response_data_empty', info: '加载的数据为空')]);
}
} }
/// ///
@ -61,13 +49,11 @@ class MWApiList {
var rcResList = await Future.wait([ var rcResList = await Future.wait([
ignoreRcNew ignoreRcNew
? Future.value( ? Future.value(MWResponse<List<RecentChangesItem>>([]))
MWResponse<List<RecentChangesItem>>(ok: true, data: []))
: getRecentChanges("new", : getRecentChanges("new",
limit: limit, continueInfo: continueInfoNew), limit: limit, continueInfo: continueInfoNew),
ignoreRcEdit ignoreRcEdit
? Future.value( ? Future.value(MWResponse<List<RecentChangesItem>>([]))
MWResponse<List<RecentChangesItem>>(ok: true, data: []))
: getRecentChanges("edit", : getRecentChanges("edit",
limit: limit, continueInfo: continueInfoEdit), limit: limit, continueInfo: continueInfoEdit),
]); ]);
@ -75,16 +61,7 @@ class MWApiList {
var rcNewRes = rcResList[0]; var rcNewRes = rcResList[0];
var rcEditRes = rcResList[1]; var rcEditRes = rcResList[1];
if (!rcNewRes.ok) { List<RecentChangesItem> mergedList = [...rcNewRes.data, ...rcEditRes.data];
return MWResponse(errorList: rcNewRes.errorList);
} else if (!rcEditRes.ok) {
return MWResponse(errorList: rcNewRes.errorList);
}
List<RecentChangesItem> mergedList = [
...rcNewRes.data!,
...rcEditRes.data!
];
mergedList.sort((a, b) { mergedList.sort((a, b) {
// 7 // 7
@ -120,8 +97,7 @@ class MWApiList {
} }
return MWResponse( return MWResponse(
ok: true, uniqueMergedList,
data: uniqueMergedList,
continueInfo: mergedContinueInfo.isNotEmpty ? mergedContinueInfo : null, continueInfo: mergedContinueInfo.isNotEmpty ? mergedContinueInfo : null,
); );
} }
@ -165,54 +141,44 @@ class MWApiList {
var mwRes = await MWApi.get("query", query: query); var mwRes = await MWApi.get("query", query: query);
if (!mwRes.ok) { var pagesRes = PagesResponse.fromJson(mwRes.data);
return MWResponse(errorList: mwRes.errorList);
}
if (mwRes.data != null) {
var pagesRes = PagesResponse.fromJson(mwRes.data!);
var pageList = pagesRes.pages.map((pageInfo) { var pageList = pagesRes.pages.map((pageInfo) {
if (pageInfo.description != null) { if (pageInfo.description != null) {
pageInfo.description = pageInfo.description =
pageInfo.description!.replaceAll(RegExp(r"\n\n"), "\n"); pageInfo.description!.replaceAll(RegExp(r"\n\n"), "\n");
}
if (pageInfo.title.contains("/")) {
var splitPos = pageInfo.title.lastIndexOf("/");
pageInfo.displayTitle = pageInfo.title.substring(splitPos + 1);
pageInfo.subtitle = pageInfo.title.substring(0, splitPos);
} else {
pageInfo.displayTitle = pageInfo.title;
}
return pageInfo;
}).toList();
//
List<PageInfo> sortedPages = [];
if (pageids != null) {
for (var pageid in pageids) {
var index =
pageList.indexWhere((element) => element.pageid == pageid);
if (index != -1) {
sortedPages.add(pageList[index]);
}
}
} }
if (pageInfo.title.contains("/")) {
var splitPos = pageInfo.title.lastIndexOf("/");
pageInfo.displayTitle = pageInfo.title.substring(splitPos + 1);
pageInfo.subtitle = pageInfo.title.substring(0, splitPos);
} else {
pageInfo.displayTitle = pageInfo.title;
}
return pageInfo;
}).toList();
//
List<PageInfo> sortedPages = [];
if (titles != null) { if (pageids != null) {
for (var title in titles) { for (var pageid in pageids) {
var index = var index = pageList.indexWhere((element) => element.pageid == pageid);
pagesRes.pages.indexWhere((element) => element.title == title); if (index != -1) {
if (index != -1) { sortedPages.add(pageList[index]);
sortedPages.add(pagesRes.pages[index]);
}
} }
} }
}
return MWResponse( if (titles != null) {
ok: true, data: sortedPages, continueInfo: mwRes.continueInfo); for (var title in titles) {
} else { var index =
return MWResponse( pagesRes.pages.indexWhere((element) => element.title == title);
errorList: [MWError(code: 'response_data_empty', info: '加载的数据为空')]); if (index != -1) {
sortedPages.add(pagesRes.pages[index]);
}
}
} }
return mwRes.replaceData(sortedPages);
} }
} }

@ -1,88 +1,64 @@
import 'dart:convert'; import 'dart:convert';
import 'dart:io'; import 'package:dio/dio.dart';
import 'package:flutter/foundation.dart';
import 'package:http/http.dart' as http;
import 'package:isekai_wiki/api/base_api.dart'; import 'package:isekai_wiki/api/base_api.dart';
import 'package:isekai_wiki/global.dart'; import 'package:isekai_wiki/global.dart';
import 'package:json_annotation/json_annotation.dart';
import '../../utils/api_utils.dart'; part 'mw_api.g.dart';
@JsonSerializable()
class MWApiErrorException implements Exception {
String code;
class MWError {
String? code;
String? info; String? info;
@JsonKey(name: "*")
String? detail; String? detail;
MWError({this.code, this.info, this.detail}); MWApiErrorException({required this.code, this.info, this.detail});
static MWError fromMap(Map errorMap) { factory MWApiErrorException.fromJson(Map<String, dynamic> json) =>
var mwError = MWError(); _$MWApiExceptionFromJson(json);
if (errorMap.containsKey("code")) {
mwError.code = errorMap["code"].toString();
}
if (errorMap.containsKey("info")) { Map<String, dynamic> toJson() => _$MWApiExceptionToJson(this);
mwError.info = errorMap["info"].toString();
}
if (errorMap.containsKey("*")) { @override
mwError.detail = errorMap["*"].toString(); String toString() {
} return "MediaWiki Api Error: ${info ?? code}";
return mwError;
} }
} }
class MWMultiError extends Error { class MWApiEmptyBodyException implements Exception {
List<MWError> errorList; dynamic resData;
MWMultiError(this.errorList); MWApiEmptyBodyException(this.resData);
@override @override
String toString() { String toString() {
return errorList.map((e) => e.info).join("\n"); return "MediaWiki Api Response body is empty";
} }
} }
class MWResponse<T> { class MWResponse<T> {
bool ok = false; List<dynamic>? warnList;
List<MWError>? errorList; T data;
T? data;
Map<String, String>? continueInfo; Map<String, String>? continueInfo;
MWResponse({this.ok = false, this.errorList, this.data, this.continueInfo}); MWResponse(this.data, {this.warnList, this.continueInfo});
}
class MWApiClient extends http.BaseClient {
final http.Client _inner;
MWApiClient(this._inner);
Future<http.StreamedResponse> send(http.BaseRequest request) async { MWResponse<N> replaceData<N>(N data) {
request.headers['user-agent'] = await ApiUtils.getUserAgent(); return MWResponse<N>(
return await _inner.send(request); data,
warnList: warnList,
continueInfo: continueInfo,
);
} }
} }
class MWApi { class MWApi {
static Uri apiBaseUri = Uri.parse(Global.wikiApiUrl); static Uri apiBaseUri = Uri.parse(Global.wikiApiUrl);
static HttpClient getHttpClient() {
return HttpClient();
}
static Future<Map<String, String>> _getHeaders() async {
var headers = {
"X-IsekaiWikiApp-Version": Global.packageInfo?.version ?? "unknow",
};
if (!kIsWeb) {
headers["User-Agent"] = await ApiUtils.getUserAgent();
}
return headers;
}
static Future<MWResponse<Map<String, dynamic>>> get(String action, static Future<MWResponse<Map<String, dynamic>>> get(String action,
{Map<String, dynamic>? query}) async { {Map<String, dynamic>? query}) async {
Map<String, String> queryStr = Map<String, String> queryStr =
@ -97,58 +73,62 @@ class MWApi {
queryStr["origin"] = Global.webOrigin!; queryStr["origin"] = Global.webOrigin!;
} }
Uri requestUri = apiBaseUri.replace(queryParameters: queryStr); var resText = "";
try {
var res = await http.get(requestUri, headers: await _getHeaders()); resText = await BaseApi.get(apiBaseUri, search: queryStr);
} on DioError catch (err) {
if (err.type == DioErrorType.response) {
if (err.response != null) {
var response = err.response!;
var contentType = response.headers[Headers.contentTypeHeader];
if (contentType != null &&
contentType.contains("application/json") &&
response.data is String) {
resText = response.data as String;
}
}
}
if (res.statusCode != 200) { if (resText.isEmpty) {
throw HttpResponseCodeError(res.statusCode); //
rethrow;
}
} }
var responseBody = res.body; return parseMWResponse(action, resText);
return parseMWResponse(action, responseBody);
} }
static MWResponse<Map<String, dynamic>> parseMWResponse(String action, String resJson) { static MWResponse<Map<String, dynamic>> parseMWResponse(
var mwRes = MWResponse<Map<String, dynamic>>(); String action, String resJson) {
List<MWError> errorList = [];
var resData = jsonDecode(resJson); var resData = jsonDecode(resJson);
if (resData is Map) { if (resData is! Map<String, dynamic>) {
// throw MWApiEmptyBodyException(resData);
var resError = resData["error"]; }
if (resError is Map) {
errorList.add(MWError.fromMap(resError));
} else if (resError is List) {
for (var errItem in resError) {
if (errItem is Map) {
errorList.add(MWError.fromMap(errItem));
}
}
}
if (errorList.isNotEmpty) {
mwRes.errorList = errorList;
return mwRes;
}
// //
if (resData.containsKey(action) && resData[action] is Map) { if (resData.containsKey("error")) {
mwRes.data = resData[action] as Map<String, dynamic>; throw MWApiErrorException.fromJson(resData["error"]!);
mwRes.ok = true; }
}
// //
var batchcomplete = resData["batchcomplete"]; if (!resData.containsKey(action) ||
if (batchcomplete is bool && batchcomplete) { resData[action] is! Map<String, dynamic>) {
var continueInfo = resData["continue"]; throw MWApiEmptyBodyException(resData);
if (continueInfo is Map) { }
mwRes.continueInfo = {};
continueInfo.forEach((key, value) { MWResponse<Map<String, dynamic>> mwRes =
var keyStr = key.toString(); MWResponse(resData[action] as Map<String, dynamic>);
mwRes.continueInfo![keyStr] = value.toString();
}); // continue
} var batchcomplete = resData["batchcomplete"];
if (batchcomplete is bool && batchcomplete) {
var continueInfo = resData["continue"];
if (continueInfo is Map<String, dynamic>) {
mwRes.continueInfo = {};
continueInfo.forEach((key, value) {
var keyStr = key;
mwRes.continueInfo![keyStr] = value.toString();
});
} }
} }

@ -0,0 +1,21 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'mw_api.dart';
// **************************************************************************
// JsonSerializableGenerator
// **************************************************************************
MWApiErrorException _$MWApiExceptionFromJson(Map<String, dynamic> json) =>
MWApiErrorException(
code: json['code'] as String,
info: json['info'] as String?,
detail: json['*'] as String?,
);
Map<String, dynamic> _$MWApiExceptionToJson(MWApiErrorException instance) =>
<String, dynamic>{
'code': instance.code,
'info': instance.info,
'*': instance.detail,
};

@ -0,0 +1 @@
class MWApiToken {}

@ -11,18 +11,9 @@ class MWApiUser {
var mwRes = await MWApi.get("mugenapp", query: query); var mwRes = await MWApi.get("mugenapp", query: query);
if (!mwRes.ok) { var data = MugenAppStartAuthResponse.fromJson(mwRes.data);
return MWResponse(errorList: mwRes.errorList);
}
if (mwRes.data != null) {
var authRes = MugenAppStartAuthResponse.fromJson(mwRes.data!);
return MWResponse( return mwRes.replaceData(data.startauth);
ok: true, data: authRes.startauth, continueInfo: mwRes.continueInfo);
} else {
return MWResponse(
errorList: [MWError(code: 'response_data_empty', info: '加载的数据为空')]);
}
} }
static Future<MWResponse<MugenAppAttemptAuthInfo>> attemptAuth( static Future<MWResponse<MugenAppAttemptAuthInfo>> attemptAuth(
@ -34,20 +25,9 @@ class MWApiUser {
var mwRes = await MWApi.get("mugenapp", query: query); var mwRes = await MWApi.get("mugenapp", query: query);
if (!mwRes.ok) { var data = MugenAppAttemptAuthResponse.fromJson(mwRes.data);
return MWResponse(errorList: mwRes.errorList);
}
if (mwRes.data != null) {
var authRes = MugenAppAttemptAuthResponse.fromJson(mwRes.data!);
return MWResponse( return mwRes.replaceData(data.attemptauth);
ok: true,
data: authRes.attemptauth,
continueInfo: mwRes.continueInfo);
} else {
return MWResponse(
errorList: [MWError(code: 'response_data_empty', info: '加载的数据为空')]);
}
} }
static Future<MWResponse<MetaUserInfoResponse>> getCurrentUserInfo() async { static Future<MWResponse<MetaUserInfoResponse>> getCurrentUserInfo() async {
@ -58,16 +38,8 @@ class MWApiUser {
var mwRes = await MWApi.get("query", query: query); var mwRes = await MWApi.get("query", query: query);
if (!mwRes.ok) { var data = MetaUserInfoResponse.fromJson(mwRes.data);
return MWResponse(errorList: mwRes.errorList);
}
if (mwRes.data != null) {
var userInfoRes = MetaUserInfoResponse.fromJson(mwRes.data!);
return MWResponse(ok: true, data: userInfoRes); return mwRes.replaceData(data);
} else {
return MWResponse(
errorList: [MWError(code: 'response_data_empty', info: '加载的数据为空')]);
}
} }
} }

@ -8,8 +8,8 @@ class MWParseCategoryInfo {
String category; String category;
MWParseCategoryInfo({ MWParseCategoryInfo({
required this.category,
this.sortkey = "", this.sortkey = "",
this.category = "",
}); });
factory MWParseCategoryInfo.fromJson(Map<String, dynamic> json) => factory MWParseCategoryInfo.fromJson(Map<String, dynamic> json) =>
@ -27,11 +27,11 @@ class MWParseLangLinkInfo {
String title; String title;
MWParseLangLinkInfo({ MWParseLangLinkInfo({
this.lang = "", required this.lang,
this.url = "", this.url = "",
this.langname = "", this.langname = "",
this.autonym = "", this.autonym = "",
this.title = "", required this.title,
}); });
factory MWParseLangLinkInfo.fromJson(Map<String, dynamic> json) => factory MWParseLangLinkInfo.fromJson(Map<String, dynamic> json) =>
@ -47,8 +47,8 @@ class MWParsePageLinkInfo {
bool exists; bool exists;
MWParsePageLinkInfo({ MWParsePageLinkInfo({
this.ns = -1, required this.ns,
this.title = "", required this.title,
this.exists = false, this.exists = false,
}); });
@ -109,8 +109,8 @@ class MWParseInfo {
Map<String, dynamic> properties; Map<String, dynamic> properties;
MWParseInfo({ MWParseInfo({
this.title = "", required this.title,
this.pageid = -1, required this.pageid,
this.revid = -1, this.revid = -1,
this.text = "", this.text = "",
this.langlink = const [], this.langlink = const [],
@ -130,7 +130,8 @@ class MWParseInfo {
this.properties = const {}, this.properties = const {},
}); });
factory MWParseInfo.fromJson(Map<String, dynamic> json) => _$MWParseInfoFromJson(json); factory MWParseInfo.fromJson(Map<String, dynamic> json) =>
_$MWParseInfoFromJson(json);
Map<String, dynamic> toJson() => _$MWParseInfoToJson(this); Map<String, dynamic> toJson() => _$MWParseInfoToJson(this);
} }
@ -141,7 +142,8 @@ class MWParseResponse {
MWParseResponse({required this.parse}); MWParseResponse({required this.parse});
factory MWParseResponse.fromJson(Map<String, dynamic> json) => _$MWParseResponseFromJson(json); factory MWParseResponse.fromJson(Map<String, dynamic> json) =>
_$MWParseResponseFromJson(json);
Map<String, dynamic> toJson() => _$MWParseResponseToJson(this); Map<String, dynamic> toJson() => _$MWParseResponseToJson(this);
} }

@ -18,7 +18,7 @@ class RecentPageListController extends GetxController {
var pageList = RxList<PageInfo>(); var pageList = RxList<PageInfo>();
var continueInfo = RxMap<String, String>(); var continueInfo = RxMap<String, String>();
var errorList = RxList<MWError>(); var errorList = RxList<MWApiErrorException>();
var hasNextPage = true.obs; var hasNextPage = true.obs;
var isLoading = false.obs; var isLoading = false.obs;
@ -28,7 +28,8 @@ class RecentPageListController extends GetxController {
@override @override
void onInit() { void onInit() {
scrollController?.addListener(() { scrollController?.addListener(() {
if (scrollController!.position.pixels > scrollController!.position.maxScrollExtent - 10) { if (scrollController!.position.pixels >
scrollController!.position.maxScrollExtent - 10) {
// //
if (hasNextPage.value && !isLoading.value) { if (hasNextPage.value && !isLoading.value) {
loadNextPages(); loadNextPages();
@ -70,49 +71,38 @@ class RecentPageListController extends GetxController {
isLoading.value = true; isLoading.value = true;
var rcListRes = await MWApiList.getMixedRecentChanges( try {
limit: 10, continueInfo: continueInfo.isNotEmpty ? continueInfo : null); var rcListRes = await MWApiList.getMixedRecentChanges(
if (!rcListRes.ok) { limit: 10,
// continueInfo: continueInfo.isNotEmpty ? continueInfo : null);
errorList.value = rcListRes.errorList ?? [];
isLoading.value = false; var pageIds = rcListRes.data.map((rcInfo) => rcInfo.pageid).toList();
hasNextPage.value = false; var pageListRes = await MWApiList.getPageInfoList(
if (shouldRefresh) { pageids: pageIds, extractChars: 515, getInWatchlist: uc!.isLoggedIn);
pageList.clear();
if (uc!.isLoggedIn) {
//
flc!.updateFromPageList(pageListRes.data);
} }
return;
}
var pageIds = rcListRes.data!.map((rcInfo) => rcInfo.pageid).toList(); //
var pageListRes = await MWApiList.getPageInfoList( if (shouldRefresh) {
pageids: pageIds, extractChars: 515, getInWatchlist: uc!.isLoggedIn); //
if (!pageListRes.ok) { pageList.value = pageListRes.data;
// shouldRefresh = false;
errorList.value = pageListRes.errorList ?? []; } else {
isLoading.value = false; pageList.addAll(pageListRes.data);
}
hasNextPage.value = rcListRes.continueInfo != null;
continueInfo.value = rcListRes.continueInfo ?? {};
} catch (err) {
hasNextPage.value = false; hasNextPage.value = false;
if (shouldRefresh) { if (shouldRefresh) {
pageList.clear(); pageList.clear();
} }
return; } finally {
} isLoading.value = false;
if (uc!.isLoggedIn) {
//
flc!.updateFromPageList(pageListRes.data!);
}
//
if (shouldRefresh) {
//
pageList.value = pageListRes.data!;
shouldRefresh = false;
} else {
pageList.addAll(pageListRes.data!);
} }
hasNextPage.value = rcListRes.continueInfo != null;
continueInfo.value = rcListRes.continueInfo ?? {};
isLoading.value = false;
} }
Future<void> reload() async { Future<void> reload() async {
@ -138,13 +128,14 @@ class RecentPageList extends StatelessWidget {
return Column( return Column(
key: key, key: key,
children: [ children: [
for (var i = 0; i < 6; i++) PageCard(key: ValueKey("rpl-card-$i"), isLoading: true), for (var i = 0; i < 6; i++)
PageCard(key: ValueKey("rpl-card-$i"), isLoading: true),
], ],
); );
} }
Widget _buildPageCard( Widget _buildPageCard(int index, PageInfo pageInfo,
int index, PageInfo pageInfo, RecentPageListController c, FavoriteListController fc) { RecentPageListController c, FavoriteListController fc) {
return PageCard( return PageCard(
key: ValueKey("rpl-card-$index"), key: ValueKey("rpl-card-$index"),
pageInfo: c.pageList[index], pageInfo: c.pageList[index],

@ -4,9 +4,13 @@ import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_web_browser/flutter_web_browser.dart'; import 'package:flutter_web_browser/flutter_web_browser.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:isekai_wiki/api/base_api.dart';
import 'package:isekai_wiki/api/mw/mw_api.dart';
import 'package:isekai_wiki/api/mw/user.dart'; import 'package:isekai_wiki/api/mw/user.dart';
import 'package:isekai_wiki/api/response/mugenapp.dart';
import 'package:isekai_wiki/global.dart'; import 'package:isekai_wiki/global.dart';
import 'package:isekai_wiki/utils/dialog.dart'; import 'package:isekai_wiki/utils/dialog.dart';
import 'package:isekai_wiki/utils/error.dart';
import 'package:json_annotation/json_annotation.dart'; import 'package:json_annotation/json_annotation.dart';
part 'user.g.dart'; part 'user.g.dart';
@ -66,18 +70,29 @@ class UserController extends GetxController {
/// ///
Future<void> updateProfile() async { Future<void> updateProfile() async {
var userInfoMWRes = await MWApiUser.getCurrentUserInfo(); try {
if (!userInfoMWRes.ok) { var userInfoMWRes = await MWApiUser.getCurrentUserInfo();
if (kDebugMode) {
print("Cannot update profile of current user"); var userInfoRes = userInfoMWRes.data;
print(userInfoMWRes.errorList);
if (userInfoRes.userinfo.id == 0) {
//
//
//
logout(logoutRemote: false);
} }
}
var userInfoRes = userInfoMWRes.data!; userId.value = userInfoRes.userinfo.id;
nickName.value = userInfoRes.userinfo.realname ?? ""; userName.value = userInfoRes.userinfo.name;
if (userInfoRes.useravatar != null) { nickName.value = userInfoRes.userinfo.realname ?? "";
avatarUrlSet.value = userInfoRes.useravatar!; if (userInfoRes.useravatar != null) {
avatarUrlSet.value = userInfoRes.useravatar!;
}
} catch (err, stack) {
if (kDebugMode) {
printError(info: "Cannot update profile of current user: $err");
stack.printError();
}
} }
} }
@ -158,29 +173,34 @@ class UserController extends GetxController {
Future<void> startAuthFlow() async { Future<void> startAuthFlow() async {
authProcessing.value = true; authProcessing.value = true;
var startAuthRes = await MWApiUser.startAuth(); try {
if (!startAuthRes.ok) { var startAuthRes = await MWApiUser.startAuth();
var startAuthInfo = startAuthRes.data;
loginRequestToken.value = startAuthInfo.loginRequestKey;
await FlutterWebBrowser.openWebPage(
url: startAuthInfo.loginUrl,
customTabsOptions: const CustomTabsOptions(
defaultColorSchemeParams: CustomTabsColorSchemeParams(
toolbarColor: Color.fromRGBO(33, 37, 41, 1),
),
shareState: CustomTabsShareState.off,
showTitle: true,
),
safariVCOptions: const SafariViewControllerOptions(
barCollapsingEnabled: true,
));
} catch (err, stack) {
authProcessing.value = false; authProcessing.value = false;
alert(Get.overlayContext!, startAuthRes.errorList?[0].info ?? "未知错误", loginRequestToken.value = "";
title: "错误");
return;
}
var startAuthInfo = startAuthRes.data!;
loginRequestToken.value = startAuthInfo.loginRequestKey;
await FlutterWebBrowser.openWebPage( alert(Get.overlayContext!, ErrorUtils.getErrorMessage(err), title: "错误");
url: startAuthInfo.loginUrl, if (kDebugMode) {
customTabsOptions: const CustomTabsOptions( print("Exception in startAuthFlow: $err");
defaultColorSchemeParams: CustomTabsColorSchemeParams( stack.printError();
toolbarColor: Color.fromRGBO(33, 37, 41, 1), }
), }
shareState: CustomTabsShareState.off,
showTitle: true,
),
safariVCOptions: const SafariViewControllerOptions(
barCollapsingEnabled: true,
));
} }
// Token // Token
@ -190,16 +210,21 @@ class UserController extends GetxController {
return; return;
} }
var attemptAuthRes = await MWApiUser.attemptAuth(loginRequestToken.value); MWResponse<MugenAppAttemptAuthInfo> attemptAuthRes;
if (!attemptAuthRes.ok) { try {
attemptAuthRes = await MWApiUser.attemptAuth(loginRequestToken.value);
} catch (err, stack) {
authProcessing.value = false; authProcessing.value = false;
alert(Get.overlayContext!, attemptAuthRes.errorList?[0].info ?? "未知错误", alert(Get.overlayContext!, ErrorUtils.getErrorMessage(err), title: "错误");
title: "错误"); if (kDebugMode) {
print("Exception in startAuthFlow: $err");
stack.printError();
}
return; return;
} }
var attemptAuthInfo = attemptAuthRes.data!; var attemptAuthInfo = attemptAuthRes.data;
if (attemptAuthInfo.status == "pending") { if (attemptAuthInfo.status == "pending") {
authProcessing.value = false; authProcessing.value = false;
loginRequestToken.value = ""; loginRequestToken.value = "";
@ -208,19 +233,28 @@ class UserController extends GetxController {
return; return;
} else if (attemptAuthInfo.status == "success") { } else if (attemptAuthInfo.status == "success") {
loginRequestToken.value = "";
userId.value = attemptAuthInfo.userid!; userId.value = attemptAuthInfo.userid!;
userName.value = attemptAuthInfo.username!; userName.value = attemptAuthInfo.username!;
try { await updateProfile();
await updateProfile();
} catch (err) {
err.printError(info: 'Cannot update profile after auth');
}
authProcessing.value = false; authProcessing.value = false;
saveToStorage(); saveToStorage();
} }
} }
Future<void> logout() async {} Future<void> logout({bool logoutRemote = true}) async {
// Cookie
await BaseApi.clearCookie();
//
userId.value = 0;
userName.value = "";
nickName.value = "";
avatarUrlSet.clear();
saveToStorage();
}
} }

@ -6,6 +6,7 @@ import 'package:isekai_wiki/components/isekai_nav_bar.dart';
import 'package:isekai_wiki/models/user.dart'; import 'package:isekai_wiki/models/user.dart';
import 'package:isekai_wiki/pages/about.dart'; import 'package:isekai_wiki/pages/about.dart';
import 'package:isekai_wiki/styles.dart'; import 'package:isekai_wiki/styles.dart';
import 'package:isekai_wiki/utils/dialog.dart';
import '../components/dummy_icon.dart'; import '../components/dummy_icon.dart';
import '../components/follow_scale.dart'; import '../components/follow_scale.dart';
@ -20,17 +21,28 @@ class OwnProfileController extends GetxController {
} }
Future<void> handleLoginClick() { Future<void> handleLoginClick() {
handleStartAuth(); if (uc.authProcessing.isFalse) {
return Future.delayed(const Duration(milliseconds: 100)); handleStartAuth();
return Future.delayed(const Duration(milliseconds: 100));
} else {
return Future.value();
}
} }
Future<void> handleStartAuth() async { Future<void> handleStartAuth() async {
await uc.startAuthFlow(); await uc.startAuthFlow();
} }
Future<void> handleLogoutClick() { Future<void> handleLogoutClick() async {
handleLogout(); if (await confirm(
return Future.delayed(const Duration(milliseconds: 100)); Get.overlayContext!,
"你想要退出登录吗?",
title: "退出登录",
positiveText: "确定",
isDanger: true,
)) {
handleLogout();
}
} }
Future<void> handleLogout() async { Future<void> handleLogout() async {
@ -41,21 +53,20 @@ class OwnProfileController extends GetxController {
class OwnProfileTab extends StatelessWidget { class OwnProfileTab extends StatelessWidget {
const OwnProfileTab({super.key}); const OwnProfileTab({super.key});
Widget _buildUserAvatar(UserController uc, {double size = 56}) { Widget _buildUserAvatar(UserController uc, {double size = 64}) {
return Obx(() { return Obx(() {
var avatarUrl = uc.getAvatar(128); var avatarUrl = uc.getAvatar(128);
if (avatarUrl != null && avatarUrl.isNotEmpty) { if (avatarUrl != null && avatarUrl.isNotEmpty) {
return Container( return ClipRRect(
decoration: BoxDecoration( borderRadius: BorderRadius.all(Radius.circular(size / 2)),
borderRadius: BorderRadius.all(Radius.circular(size / 2)),
),
child: CachedNetworkImage( child: CachedNetworkImage(
width: size, width: size,
height: size, height: size,
placeholder: (_, __) => placeholder: (_, __) =>
const CupertinoActivityIndicator(radius: 12), const CupertinoActivityIndicator(radius: 12),
imageUrl: avatarUrl, imageUrl: avatarUrl,
fit: BoxFit.cover,
), ),
); );
} else { } else {
@ -75,58 +86,65 @@ class OwnProfileTab extends StatelessWidget {
return FollowTextScale( return FollowTextScale(
child: Obx( child: Obx(
() => !uc.isLoggedIn () => CupertinoListSection.insetGrouped(
? CupertinoListSection.insetGrouped( backgroundColor: Styles.themePageBackgroundColor,
backgroundColor: Styles.themePageBackgroundColor, dividerMargin: uc.isLoggedIn ? 14 : double.infinity,
children: <CupertinoListTile>[ children: <Widget>[
CupertinoListTile.notched( Obx(
title: () => uc.isLoggedIn
const Text('登录/注册', style: Styles.listTileLargeTitle), ? CupertinoListTile.notched(
padding: const EdgeInsets.only( title: Text(uc.getDisplayName,
left: 6, right: 14, top: 0, bottom: 0), style: Styles.listTileLargeTitle),
leading: const DummyIcon( padding: const EdgeInsets.only(
color: CupertinoColors.systemGrey, left: 6, right: 14, top: 0, bottom: 0),
icon: CupertinoIcons.person_fill, leading: _buildUserAvatar(uc),
size: 60, leadingSize: 80,
rounded: true, leadingToTitle: 4,
trailing: const CupertinoListTileChevron(),
onTap: () {},
)
: CupertinoListTile.notched(
title:
const Text('登录/注册', style: Styles.listTileLargeTitle),
padding: const EdgeInsets.only(
left: 6, right: 14, top: 0, bottom: 0),
leading: const DummyIcon(
color: CupertinoColors.systemGrey,
icon: CupertinoIcons.person_fill,
size: 64,
rounded: true,
),
leadingSize: 80,
leadingToTitle: 4,
trailing: uc.authProcessing.value
? const Padding(
padding: EdgeInsets.only(right: 5),
child: CupertinoActivityIndicator(
radius: 12,
),
)
: const CupertinoListTileChevron(),
onTap: c.handleLoginClick,
), ),
leadingSize: 80, ),
leadingToTitle: 4, AnimatedSize(
trailing: uc.authProcessing.value duration: const Duration(milliseconds: 250),
? const Padding( curve: Curves.easeInOut,
padding: EdgeInsets.only(right: 5), child: SizedBox(
child: CupertinoActivityIndicator( height: uc.isLoggedIn ? null : 0,
radius: 12, child: CupertinoListTile.notched(
), title: const Text('退出登录'),
) leading: const DummyIcon(
: const CupertinoListTileChevron(), color: CupertinoColors.systemRed,
onTap: c.handleLoginClick, icon: CupertinoIcons.arrow_right_square,
),
],
)
: CupertinoListSection.insetGrouped(
backgroundColor: Styles.themePageBackgroundColor,
children: <CupertinoListTile>[
CupertinoListTile.notched(
title: Text(uc.getDisplayName,
style: Styles.listTileLargeTitle),
leading: _buildUserAvatar(uc),
leadingSize: 80,
leadingToTitle: 4,
trailing: const CupertinoListTileChevron(),
onTap: () {},
), ),
CupertinoListTile.notched( trailing: const CupertinoListTileChevron(),
title: const Text('退出登录'), onTap: c.handleLogoutClick,
leading: const DummyIcon( ),
color: CupertinoColors.systemRed,
icon: CupertinoIcons.arrow_right_square,
),
trailing: const CupertinoListTileChevron(),
onTap: () {},
),
],
), ),
),
],
),
), ),
); );
} }

@ -26,7 +26,10 @@ Future<void> alert(BuildContext context, String content, {String? title}) {
} }
Future<bool> confirm(BuildContext context, String content, Future<bool> confirm(BuildContext context, String content,
{String? title, String? positiveText, String? negativeText}) { {String? title,
String? positiveText,
String? negativeText,
bool isDanger = false}) {
var c = Completer<bool>(); var c = Completer<bool>();
positiveText ??= ""; positiveText ??= "";
@ -46,7 +49,8 @@ Future<bool> confirm(BuildContext context, String content,
child: Text(negativeText!), child: Text(negativeText!),
), ),
CupertinoDialogAction( CupertinoDialogAction(
isDefaultAction: true, isDestructiveAction: isDanger,
isDefaultAction: !isDanger,
onPressed: () { onPressed: () {
Navigator.of(context).pop(); Navigator.of(context).pop();
c.complete(true); c.complete(true);

@ -0,0 +1,34 @@
import 'package:dio/dio.dart';
import 'package:http2/http2.dart';
import 'package:isekai_wiki/api/base_api.dart';
import 'package:isekai_wiki/api/mw/mw_api.dart';
class ErrorUtils {
static String getErrorMessage(dynamic err) {
if (err is MWApiEmptyBodyException) {
return "加载的数据为空";
} else if (err is MWApiErrorException) {
return err.info ?? err.code;
} else if (err is HttpResponseException) {
return "服务器错误:${err.statusCode} ${err.statusText}";
} else if (err is TypeError) {
// JSON
return "服务器返回的数据不正确";
} else if (err is DioError) {
if (err.type == DioErrorType.connectTimeout ||
err.type == DioErrorType.sendTimeout ||
err.type == DioErrorType.receiveTimeout) {
return "链接超时,请重试";
} else if (err.type == DioErrorType.response) {
return "HTTP错误${err.response?.statusCode} ${err.response?.statusMessage}";
} else if (err.type == DioErrorType.other) {
var innerErr = err.error! as Exception;
if (innerErr is StreamTransportException) {
return innerErr.message;
}
}
return err.message;
}
return err.toString();
}
}

@ -428,7 +428,7 @@ packages:
source: hosted source: hosted
version: "0.13.5" version: "0.13.5"
http2: http2:
dependency: transitive dependency: "direct main"
description: description:
name: http2 name: http2
url: "https://pub.dartlang.org" url: "https://pub.dartlang.org"
@ -603,7 +603,7 @@ packages:
source: hosted source: hosted
version: "0.2.1" version: "0.2.1"
path_provider: path_provider:
dependency: transitive dependency: "direct main"
description: description:
name: path_provider name: path_provider
url: "https://pub.dartlang.org" url: "https://pub.dartlang.org"

@ -58,6 +58,8 @@ dependencies:
package_info_plus: ^3.0.2 package_info_plus: ^3.0.2
pull_down_button: ^0.4.1 pull_down_button: ^0.4.1
cached_network_image: ^3.2.3 cached_network_image: ^3.2.3
path_provider: ^2.0.11
http2: ^2.0.1
dio: ^4.0.6 dio: ^4.0.6
dio_cookie_manager: ^2.0.0 dio_cookie_manager: ^2.0.0
dio_http2_adapter: ^2.0.0 dio_http2_adapter: ^2.0.0

Loading…
Cancel
Save