diff --git a/pkgs/collection/CHANGELOG.md b/pkgs/collection/CHANGELOG.md index e38e1a8c..31ac61e2 100644 --- a/pkgs/collection/CHANGELOG.md +++ b/pkgs/collection/CHANGELOG.md @@ -4,6 +4,7 @@ - Shuffle `IterableExtension.sample` results. - Fix `mergeSort` when the runtime iterable generic is a subtype of the static generic. +- `CanonicalizedMap`: added constructor `fromEntries`. - Require Dart `^3.1.0` - Mark "mixin" classes as `mixin`. diff --git a/pkgs/collection/lib/src/canonicalized_map.dart b/pkgs/collection/lib/src/canonicalized_map.dart index 84909246..41038431 100644 --- a/pkgs/collection/lib/src/canonicalized_map.dart +++ b/pkgs/collection/lib/src/canonicalized_map.dart @@ -46,6 +46,23 @@ class CanonicalizedMap implements Map { addAll(other); } + /// Creates a canonicalized map that is initialized with the key/value pairs + /// of [entries]. + /// + /// The [canonicalize] function should return the canonical value for the + /// given key. Keys with the same canonical value are considered equivalent. + /// + /// The [isValidKey] function is called before calling [canonicalize] for + /// methods that take arbitrary objects. It can be used to filter out keys + /// that can't be canonicalized. + CanonicalizedMap.fromEntries( + Iterable> entries, C Function(K key) canonicalize, + {bool Function(K key)? isValidKey}) + : _canonicalize = canonicalize, + _isValidKeyFn = isValidKey { + addEntries(entries); + } + CanonicalizedMap._( this._canonicalize, this._isValidKeyFn, Map> base) { _base.addAll(base); diff --git a/pkgs/collection/test/canonicalized_map_test.dart b/pkgs/collection/test/canonicalized_map_test.dart index 9ae46579..aadb7346 100644 --- a/pkgs/collection/test/canonicalized_map_test.dart +++ b/pkgs/collection/test/canonicalized_map_test.dart @@ -205,6 +205,24 @@ void main() { }); }); + group('CanonicalizedMap.fromEntries', () { + test('canonicalizes its keys', () { + var map = CanonicalizedMap.fromEntries( + {'1': 'value 1', '2': 'value 2', '3': 'value 3'}.entries, int.parse); + expect(map['01'], equals('value 1')); + expect(map['02'], equals('value 2')); + expect(map['03'], equals('value 3')); + }); + + test('uses the final value for collisions', () { + var map = CanonicalizedMap.fromEntries( + {'1': 'value 1', '01': 'value 2', '001': 'value 3'}.entries, + int.parse); + expect(map.length, equals(1)); + expect(map['0001'], equals('value 3')); + }); + }); + group('CanonicalizedMap.toMapOfCanonicalKeys', () { test('convert to a `Map`', () { var map = CanonicalizedMap.from(