MCSI-1 feat: download progress

This commit is contained in:
SquidSpirit 2025-06-29 17:23:14 +09:00
parent 95bf2bc86e
commit f1716b0505
13 changed files with 152 additions and 44 deletions

View File

@ -1,3 +1,4 @@
abstract class Constants { abstract class Constants {
static const gameVersionListUrl = 'https://www.dropbox.com/s/mtz3moc9dpjtz7s/GameVersions.txt?dl=1';
static const serverFileName = 'server.jar'; static const serverFileName = 'server.jar';
} }

View File

@ -1,8 +1,10 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:gap/gap.dart';
import 'package:minecraft_server_installer/main/framework/ui/strings.dart'; import 'package:minecraft_server_installer/main/framework/ui/strings.dart';
import 'package:minecraft_server_installer/vanila/adapter/presentation/vanila_bloc.dart'; import 'package:minecraft_server_installer/vanila/adapter/presentation/vanila_bloc.dart';
import 'package:minecraft_server_installer/vanila/adapter/presentation/game_version_view_model.dart'; import 'package:minecraft_server_installer/vanila/adapter/presentation/game_version_view_model.dart';
import 'package:minecraft_server_installer/vanila/adapter/presentation/vanila_state.dart';
import 'package:minecraft_server_installer/vanila/framework/ui/game_version_dropdown.dart'; import 'package:minecraft_server_installer/vanila/framework/ui/game_version_dropdown.dart';
class BasicConfigurationTab extends StatefulWidget { class BasicConfigurationTab extends StatefulWidget {
@ -21,10 +23,27 @@ class _BasicConfigurationTabState extends State<BasicConfigurationTab> {
children: [ children: [
const GameVersionDropdown(), const GameVersionDropdown(),
const Spacer(), const Spacer(),
ElevatedButton.icon( BlocConsumer<VanilaBloc, VanilaState>(
onPressed: context.watch<VanilaBloc>().state.isGameVersionSelected ? _downloadServerFile : null, listener: (_, __) {},
icon: const Icon(Icons.download), builder:
label: const Text(Strings.buttonStartToInstall), (context, state) => Row(
mainAxisAlignment: MainAxisAlignment.end,
children: [
if (state.isDownloading) Expanded(child: LinearProgressIndicator(value: state.downloadProgress)),
const Gap(32),
ElevatedButton.icon(
style: ElevatedButton.styleFrom(
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
),
onPressed: state.isGameVersionSelected ? _downloadServerFile : null,
icon: const Icon(Icons.download),
label: const Padding(
padding: EdgeInsets.symmetric(vertical: 12),
child: Text(Strings.buttonStartToInstall),
),
),
],
),
), ),
], ],
); );

View File

@ -3,6 +3,7 @@ import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:minecraft_server_installer/main/framework/ui/basic_configuration_tab.dart'; import 'package:minecraft_server_installer/main/framework/ui/basic_configuration_tab.dart';
import 'package:minecraft_server_installer/vanila/adapter/gateway/vanila_repository_impl.dart'; import 'package:minecraft_server_installer/vanila/adapter/gateway/vanila_repository_impl.dart';
import 'package:minecraft_server_installer/vanila/adapter/presentation/vanila_bloc.dart'; import 'package:minecraft_server_installer/vanila/adapter/presentation/vanila_bloc.dart';
import 'package:minecraft_server_installer/vanila/adapter/presentation/vanila_state.dart';
import 'package:minecraft_server_installer/vanila/application/use_case/download_server_file_use_case.dart'; import 'package:minecraft_server_installer/vanila/application/use_case/download_server_file_use_case.dart';
import 'package:minecraft_server_installer/vanila/application/use_case/get_game_version_list_use_case.dart'; import 'package:minecraft_server_installer/vanila/application/use_case/get_game_version_list_use_case.dart';
import 'package:minecraft_server_installer/vanila/framework/api/vanila_api_service_impl.dart'; import 'package:minecraft_server_installer/vanila/framework/api/vanila_api_service_impl.dart';
@ -11,6 +12,9 @@ import 'package:minecraft_server_installer/vanila/framework/storage/vanila_file_
class MinecraftServerInstaller extends StatelessWidget { class MinecraftServerInstaller extends StatelessWidget {
const MinecraftServerInstaller({super.key}); const MinecraftServerInstaller({super.key});
Widget get _body =>
const Padding(padding: EdgeInsets.symmetric(horizontal: 24, vertical: 32), child: BasicConfigurationTab());
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final gameVersionApiService = VanilaApiServiceImpl(); final gameVersionApiService = VanilaApiServiceImpl();
@ -21,15 +25,27 @@ class MinecraftServerInstaller extends StatelessWidget {
return MaterialApp( return MaterialApp(
title: 'Minecraft Server Installer', title: 'Minecraft Server Installer',
theme: ThemeData(primarySwatch: Colors.blue), theme: ThemeData.light().copyWith(colorScheme: ColorScheme.fromSeed(seedColor: Colors.blue.shade900)),
home: MultiBlocProvider( home: MultiBlocProvider(
providers: [ providers: [
BlocProvider<VanilaBloc>( BlocProvider<VanilaBloc>(
create: (context) => VanilaBloc(getGameVersionListUseCase, downloadServerFileUseCase), create:
(context) =>
VanilaBloc(getGameVersionListUseCase, downloadServerFileUseCase)
..add(VanilaGameVersionListLoadedEvent()),
), ),
], ],
child: const Scaffold( child: Scaffold(
body: Padding(padding: EdgeInsets.symmetric(horizontal: 24, vertical: 32), child: BasicConfigurationTab()), body: BlocConsumer<VanilaBloc, VanilaState>(
listener: (_, __) {},
builder: (_, state) {
if (state.isLocked) {
return MouseRegion(cursor: SystemMouseCursors.forbidden, child: AbsorbPointer(child: _body));
}
return _body;
},
),
), ),
), ),
); );

View File

@ -1,9 +1,10 @@
import 'dart:typed_data'; import 'dart:typed_data';
import 'package:minecraft_server_installer/vanila/application/use_case/download_server_file_use_case.dart';
import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart'; import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart';
abstract interface class VanilaApiService { abstract interface class VanilaApiService {
Future<List<GameVersion>> fetchGameVersionList(); Future<List<GameVersion>> fetchGameVersionList();
Future<Uint8List> fetchServerFile(Uri url); Future<Uint8List> fetchServerFile(Uri url, {DownloadProgressCallback? onProgressChanged});
} }

View File

@ -1,6 +1,7 @@
import 'package:minecraft_server_installer/vanila/adapter/gateway/vanila_api_service.dart'; import 'package:minecraft_server_installer/vanila/adapter/gateway/vanila_api_service.dart';
import 'package:minecraft_server_installer/vanila/adapter/gateway/vanila_file_storage.dart'; import 'package:minecraft_server_installer/vanila/adapter/gateway/vanila_file_storage.dart';
import 'package:minecraft_server_installer/vanila/application/repository/vanila_repository.dart'; import 'package:minecraft_server_installer/vanila/application/repository/vanila_repository.dart';
import 'package:minecraft_server_installer/vanila/application/use_case/download_server_file_use_case.dart';
import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart'; import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart';
class VanilaRepositoryImpl implements VanilaRepository { class VanilaRepositoryImpl implements VanilaRepository {
@ -13,8 +14,12 @@ class VanilaRepositoryImpl implements VanilaRepository {
Future<List<GameVersion>> getGameVersionList() => _gameVersionApiService.fetchGameVersionList(); Future<List<GameVersion>> getGameVersionList() => _gameVersionApiService.fetchGameVersionList();
@override @override
Future<void> downloadServerFile(GameVersion version, String savePath) async { Future<void> downloadServerFile(
final fileBytes = await _gameVersionApiService.fetchServerFile(version.url); GameVersion version,
String savePath, {
DownloadProgressCallback? onProgressChanged,
}) async {
final fileBytes = await _gameVersionApiService.fetchServerFile(version.url, onProgressChanged: onProgressChanged);
await _gameVersionFileStorage.saveFile(fileBytes, savePath); await _gameVersionFileStorage.saveFile(fileBytes, savePath);
} }
} }

View File

@ -15,9 +15,8 @@ class VanilaBloc extends Bloc<VanilaEvent, VanilaState> {
try { try {
final gameVersions = await _getGameVersionListUseCase(); final gameVersions = await _getGameVersionListUseCase();
emit( emit(
VanilaState( const VanilaState.empty().copyWith(
gameVersions: gameVersions.map((entity) => GameVersionViewModel.from(entity)).toList(), gameVersions: gameVersions.map((entity) => GameVersionViewModel.from(entity)).toList(),
selectedGameVersion: null,
), ),
); );
} on Exception { } on Exception {
@ -35,7 +34,23 @@ class VanilaBloc extends Bloc<VanilaEvent, VanilaState> {
return; return;
} }
await _downloadServerFileUseCase(gameVersion.toEntity(), path.join('.', Constants.serverFileName)); emit(state.copyWith(isLocked: true));
await _downloadServerFileUseCase(
gameVersion.toEntity(),
path.join('.', Constants.serverFileName),
onProgressChanged: (progress) => add(_VanilaDownloadProgressChangedEvent(progress)),
);
emit(state.copyWith(isLocked: false));
});
on<_VanilaDownloadProgressChangedEvent>((event, emit) {
if (event.progress < 0) {
emit(state.copyWith(downloadProgress: 0));
} else if (event.progress > 1) {
emit(state.copyWith(downloadProgress: 1));
} else {
emit(state.copyWith(downloadProgress: event.progress));
}
}); });
} }
} }
@ -51,3 +66,9 @@ class VanilaGameVersionSelectedEvent extends VanilaEvent {
} }
class VanilaServerFileDownloadedEvent extends VanilaEvent {} class VanilaServerFileDownloadedEvent extends VanilaEvent {}
class _VanilaDownloadProgressChangedEvent extends VanilaEvent {
final double progress;
_VanilaDownloadProgressChangedEvent(this.progress);
}

