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;
 | 
						|
  }
 | 
						|
}
 |