mirror of
https://github.com/kuhyx/diet-guard.git
synced 2026-07-04 15:23:16 +02:00
Milestone 2 of the diet-app-as-wise-balloon plan, plus feedback from manually testing it on-device: - PhotoAttachService wraps image_picker and copies the picked photo into <app documents>/images/<uuid>.<ext>, so the file survives after the picker's own (possibly cache-cleared) temp copy is gone. Phone-local only, per the sync plan: imagePath is never synced. - PhotoAttachField is a shared attach/preview/remove widget, used identically by both the single-item and composite-meal logging screens, so logging a multi-item meal can now carry a photo too. - PhotoViewerScreen gives a full-screen, pinch-to-zoom view of an attached photo -- the 64x64 inline thumbnail was too small to actually check the photo. - HistoryScreen lists every logged entry across all days, newest first, with a thumbnail when one is attached. There was previously no way to confirm what got logged (or whether a photo actually attached) short of inspecting food_log.json directly. Verified on a physical device (BL9000): built, installed, and the user confirmed the photo-attach flow logs a real entry with a real photo, visible afterward in the new history list. 88 Flutter tests passing, `flutter analyze` clean against very_good_analysis. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> Claude-Session: https://claude.ai/code/session_01FU3f5KQ1GHXsbbSecfVEyF
65 lines
1.9 KiB
Dart
65 lines
1.9 KiB
Dart
import 'dart:io';
|
|
|
|
import 'package:diet_guard_app/services/photo_attach_service.dart';
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
import 'package:image_picker/image_picker.dart';
|
|
import 'package:image_picker_platform_interface/image_picker_platform_interface.dart';
|
|
|
|
/// Returns a fixed [XFile] (or null, to simulate a cancelled picker) without
|
|
/// touching any real platform channel.
|
|
class _FakeImagePickerPlatform extends ImagePickerPlatform {
|
|
_FakeImagePickerPlatform(this._result);
|
|
|
|
final XFile? _result;
|
|
|
|
@override
|
|
Future<XFile?> getImageFromSource({
|
|
required ImageSource source,
|
|
ImagePickerOptions options = const ImagePickerOptions(),
|
|
}) async => _result;
|
|
}
|
|
|
|
void main() {
|
|
late Directory tempDir;
|
|
late ImagePickerPlatform originalPlatform;
|
|
|
|
setUp(() async {
|
|
tempDir = await Directory.systemTemp.createTemp('diet_guard_photo_');
|
|
originalPlatform = ImagePickerPlatform.instance;
|
|
PhotoAttachService.resetForTesting(testDir: tempDir);
|
|
});
|
|
|
|
tearDown(() async {
|
|
ImagePickerPlatform.instance = originalPlatform;
|
|
PhotoAttachService.resetForTesting();
|
|
await tempDir.delete(recursive: true);
|
|
});
|
|
|
|
test('copies the picked file into <documents>/images with a new name', () async {
|
|
final source = File('${tempDir.path}/source.jpg')
|
|
..writeAsBytesSync([1, 2, 3, 4]);
|
|
ImagePickerPlatform.instance = _FakeImagePickerPlatform(
|
|
XFile(source.path),
|
|
);
|
|
|
|
final result = await PhotoAttachService.instance.pickAndStore(
|
|
ImageSource.gallery,
|
|
);
|
|
|
|
expect(result, isNotNull);
|
|
expect(result, startsWith('${tempDir.path}/images/'));
|
|
expect(result, endsWith('.jpg'));
|
|
expect(File(result!).readAsBytesSync(), [1, 2, 3, 4]);
|
|
});
|
|
|
|
test('returns null when the picker is cancelled', () async {
|
|
ImagePickerPlatform.instance = _FakeImagePickerPlatform(null);
|
|
|
|
final result = await PhotoAttachService.instance.pickAndStore(
|
|
ImageSource.camera,
|
|
);
|
|
|
|
expect(result, isNull);
|
|
});
|
|
}
|