feat(save): persist booster inventory
This commit is contained in:
@@ -2,6 +2,7 @@ import 'dart:convert';
|
|||||||
|
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
|
import '../game/models/booster.dart';
|
||||||
import 'streak.dart';
|
import 'streak.dart';
|
||||||
|
|
||||||
class StageProgress {
|
class StageProgress {
|
||||||
@@ -51,6 +52,10 @@ class SaveRepository {
|
|||||||
_reviewRequested = (json['flags']
|
_reviewRequested = (json['flags']
|
||||||
as Map<String, dynamic>?)?['reviewRequested'] as bool? ??
|
as Map<String, dynamic>?)?['reviewRequested'] as bool? ??
|
||||||
false;
|
false;
|
||||||
|
final boosters = json['boosters'] as Map<String, dynamic>? ?? {};
|
||||||
|
for (final t in BoosterType.values) {
|
||||||
|
_boosters[t] = boosters[t.name] as int? ?? 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -68,6 +73,9 @@ class SaveRepository {
|
|||||||
bool _soundEnabled = true;
|
bool _soundEnabled = true;
|
||||||
bool _musicEnabled = true;
|
bool _musicEnabled = true;
|
||||||
bool _reviewRequested = false;
|
bool _reviewRequested = false;
|
||||||
|
final Map<BoosterType, int> _boosters = {
|
||||||
|
for (final t in BoosterType.values) t: 0,
|
||||||
|
};
|
||||||
|
|
||||||
StreakState get streak => _streak;
|
StreakState get streak => _streak;
|
||||||
bool get tutorialDone => _tutorialDone;
|
bool get tutorialDone => _tutorialDone;
|
||||||
@@ -109,6 +117,22 @@ class SaveRepository {
|
|||||||
return _flush();
|
return _flush();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int boosterCount(BoosterType type) => _boosters[type] ?? 0;
|
||||||
|
|
||||||
|
Future<void> grantBooster(BoosterType type, [int n = 1]) {
|
||||||
|
_boosters[type] = (_boosters[type] ?? 0) + n;
|
||||||
|
return _flush();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Spends one booster. Returns false (and changes nothing) when none are left.
|
||||||
|
Future<bool> consumeBooster(BoosterType type) async {
|
||||||
|
final have = _boosters[type] ?? 0;
|
||||||
|
if (have <= 0) return false;
|
||||||
|
_boosters[type] = have - 1;
|
||||||
|
await _flush();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
Future<void> recordEndlessScore(int score) {
|
Future<void> recordEndlessScore(int score) {
|
||||||
if (score > _endlessBest) _endlessBest = score;
|
if (score > _endlessBest) _endlessBest = score;
|
||||||
return _flush();
|
return _flush();
|
||||||
@@ -185,6 +209,7 @@ class SaveRepository {
|
|||||||
'reviewRequested': _reviewRequested,
|
'reviewRequested': _reviewRequested,
|
||||||
},
|
},
|
||||||
'endless': {'best': _endlessBest},
|
'endless': {'best': _endlessBest},
|
||||||
|
'boosters': {for (final t in BoosterType.values) t.name: _boosters[t]},
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,37 @@
|
|||||||
|
import 'package:block_seasons/data/save_repository.dart';
|
||||||
|
import 'package:block_seasons/game/models/booster.dart';
|
||||||
|
import 'package:flutter_test/flutter_test.dart';
|
||||||
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
TestWidgetsFlutterBinding.ensureInitialized();
|
||||||
|
|
||||||
|
Future<SaveRepository> fresh() async {
|
||||||
|
SharedPreferences.setMockInitialValues({});
|
||||||
|
return SaveRepository(await SharedPreferences.getInstance());
|
||||||
|
}
|
||||||
|
|
||||||
|
test('booster counts start at zero', () async {
|
||||||
|
final repo = await fresh();
|
||||||
|
for (final t in BoosterType.values) {
|
||||||
|
expect(repo.boosterCount(t), 0);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test('grantBooster adds and persists across reload', () async {
|
||||||
|
final repo = await fresh();
|
||||||
|
await repo.grantBooster(BoosterType.hammer, 2);
|
||||||
|
expect(repo.boosterCount(BoosterType.hammer), 2);
|
||||||
|
|
||||||
|
final reloaded = SaveRepository(await SharedPreferences.getInstance());
|
||||||
|
expect(reloaded.boosterCount(BoosterType.hammer), 2);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('consumeBooster decrements and returns false at zero', () async {
|
||||||
|
final repo = await fresh();
|
||||||
|
await repo.grantBooster(BoosterType.shuffle, 1);
|
||||||
|
expect(await repo.consumeBooster(BoosterType.shuffle), isTrue);
|
||||||
|
expect(repo.boosterCount(BoosterType.shuffle), 0);
|
||||||
|
expect(await repo.consumeBooster(BoosterType.shuffle), isFalse);
|
||||||
|
});
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user