ToneSnap_FSX_Flutter/lib/data/network/dio_client.dart
fengshengxiong 7611e78244 1.变声加入广告和Firebase
2.音乐播放器首页完成数据加载展示
2024-06-26 15:32:21 +08:00

182 lines
5.2 KiB
Dart

// Author: fengshengxiong
// Date: 2024/5/9
// Description: dio封装
import 'dart:io';
import 'package:dio/dio.dart';
import 'package:pretty_dio_logger/pretty_dio_logger.dart';
import 'package:tone_snap/components/base_easyloading.dart';
import 'package:tone_snap/data/sideb/api/music_api.dart';
import 'package:tone_snap/data/network/base_error.dart';
import 'package:tone_snap/data/network/dio_interceptor.dart';
import 'package:tone_snap/utils/log_util.dart';
class DioClient {
static final DioClient _instance = DioClient._internal();
factory DioClient({String? baseUrl}) => getInstance(baseUrl: baseUrl);
late Dio _dio;
static DioClient getInstance({String? baseUrl}) {
if (baseUrl == null) {
return _instance._defaultBaseUrl();
} else {
return _instance._newBaseUrl(baseUrl);
}
}
/// 默认域名
DioClient _defaultBaseUrl() {
if (_dio.options.baseUrl != MusicApi.baseUrl) {
_dio.options.baseUrl = MusicApi.baseUrl;
}
return _instance;
}
/// 用于指定特定域名
DioClient _newBaseUrl(String baseUrl) {
_dio.options.baseUrl = baseUrl;
return _instance;
}
DioClient._internal() {
_dio = Dio();
final baseOptions = BaseOptions(
baseUrl: MusicApi.baseUrl,
connectTimeout: const Duration(seconds: 15),
receiveTimeout: const Duration(seconds: 15),
);
_dio.options = baseOptions;
_dio.interceptors.add(DioInterceptor());
_dio.interceptors.add(PrettyDioLogger(
requestHeader: false,
requestBody: true,
responseBody: false,
compact: false,
maxWidth: 160,
));
}
/// 请求
Future request<T>(
String path, {
required RequestMethod requestMethod,
dynamic data,
Map<String, dynamic>? queryParameters,
CancelToken? cancelToken,
Options? options,
bool isFormData = false,
bool showLoading = false,
bool showToast = false,
T Function(Map<String, dynamic>)? formJson,
required Function(T? result) success,
Function(BaseError baseError)? fail,
}) async {
try {
BaseEasyLoading.loading(show: showLoading);
_dio.options.method = requestMethod.method;
Response response = await _dio.request(
path,
data: isFormData ? FormData.fromMap(data) : data,
queryParameters: queryParameters,
cancelToken: cancelToken,
options: options,
);
BaseEasyLoading.dismiss(dismiss: showLoading);
if (response.statusCode == HttpStatus.ok) {
if (response.data != null && formJson != null) {
success(formJson(response.data));
} else {
success(response.data);
}
}
} catch (e) {
BaseError baseError = getError(e);
if (fail != null) fail(baseError);
BaseEasyLoading.toast(baseError.message);
LogUtil.e(e.toString());
}
}
/// 下载
Future<void> download(
String urlPath,
dynamic savePath, {
ProgressCallback? onReceiveProgress,
Map<String, dynamic>? queryParameters,
CancelToken? cancelToken,
data,
Options? options,
required Function() success,
Function(BaseError? err)? fail,
}) async {
try {
await _dio.download(
urlPath,
savePath,
onReceiveProgress: onReceiveProgress,
queryParameters: queryParameters,
cancelToken: cancelToken,
data: data,
options: Options(receiveTimeout: const Duration(seconds: 0)),
);
success();
} catch (e) {
BaseError baseError = getError(e);
if (fail != null) fail(baseError);
BaseEasyLoading.toast(baseError.message);
LogUtil.e(baseError.message);
}
}
BaseError getError(Object e) {
if (e.runtimeType == DioException) {
switch ((e as DioException).type) {
case DioExceptionType.connectionTimeout:
return OtherError(statusCode: -1, statusMessage: 'connection timed out');
case DioExceptionType.sendTimeout:
return OtherError(statusCode: -1, statusMessage: 'send timeout');
case DioExceptionType.receiveTimeout:
return OtherError(statusCode: -1, statusMessage: 'receive timeout');
case DioExceptionType.badCertificate:
return OtherError(statusCode: -1, statusMessage: 'certificate error');
case DioExceptionType.cancel:
return OtherError(statusCode: -1, statusMessage: 'request canceled');
case DioExceptionType.connectionError:
return OtherError(statusCode: -1, statusMessage: 'connection error');
case DioExceptionType.unknown:
return OtherError(statusCode: -1, statusMessage: 'unknown error');
case DioExceptionType.badResponse:
final response = e.response;
if (response!.statusCode == 401) {
return NeedLogin();
} else if (response.statusCode == 403) {
return NeedAuth();
} else {
return OtherError(
statusCode: response.statusCode,
statusMessage: response.statusMessage,
);
}
}
}
return OtherError(statusCode: -1, statusMessage: 'unknown error');
}
}
enum RequestMethod {
get('GET'),
post('POST'),
put('PUT'),
head('HEAD'),
delete('DELETE'),
patch('PATCH');
const RequestMethod(
this.method,
);
final String method;
}