View File

@ -1,23 +1,38 @@
import 'package:equatable/equatable.dart'; import 'package:equatable/equatable.dart';
import 'package:minecraft_server_installer/vanila/adapter/presentation/game_version_view_model.dart'; import 'package:minecraft_server_installer/vanila/adapter/presentation/game_version_view_model.dart';
class VanilaState with EquatableMixin { class VanilaState with EquatableMixin {
final bool isLocked;
final double downloadProgress;
final List<GameVersionViewModel> gameVersions; final List<GameVersionViewModel> gameVersions;
final GameVersionViewModel? selectedGameVersion; final GameVersionViewModel? selectedGameVersion;
const VanilaState({required this.gameVersions, required this.selectedGameVersion}); const VanilaState({
required this.isLocked,
required this.downloadProgress,
required this.gameVersions,
required this.selectedGameVersion,
});
const VanilaState.empty() : this(gameVersions: const [], selectedGameVersion: null); const VanilaState.empty()
: this(isLocked: false, downloadProgress: 0, gameVersions: const [], selectedGameVersion: null);
@override @override
List<Object?> get props => [gameVersions, selectedGameVersion]; List<Object?> get props => [isLocked, downloadProgress, gameVersions, selectedGameVersion];
bool get isGameVersionSelected => selectedGameVersion != null; bool get isGameVersionSelected => selectedGameVersion != null;
VanilaState copyWith({List<GameVersionViewModel>? gameVersions, GameVersionViewModel? selectedGameVersion}) => bool get isDownloading => downloadProgress > 0 && downloadProgress < 1;
VanilaState(
gameVersions: gameVersions ?? this.gameVersions, VanilaState copyWith({
selectedGameVersion: selectedGameVersion ?? this.selectedGameVersion, bool? isLocked,
); double? downloadProgress,
} List<GameVersionViewModel>? gameVersions,
GameVersionViewModel? selectedGameVersion,
}) => VanilaState(
isLocked: isLocked ?? this.isLocked,
downloadProgress: downloadProgress ?? this.downloadProgress,
gameVersions: gameVersions ?? this.gameVersions,
selectedGameVersion: selectedGameVersion ?? this.selectedGameVersion,
);
}

