Newer
Older
MiniTias / lib / services / file_service.dart
import 'dart:io';

import 'package:camera/camera.dart';
import 'package:flutter/foundation.dart';
import 'package:image/image.dart' as img;

/// 撮影画像の保存・命名・削除を行うサービス.
class FileService {
  static const _basePath = '/storage/emulated/0/Pictures/MiniTIAS';

  /// 保存先ディレクトリのパスを返す.
  String get directoryPath => _basePath;

  /// CameraImage(YUV420)を PNG に変換して保存する.
  ///
  /// JPEG を経由せず,生データから直接 PNG に変換するため画質劣化がない.
  /// 保存したファイルのパスを返す.
  Future<String> saveImageFromYuv(CameraImage cameraImage) async {
    final directory = Directory(_basePath);
    if (!await directory.exists()) {
      await directory.create(recursive: true);
    }

    // YUV → PNG 変換(重い処理なので isolate で実行)
    final params = {
      'width': cameraImage.width,
      'height': cameraImage.height,
      'yPlane': Uint8List.fromList(cameraImage.planes[0].bytes),
      'uPlane': Uint8List.fromList(cameraImage.planes[1].bytes),
      'vPlane': Uint8List.fromList(cameraImage.planes[2].bytes),
      'yRowStride': cameraImage.planes[0].bytesPerRow,
      'uvRowStride': cameraImage.planes[1].bytesPerRow,
      'uvPixelStride': cameraImage.planes[1].bytesPerPixel ?? 1,
    };
    final pngBytes = await compute(_convertYuvToPng, params);

    final fileName = await generateFileName();
    final filePath = '$_basePath/$fileName';
    final file = File(filePath);
    await file.writeAsBytes(pngBytes);

    return filePath;
  }

  /// ファイル名を生成する.
  ///
  /// 形式: MiniTIAS_YYYYMMDD_HHmmss.png
  /// 同秒の重複がある場合は _1, _2, ... と連番を付与する.
  Future<String> generateFileName() async {
    final now = DateTime.now();
    final timestamp =
        '${now.year}'
        '${now.month.toString().padLeft(2, '0')}'
        '${now.day.toString().padLeft(2, '0')}'
        '_'
        '${now.hour.toString().padLeft(2, '0')}'
        '${now.minute.toString().padLeft(2, '0')}'
        '${now.second.toString().padLeft(2, '0')}';

    final baseName = 'MiniTIAS_$timestamp';
    final candidate = '$baseName.png';

    if (!await File('$_basePath/$candidate').exists()) {
      return candidate;
    }

    var suffix = 1;
    while (await File('$_basePath/${baseName}_$suffix.png').exists()) {
      suffix++;
    }
    return '${baseName}_$suffix.png';
  }

  /// テスト用: タイムスタンプとファイル名一覧からファイル名を生成する.
  static String generateFileNameSync(
    String timestamp,
    List<String> existingFiles,
  ) {
    final baseName = 'MiniTIAS_$timestamp';
    final candidate = '$baseName.png';

    if (!existingFiles.contains(candidate)) {
      return candidate;
    }

    var suffix = 1;
    while (existingFiles.contains('${baseName}_$suffix.png')) {
      suffix++;
    }
    return '${baseName}_$suffix.png';
  }

  /// YUV420 を RGB に変換し PNG エンコードする(isolate 内で実行).
  static Uint8List _convertYuvToPng(Map<String, dynamic> params) {
    final width = params['width'] as int;
    final height = params['height'] as int;
    final yPlane = params['yPlane'] as Uint8List;
    final uPlane = params['uPlane'] as Uint8List;
    final vPlane = params['vPlane'] as Uint8List;
    final yRowStride = params['yRowStride'] as int;
    final uvRowStride = params['uvRowStride'] as int;
    final uvPixelStride = params['uvPixelStride'] as int;

    final image = img.Image(width: width, height: height);

    for (var y = 0; y < height; y++) {
      for (var x = 0; x < width; x++) {
        final yIndex = y * yRowStride + x;
        final uvIndex = (y ~/ 2) * uvRowStride + (x ~/ 2) * uvPixelStride;

        final yValue = yPlane[yIndex];
        final uValue = uPlane[uvIndex];
        final vValue = vPlane[uvIndex];

        final r = (yValue + 1.402 * (vValue - 128)).round().clamp(0, 255);
        final g = (yValue - 0.344 * (uValue - 128) - 0.714 * (vValue - 128))
            .round()
            .clamp(0, 255);
        final b = (yValue + 1.772 * (uValue - 128)).round().clamp(0, 255);

        image.setPixelRgb(x, y, r, g, b);
      }
    }

    return Uint8List.fromList(img.encodePng(image));
  }
}