/* * SPDX-FileCopyrightText: 2019-2021 Vishesh Handa * * SPDX-License-Identifier: AGPL-3.0-or-later */ import 'package:flutter/foundation.dart'; import 'package:dart_git/plumbing/git_hash.dart'; import 'package:path/path.dart' as p; import 'package:quiver/core.dart'; import 'package:gitjournal/utils/datetime.dart'; export 'package:dart_git/plumbing/git_hash.dart'; class File { final GitHash oid; final String repoPath; final String filePath; String get fullFilePath => p.join(repoPath, filePath); final DateTime modified; final DateTime created; // Maybe attach the entire GitFileIndex? final DateTime fileLastModified; // I would attach the entire file stat's result File({ required this.oid, required this.filePath, required this.repoPath, required this.modified, required this.created, required this.fileLastModified, }) { assert(repoPath.isNotEmpty); assert(filePath.isNotEmpty); } @visibleForTesting File.short(this.filePath, this.repoPath) : oid = GitHash.zero(), fileLastModified = DateTime.now(), modified = DateTime.now(), created = DateTime.now() { assert(!filePath.startsWith(p.separator)); assert(repoPath.startsWith(p.separator)); assert(repoPath.endsWith(p.separator)); } File.empty({required this.repoPath}) : filePath = '', oid = GitHash.zero(), fileLastModified = DateTime.now(), modified = DateTime.now(), created = DateTime.now() { assert(repoPath.startsWith(p.separator)); assert(repoPath.endsWith(p.separator)); } String get fileName => p.basename(filePath); File copyFile({ GitHash? oid, String? filePath, DateTime? modified, DateTime? created, DateTime? fileLastModified, }) { assert(repoPath.isNotEmpty); assert(filePath != null ? filePath.isNotEmpty : true); return File( oid: oid ?? this.oid, repoPath: repoPath, filePath: filePath ?? this.filePath, modified: modified ?? this.modified, created: created ?? this.created, fileLastModified: fileLastModified ?? this.fileLastModified, ); } @override bool operator ==(Object other) => identical(this, other) || other is File && runtimeType == other.runtimeType && oid == other.oid && repoPath == other.repoPath && filePath == other.filePath && modified == other.modified && created == other.created && fileLastModified == other.fileLastModified; @override int get hashCode => hashObjects( [oid, repoPath, filePath, created, modified, fileLastModified]); @override String toString() => 'File{oid: $oid, filePath: $filePath, created: $created, modified: $modified, fileLastModified: $fileLastModified}'; // Add toString Map toMap() { return { 'oid': oid.toString(), 'repoPath': repoPath, 'filePath': filePath, 'modified': modified.toIso8601String(), 'created': created.toIso8601String(), 'fileLastModified': fileLastModified.toIso8601String(), }; } static File fromMap(Map map) { // oid var oidV = map['oid']; if (oidV == null) { return throw Exception('Missing oid'); } if (oidV is! String) { return throw Exception('Invalid Type oid'); } var oid = GitHash(oidV); // repoPath var repoPath = map['repoPath']; if (repoPath == null) { return throw Exception('Missing repoPath'); } if (repoPath is! String) { return throw Exception('Invalid Type repoPath'); } // filePath var filePath = map['filePath']; if (filePath == null) { return throw Exception('Missing filePath'); } if (filePath is! String) { return throw Exception('Invalid Type filePath'); } // modified DateTime? modified; var modifiedV = map['modified']; if (modifiedV != null) { modified = parseDateTime(modifiedV); } if (modified == null) { return throw Exception('Failed to parse modified'); } // created DateTime? created; var createdV = map['created']; if (createdV != null) { created = parseDateTime(createdV); } if (created == null) { return throw Exception('Failed to parse created'); } // fileLastModified DateTime? fileLastModified; var fileLastModifiedV = map['fileLastModified']; if (fileLastModifiedV != null) { fileLastModified = parseDateTime(fileLastModifiedV); if (fileLastModified == null) { return throw Exception('Invalid Type fileLastModified'); } } else { return throw Exception('Missing fileLastModified'); } return File( oid: oid, repoPath: repoPath, filePath: filePath, modified: modified, created: created, fileLastModified: fileLastModified, ); } } // on save - compute hash, and commit, only then should we add it to the parent // dir? -> This can get expensive! // // FolderView -> make it show IgnoredFiles as well // -> When clicking on an IgnoredFile allow it to be opened // via the RawEditor // RawEditor(File file) -> add supports(File file) // TextFile can have the (text + encoding) + File // alternatives -> body / data // Image should also be a file // The NewNote FAB can have more options .. ? // Remove the 'Note' class entrirely // With this, we can easily add 'grpc' support for the 'loaders' of each of // these 'Files'. // And then the only thing left would be 'Repository' and 'RepositoryManager'