View File

@ -1,7 +1,8 @@
import 'package:minecraft_server_installer/vanila/application/use_case/download_server_file_use_case.dart';
import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart'; import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart';
abstract interface class VanilaRepository { abstract interface class VanilaRepository {
Future<List<GameVersion>> getGameVersionList(); Future<List<GameVersion>> getGameVersionList();
Future<void> downloadServerFile(GameVersion version, String savePath); Future<void> downloadServerFile(GameVersion version, String savePath, {DownloadProgressCallback? onProgressChanged});
} }

View File

@ -1,10 +1,13 @@
import 'package:minecraft_server_installer/vanila/application/repository/vanila_repository.dart'; import 'package:minecraft_server_installer/vanila/application/repository/vanila_repository.dart';
import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart'; import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart';
typedef DownloadProgressCallback = void Function(double progress);
class DownloadServerFileUseCase { class DownloadServerFileUseCase {
final VanilaRepository _gameVersionRepository; final VanilaRepository _gameVersionRepository;
DownloadServerFileUseCase(this._gameVersionRepository); DownloadServerFileUseCase(this._gameVersionRepository);
Future<void> call(GameVersion version, String savePath) => _gameVersionRepository.downloadServerFile(version, savePath); Future<void> call(GameVersion version, String savePath, {DownloadProgressCallback? onProgressChanged}) =>
_gameVersionRepository.downloadServerFile(version, savePath, onProgressChanged: onProgressChanged);
} }

