forked from Cutlery/immich
		
	* compare different sha1 implementations * remove openssl sha1 * sync via checksum * hash assets in batches * hash in background, show spinner in tab * undo tmp changes * migrate by clearing assets * ignore duplicate assets * error handling * trigger sync/merge after download and update view * review feedback improvements * hash in background isolate on iOS * rework linking assets with existing from DB * fine-grained errors on unique index violation * hash lenth validation * revert compute in background on iOS * ignore duplicate assets on device * fix bug with batching based on accumulated size --------- Co-authored-by: Fynn Petersen-Frey <zoodyy@users.noreply.github.com>
		
			
				
	
	
		
			63 lines
		
	
	
		
			1.4 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			63 lines
		
	
	
		
			1.4 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:typed_data';
 | |
| 
 | |
| import 'package:collection/collection.dart';
 | |
| 
 | |
| extension DurationExtension on String {
 | |
|   Duration? toDuration() {
 | |
|     try {
 | |
|       final parts = split(':')
 | |
|           .map((e) => double.parse(e).toInt())
 | |
|           .toList(growable: false);
 | |
|       return Duration(hours: parts[0], minutes: parts[1], seconds: parts[2]);
 | |
|     } catch (e) {
 | |
|       return null;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   double toDouble() {
 | |
|     return double.parse(this);
 | |
|   }
 | |
| 
 | |
|   int toInt() {
 | |
|     return int.parse(this);
 | |
|   }
 | |
| }
 | |
| 
 | |
| extension ListExtension<E> on List<E> {
 | |
|   List<E> uniqueConsecutive({
 | |
|     int Function(E a, E b)? compare,
 | |
|     void Function(E a, E b)? onDuplicate,
 | |
|   }) {
 | |
|     compare ??= (E a, E b) => a == b ? 0 : 1;
 | |
|     int i = 1, j = 1;
 | |
|     for (; i < length; i++) {
 | |
|       if (compare(this[i - 1], this[i]) != 0) {
 | |
|         if (i != j) {
 | |
|           this[j] = this[i];
 | |
|         }
 | |
|         j++;
 | |
|       } else if (onDuplicate != null) {
 | |
|         onDuplicate(this[i - 1], this[i]);
 | |
|       }
 | |
|     }
 | |
|     length = length == 0 ? 0 : j;
 | |
|     return this;
 | |
|   }
 | |
| 
 | |
|   ListSlice<E> nestedSlice(int start, int end) {
 | |
|     if (this is ListSlice) {
 | |
|       final ListSlice<E> self = this as ListSlice<E>;
 | |
|       return ListSlice<E>(self.source, self.start + start, self.start + end);
 | |
|     }
 | |
|     return ListSlice<E>(this, start, end);
 | |
|   }
 | |
| }
 | |
| 
 | |
| extension IntListExtension on Iterable<int> {
 | |
|   Int64List toInt64List() {
 | |
|     final list = Int64List(length);
 | |
|     list.setAll(0, this);
 | |
|     return list;
 | |
|   }
 | |
| }
 |