View File

@ -1,13 +1,16 @@
import 'dart:async';
import 'dart:typed_data'; import 'dart:typed_data';
import 'package:http/http.dart' as http; import 'package:http/http.dart' as http;
import 'package:minecraft_server_installer/main/constants.dart';
import 'package:minecraft_server_installer/vanila/adapter/gateway/vanila_api_service.dart'; import 'package:minecraft_server_installer/vanila/adapter/gateway/vanila_api_service.dart';
import 'package:minecraft_server_installer/vanila/application/use_case/download_server_file_use_case.dart';
import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart'; import 'package:minecraft_server_installer/vanila/domain/entity/game_version.dart';
class VanilaApiServiceImpl implements VanilaApiService { class VanilaApiServiceImpl implements VanilaApiService {
@override @override
Future<List<GameVersion>> fetchGameVersionList() async { Future<List<GameVersion>> fetchGameVersionList() async {
final sourceUrl = Uri.parse('https://www.dropbox.com/s/mtz3moc9dpjtz7s/GameVersions.txt?dl=1'); final sourceUrl = Uri.parse(Constants.gameVersionListUrl);
final response = await http.get(sourceUrl); final response = await http.get(sourceUrl);
final rawGameVersionList = response.body.split('\n'); final rawGameVersionList = response.body.split('\n');
@ -21,8 +24,34 @@ class VanilaApiServiceImpl implements VanilaApiService {
} }
@override @override
Future<Uint8List> fetchServerFile(Uri url) async { Future<Uint8List> fetchServerFile(Uri url, {DownloadProgressCallback? onProgressChanged}) async {
final response = await http.get(url); final client = http.Client();
return response.bodyBytes; final request = http.Request('GET', url);
final response = await client.send(request);
final contentLength = response.contentLength;
final completer = Completer<Uint8List>();
final bytes = <int>[];
var receivedBytes = 0;
response.stream.listen(
(chunk) {
bytes.addAll(chunk);
receivedBytes += chunk.length;
if (onProgressChanged != null && contentLength != null) {
onProgressChanged(receivedBytes / contentLength);
}
},
onDone: () {
if (onProgressChanged != null) {
onProgressChanged(1);
}
completer.complete(Uint8List.fromList(bytes));
},
onError: completer.completeError,
cancelOnError: true,
);
return completer.future;
} }
} }

View File

@ -5,20 +5,9 @@ import 'package:minecraft_server_installer/vanila/adapter/presentation/vanila_bl
import 'package:minecraft_server_installer/vanila/adapter/presentation/game_version_view_model.dart'; import 'package:minecraft_server_installer/vanila/adapter/presentation/game_version_view_model.dart';
import 'package:minecraft_server_installer/vanila/adapter/presentation/vanila_state.dart'; import 'package:minecraft_server_installer/vanila/adapter/presentation/vanila_state.dart';
class GameVersionDropdown extends StatefulWidget { class GameVersionDropdown extends StatelessWidget {
const GameVersionDropdown({super.key}); const GameVersionDropdown({super.key});
@override
State<GameVersionDropdown> createState() => _GameVersionDropdownState();
}
class _GameVersionDropdownState extends State<GameVersionDropdown> {
@override
void initState() {
super.initState();
context.read<VanilaBloc>().add(VanilaGameVersionListLoadedEvent());
}
@override @override
Widget build(BuildContext context) => BlocConsumer<VanilaBloc, VanilaState>( Widget build(BuildContext context) => BlocConsumer<VanilaBloc, VanilaState>(
listener: (_, __) {}, listener: (_, __) {},

View File

@ -99,6 +99,14 @@ packages:
description: flutter description: flutter
source: sdk source: sdk
version: "0.0.0" version: "0.0.0"
gap:
dependency: "direct main"
description:
name: gap
sha256: f19387d4e32f849394758b91377f9153a1b41d79513ef7668c088c77dbc6955d
url: "https://pub.dev"
source: hosted
version: "3.0.1"
http: http:
dependency: "direct main" dependency: "direct main"
description: description:

View File

@ -2,7 +2,7 @@ name: minecraft_server_installer
description: "A tool that makes installing a Minecraft Server easier." description: "A tool that makes installing a Minecraft Server easier."
# The following line prevents the package from being accidentally published to # The following line prevents the package from being accidentally published to
# pub.dev using `flutter pub publish`. This is preferred for private packages. # pub.dev using `flutter pub publish`. This is preferred for private packages.
publish_to: 'none' # Remove this line if you wish to publish to pub.dev publish_to: "none" # Remove this line if you wish to publish to pub.dev
# The following defines the version and build number for your application. # The following defines the version and build number for your application.
# A version number is three numbers separated by dots, like 1.2.43 # A version number is three numbers separated by dots, like 1.2.43
@ -36,6 +36,7 @@ dependencies:
cupertino_icons: ^1.0.8 cupertino_icons: ^1.0.8
equatable: ^2.0.7 equatable: ^2.0.7
flutter_bloc: ^9.1.1 flutter_bloc: ^9.1.1
gap: ^3.0.1
http: ^1.4.0 http: ^1.4.0
path: ^1.9.1 path: ^1.9.1
@ -55,7 +56,6 @@ dev_dependencies:
# The following section is specific to Flutter packages. # The following section is specific to Flutter packages.
flutter: flutter:
# The following line ensures that the Material Icons font is # The following line ensures that the Material Icons font is
# included with your application, so that you can use the icons in # included with your application, so that you can use the icons in
# the material Icons class. # the material Icons class.