diff --git a/.github/workflows/dart_ci.yml b/.github/workflows/dart_ci.yml index 5165d742..1b2548f5 100644 --- a/.github/workflows/dart_ci.yml +++ b/.github/workflows/dart_ci.yml @@ -46,7 +46,7 @@ jobs: strategy: fail-fast: false matrix: - sdk: [ 2.13.4, stable ] + sdk: [ 2.18.7, 2.19.6 ] steps: - uses: actions/checkout@v2 - uses: dart-lang/setup-dart@v0.2 @@ -71,14 +71,6 @@ jobs: name: ddc-test-results@${{ matrix.sdk }} path: reports/${{ matrix.sdk }}/ddc/test-results.json - - name: Report Unit Test Results - uses: dorny/test-reporter@v1 - if: ${{ always() && steps.install.outcome == 'success' }} - with: - name: Unit Test Results (ddc - ${{ matrix.sdk }}) - path: 'reports/${{ matrix.sdk }}/ddc/test-results.json' - reporter: dart-json - test_dart2js: permissions: id-token: write @@ -112,11 +104,3 @@ jobs: with: name: dart2js-test-results@${{ matrix.sdk }} path: reports/${{ matrix.sdk }}/dart2js/test-results.json - - - name: Report Unit Test Results - uses: dorny/test-reporter@v1 - if: ${{ always() && steps.install.outcome == 'success' }} - with: - name: Unit Test Results (dart2js - ${{ matrix.sdk }}) - path: 'reports/${{ matrix.sdk }}/dart2js/test-results.json' - reporter: dart-json diff --git a/lib/src/over_react_test/common_component_util.dart b/lib/src/over_react_test/common_component_util.dart index 027be519..17bf0dfd 100644 --- a/lib/src/over_react_test/common_component_util.dart +++ b/lib/src/over_react_test/common_component_util.dart @@ -17,18 +17,17 @@ import 'dart:html'; import 'dart:js'; import 'package:meta/meta.dart'; -import 'package:over_react/over_react.dart'; import 'package:over_react/component_base.dart' as component_base; +import 'package:over_react/over_react.dart'; import 'package:over_react_test/over_react_test.dart'; import 'package:over_react_test/src/over_react_test/props_meta.dart'; import 'package:over_react_test/src/over_react_test/test_helpers.dart'; import 'package:react/react_client.dart'; +import 'package:react/react_client/js_backed_map.dart'; import 'package:react/react_client/react_interop.dart'; import 'package:react/react_test_utils.dart' as react_test_utils; import 'package:test/test.dart'; -import './custom_matchers.dart'; -import './react_util.dart'; import 'dart_util.dart'; /// Run common component tests around default props, prop forwarding, class name merging, and class name overrides. @@ -81,16 +80,16 @@ void commonComponentTests(BuilderOnlyUiFactory factory, { bool shouldTestPropForwarding = true, List unconsumedPropKeys = const [], List skippedPropKeys = const [], - List Function(PropsMetaCollection) getUnconsumedPropKeys, - List Function(PropsMetaCollection) getSkippedPropKeys, + List Function(PropsMetaCollection)? getUnconsumedPropKeys, + List Function(PropsMetaCollection)? getSkippedPropKeys, Map nonDefaultForwardingTestProps = const {}, bool shouldTestClassNameMerging = true, bool shouldTestClassNameOverrides = true, bool ignoreDomProps = true, bool shouldTestRequiredProps = true, @Deprecated('This flag is not needed as the test will auto detect the version') - bool isComponent2, - dynamic childrenFactory() + bool? isComponent2, + dynamic childrenFactory()? }) { childrenFactory ??= _defaultChildrenFactory; @@ -132,14 +131,14 @@ Iterable _flatten(Iterable iterable) => /// }); /// } void expectCleanTestSurfaceAtEnd() { - Set nodesBefore; + late Set nodesBefore; setUpAll(() { - nodesBefore = document.body.children.toSet(); + nodesBefore = document.body!.children.toSet(); }); tearDownAll(() { - Set nodesAfter = document.body.children.toSet(); + Set nodesAfter = document.body!.children.toSet(); var nodesAdded = nodesAfter.difference(nodesBefore).map((element) => element.outerHtml).toList(); expect(nodesAdded, isEmpty, reason: 'tests should leave the test surface clean.'); @@ -161,12 +160,12 @@ void expectCleanTestSurfaceAtEnd() { /// todo make this public again if there's a need to expose it, once the API has stabilized @isTest void _testPropForwarding(BuilderOnlyUiFactory factory, dynamic childrenFactory(), { - @required List unconsumedPropKeys, - @required List skippedPropKeys, - @required List Function(PropsMetaCollection) getUnconsumedPropKeys, - @required List Function(PropsMetaCollection) getSkippedPropKeys, - @required bool ignoreDomProps, - @required Map nonDefaultForwardingTestProps, + required List unconsumedPropKeys, + required List skippedPropKeys, + required List Function(PropsMetaCollection)? getUnconsumedPropKeys, + required List Function(PropsMetaCollection)? getSkippedPropKeys, + required bool ignoreDomProps, + required Map nonDefaultForwardingTestProps, }) { testFunction('forwards unconsumed props as expected', () { // This needs to be retrieved inside a `test`/`setUp`/etc, not inside a group, @@ -335,13 +334,13 @@ void _testPropForwarding(BuilderOnlyUiFactory factory, dynamic childrenFactory() /// Test for prop keys that both are forwarded and exist on the forwarding target's default props. if (isDartComponent(forwardingTarget)) { final forwardingTargetType = (forwardingTarget as ReactElement).type as ReactClass; - Map forwardingTargetDefaults; + late Map forwardingTargetDefaults; switch (forwardingTargetType.dartComponentVersion) { // ignore: invalid_use_of_protected_member case ReactDartComponentVersion.component: // ignore: invalid_use_of_protected_member - forwardingTargetDefaults = forwardingTargetType.dartDefaultProps; // ignore: deprecated_member_use + forwardingTargetDefaults = forwardingTargetType.dartDefaultProps!; // ignore: deprecated_member_use break; case ReactDartComponentVersion.component2: // ignore: invalid_use_of_protected_member - forwardingTargetDefaults = JsBackedMap.backedBy(forwardingTargetType.defaultProps); + forwardingTargetDefaults = JsBackedMap.backedBy(forwardingTargetType.defaultProps!); break; } @@ -427,7 +426,7 @@ void testClassNameMerging(BuilderOnlyUiFactory factory, dynamic childrenFactory( ..classNameBlacklist = 'blacklisted-class-1 blacklisted-class-2'; var renderedInstance = render(builder(childrenFactory())); - Iterable forwardingTargetNodes = getForwardingTargets(renderedInstance).map(findDomNode); + final forwardingTargetNodes = getForwardingTargets(renderedInstance).map(findDomNode); expect(forwardingTargetNodes, everyElement( allOf( @@ -461,7 +460,7 @@ void testClassNameMerging(BuilderOnlyUiFactory factory, dynamic childrenFactory( /// > Related: [testClassNameMerging] @isTestGroup void testClassNameOverrides(BuilderOnlyUiFactory factory, dynamic childrenFactory()) { - Set classesToOverride; + late Set classesToOverride; var error; setUp(() { @@ -477,7 +476,7 @@ void testClassNameOverrides(BuilderOnlyUiFactory factory, dynamic childrenFactor // but still fail the test if something goes wrong. try { classesToOverride = getForwardingTargets(reactInstanceWithoutOverrides) - .map((target) => findDomNode(target).classes) + .map((target) => findDomNode(target)!.classes) .expand((CssClassSet classSet) => classSet) .toSet(); } catch(e) { @@ -500,7 +499,7 @@ void testClassNameOverrides(BuilderOnlyUiFactory factory, dynamic childrenFactor )(childrenFactory()) ); - Iterable forwardingTargetNodes = getForwardingTargets(reactInstance).map(findDomNode); + final forwardingTargetNodes = getForwardingTargets(reactInstance).map(findDomNode); expect(forwardingTargetNodes, everyElement( hasExactClasses('') ), reason: '$classesToOverride not overridden'); @@ -514,7 +513,7 @@ void testClassNameOverrides(BuilderOnlyUiFactory factory, dynamic childrenFactor /// __Note__: All required props must be provided by [factory]. @isTestGroup void testRequiredProps(BuilderOnlyUiFactory factory, dynamic childrenFactory()) { - bool isComponent2; + late bool isComponent2; var keyToErrorMessage = {}; var nullableProps = []; @@ -531,7 +530,7 @@ void testRequiredProps(BuilderOnlyUiFactory factory, dynamic childrenFactory()) isComponent2 = version == ReactDartComponentVersion.component2; var jacket = mount(factory()(childrenFactory()), autoTearDown: false); - var consumedProps = (jacket.getDartInstance() as component_base.UiComponent).consumedProps; + var consumedProps = (jacket.getDartInstance() as component_base.UiComponent).consumedProps!; jacket.unmount(); for (var consumedProp in consumedProps) { @@ -542,7 +541,7 @@ void testRequiredProps(BuilderOnlyUiFactory factory, dynamic childrenFactory()) requiredProps.add(prop.key); } - keyToErrorMessage[prop.key] = prop.errorMessage ?? ''; + keyToErrorMessage[prop.key] = prop.errorMessage; } } }); @@ -580,8 +579,8 @@ void testRequiredProps(BuilderOnlyUiFactory factory, dynamic childrenFactory()) void component2RequiredPropsTest() { PropTypes.resetWarningCache(); - List consoleErrors = []; - JsFunction originalConsoleError = context['console']['error']; + var consoleErrors = []; + final originalConsoleError = context['console']['error'] as JsFunction; addTearDown(() => context['console']['error'] = originalConsoleError); context['console']['error'] = JsFunction.withThis((self, [message, arg1, arg2, arg3, arg4, arg5]) { consoleErrors.add(message); @@ -590,7 +589,7 @@ void testRequiredProps(BuilderOnlyUiFactory factory, dynamic childrenFactory()) }); final reactComponentFactory = factory().componentFactory as - ReactDartComponentFactoryProxy2; // ignore: avoid_as + ReactDartComponentFactoryProxy2; // ignore: avoid_as for (var propKey in requiredProps) { if (!reactComponentFactory.defaultProps.containsKey(propKey)) { @@ -657,8 +656,8 @@ void testRequiredProps(BuilderOnlyUiFactory factory, dynamic childrenFactory()) } else { PropTypes.resetWarningCache(); - List consoleErrors = []; - JsFunction originalConsoleError = context['console']['error']; + var consoleErrors = []; + final originalConsoleError = context['console']['error'] as JsFunction; addTearDown(() => context['console']['error'] = originalConsoleError); context['console']['error'] = JsFunction.withThis((self, [message, arg1, arg2, arg3, arg4, arg5]) { consoleErrors.add(message); diff --git a/lib/src/over_react_test/console_log_utils.dart b/lib/src/over_react_test/console_log_utils.dart index 4d485f64..0d0a63a3 100644 --- a/lib/src/over_react_test/console_log_utils.dart +++ b/lib/src/over_react_test/console_log_utils.dart @@ -34,12 +34,12 @@ import 'package:react/react_client/react_interop.dart'; /// props that are not valid, the errors will be caught to allow the test to complete. /// /// To handle asynchronous behavior, see [recordConsoleLogsAsync]. -List recordConsoleLogs( +List recordConsoleLogs( Function() callback, { ConsoleConfiguration configuration = allConfig, bool shouldResetPropTypesWarningCache = true, }) { - final consoleLogs = []; + final consoleLogs = []; final logTypeToCapture = configuration.logType == 'all' ? ConsoleConfiguration.types : [configuration.logType]; @@ -54,7 +54,7 @@ List recordConsoleLogs( // NOTE: Using console.log or print within this function will cause an infinite // loop when the logType is set to `log`. consoleLogs.add(message); - consoleRefs[config] + consoleRefs[config]! .apply([message, arg1, arg2, arg3, arg4, arg5], thisArg: self); }); } @@ -81,12 +81,12 @@ List recordConsoleLogs( /// with the exception being the provided callback should be asynchronous. /// /// Related: [recordConsoleLogs] -FutureOr> recordConsoleLogsAsync( +FutureOr> recordConsoleLogsAsync( Future Function() asyncCallback, { ConsoleConfiguration configuration = allConfig, bool shouldResetPropTypesWarningCache = true, }) async { - var consoleLogs = []; + var consoleLogs = []; final logTypeToCapture = configuration.logType == 'all' ? ConsoleConfiguration.types : [configuration.logType]; @@ -101,7 +101,7 @@ FutureOr> recordConsoleLogsAsync( // NOTE: Using console.log or print within this function will cause an infinite // loop when the logType is set to `log`. consoleLogs.add(message); - consoleRefs[config] + consoleRefs[config]! .apply([message, arg1, arg2, arg3, arg4, arg5], thisArg: self); }); } diff --git a/lib/src/over_react_test/custom_matchers.dart b/lib/src/over_react_test/custom_matchers.dart index 181037a2..7f9abe96 100644 --- a/lib/src/over_react_test/custom_matchers.dart +++ b/lib/src/over_react_test/custom_matchers.dart @@ -15,8 +15,8 @@ import 'dart:html'; import 'dart:svg'; +import 'package:collection/collection.dart' show IterableNullableExtension; import 'package:over_react/over_react.dart'; -import 'package:matcher/matcher.dart'; import 'package:over_react_test/src/over_react_test/dart_util.dart'; import 'package:react/react.dart' as react; import 'package:react/react_test_utils.dart' as react_test_utils; @@ -46,7 +46,7 @@ class ClassNameMatcher extends Matcher { static Iterable getClassIterable(dynamic classNames) { Iterable classes; if (classNames is Iterable) { - classes = classNames.where((className) => className != null).expand(splitSpaceDelimitedString); + classes = classNames.whereNotNull().expand(splitSpaceDelimitedString); } else if (classNames is String) { classes = splitSpaceDelimitedString(classNames); } else { @@ -61,7 +61,7 @@ class ClassNameMatcher extends Matcher { // There's a bug in DDC where, though the docs say `className` should // return `String`, it will return `AnimatedString` for `SvgElement`s. See // https://github.com/dart-lang/sdk/issues/36200. - String castClassName; + String? castClassName; if (className is String) { castClassName = className; } else if (className is AnimatedString) { @@ -164,7 +164,7 @@ class _HasToStringValue extends CustomMatcher { _HasToStringValue(matcher) : super('Object with toString() value', 'toString()', matcher); @override - featureValueOf(Object item) => item.toString(); + featureValueOf(Object? item) => item.toString(); } class _HasPropMatcher extends CustomMatcher { @@ -187,7 +187,7 @@ class _HasPropMatcher extends CustomMatcher { @override Map featureValueOf(item) { - if (_useDomAttributes(item)) return findDomNode(item).attributes; + if (_useDomAttributes(item)) return findDomNode(item)!.attributes; if (item is react.Component) return item.props; return getProps(item); @@ -269,7 +269,7 @@ class _IsFocused extends Matcher { ..add('is not a valid Element.'); } - if (!document.documentElement.contains(item)) { + if (!document.documentElement!.contains(item)) { return mismatchDescription ..add('is not attached to the document, and thus cannot be focused.') ..add(' If testing with React, you can use `renderAttachedToDocument`.'); @@ -314,7 +314,7 @@ Matcher throwsPropError(String propName, [String message = '']) { /// /// __Note__: The [message] is matched rather than the [Error] instance due to Dart's wrapping of all `throw` /// as a [DomException] -Matcher throwsPropError_Required(String propName, [String message = '']) { +Matcher throwsPropError_Required(String propName, [String? message = '']) { return throwsA(anyOf( hasToStringValue('V8 Exception'), /* workaround for https://github.com/dart-lang/sdk/issues/26093 */ hasToStringValue(contains('RequiredPropError: Prop $propName is required. $message'.trim())) @@ -362,10 +362,10 @@ Matcher throwsPropError_Combination(String propName, String prop2Name, [String m /// and pass it to [recordConsoleLogs] to run the function and record the resulting /// logs that are emitted during the function runtime. class _LoggingFunctionMatcher extends CustomMatcher { - _LoggingFunctionMatcher(dynamic matcher, {this.config, String description, String name, bool onlyIfAssertsAreEnabled = false}) + _LoggingFunctionMatcher(dynamic matcher, {this.config, String? description, String? name, bool onlyIfAssertsAreEnabled = false}) : super(description ?? 'emits the logs', name ?? 'logs', _wrapMatcherForSingleLog(matcher, onlyIfAssertsAreEnabled)); - final ConsoleConfiguration config; + final ConsoleConfiguration? config; static dynamic _wrapMatcherForSingleLog(dynamic expected, [bool onlyIfAssertsAreEnabled = false]) { if (onlyIfAssertsAreEnabled && !assertsEnabled()) return anything; @@ -375,17 +375,13 @@ class _LoggingFunctionMatcher extends CustomMatcher { @override featureValueOf(actual) { - var logs = []; - if (actual is List) return actual; if (actual is! Function()) { throw ArgumentError('The actual value must be a callback or a List.'); } - logs = recordConsoleLogs(actual, configuration: config ?? logConfig); - - return logs; + return recordConsoleLogs(actual, configuration: config ?? logConfig); } } @@ -400,7 +396,7 @@ class _LoggingFunctionMatcher extends CustomMatcher { /// caught error. /// /// Related: [logsToConsole], [hasNoLogs] -Matcher hasLog(dynamic expected, {ConsoleConfiguration consoleConfig, bool onlyIfAssertsAreEnabled = false}) => +Matcher hasLog(dynamic expected, {ConsoleConfiguration? consoleConfig, bool onlyIfAssertsAreEnabled = false}) => _LoggingFunctionMatcher(anyElement(contains(expected)), config: consoleConfig, onlyIfAssertsAreEnabled: onlyIfAssertsAreEnabled); /// A Matcher used to compare a list of logs against a provided matcher. @@ -440,7 +436,7 @@ Matcher hasLog(dynamic expected, {ConsoleConfiguration consoleConfig, bool onlyI /// ``` /// /// Related: [hasLog], [hasNoLogs] -Matcher logsToConsole(dynamic expected, {ConsoleConfiguration consoleConfig, bool onlyIfAssertsAreEnabled = false}) => +Matcher logsToConsole(dynamic expected, {ConsoleConfiguration? consoleConfig, bool onlyIfAssertsAreEnabled = false}) => _LoggingFunctionMatcher(expected, config: consoleConfig, onlyIfAssertsAreEnabled: onlyIfAssertsAreEnabled); /// A matcher to verify that a callback function does not emit any logs. diff --git a/lib/src/over_react_test/dom_util.dart b/lib/src/over_react_test/dom_util.dart index f54712a2..ed2b1e97 100644 --- a/lib/src/over_react_test/dom_util.dart +++ b/lib/src/over_react_test/dom_util.dart @@ -61,7 +61,7 @@ void triggerDocumentClick(Element target) { /// /// Verifies that the [target] element is not a detached node. void triggerDocumentMouseEvent(Element target, String event) { - if (!document.documentElement.contains(target)) { + if (!document.documentElement!.contains(target)) { throw ArgumentError.value(target, 'target', 'Target should be attached to the document.'); } @@ -76,7 +76,7 @@ void triggerDocumentMouseEvent(Element target, String event) { /// /// See: . Future triggerFocus(Element target, {Duration timeout = _defaultTriggerTimeout}) { - if (!document.documentElement.contains(target)) { + if (!document.documentElement!.contains(target)) { throw ArgumentError.value(target, 'target', 'Target should be attached to the document.'); } diff --git a/lib/src/over_react_test/jacket.dart b/lib/src/over_react_test/jacket.dart index fb0b7a08..9af364cf 100644 --- a/lib/src/over_react_test/jacket.dart +++ b/lib/src/over_react_test/jacket.dart @@ -15,12 +15,11 @@ import 'dart:html'; import 'package:over_react/over_react.dart' as over_react; +import 'package:over_react_test/src/over_react_test/react_util.dart' as react_util; import 'package:react/react.dart' as react; import 'package:react/react_client/react_interop.dart' show ReactComponent; import 'package:react/react_test_utils.dart' as react_test_utils; -import 'package:over_react_test/src/over_react_test/react_util.dart' as react_util; - import './zone_util.dart'; // Notes @@ -46,7 +45,7 @@ import './zone_util.dart'; /// /// To have the instance not automatically unmounted when the test if over set [autoTearDown] to `false`. TestJacket mount(over_react.ReactElement reactElement, { - Element mountNode, + Element? mountNode, bool attachedToDocument = false, bool autoTearDown = true }) { @@ -60,14 +59,14 @@ TestJacket mount(over_react.ReactElement reactElem /// Provides more a more consistent and easier to use API to test and manipulate a rendered [ReactComponent]. class TestJacket { - TestJacket._(over_react.ReactElement reactElement, {Element mountNode, this.attachedToDocument = false, this.autoTearDown = true}) + TestJacket._(over_react.ReactElement reactElement, {Element? mountNode, this.attachedToDocument = false, this.autoTearDown = true}) : mountNode = mountNode ?? (DivElement() ..style.height = '800px' ..style.width = '800px') { _render(reactElement); } - /* [1] */ Object _renderedInstance; + /* [1] */ Object? _renderedInstance; final Element mountNode; final bool attachedToDocument; final bool autoTearDown; @@ -169,7 +168,7 @@ class TestJacket { /// > ``` /// > jacket.mountNode.querySelector(someSelectorThatTargetsTheRootNode); /// > ``` - Element getNode() { + Element? getNode() { if (!_isCompositeComponent && !_isDomComponent) { throw StateError(over_react.unindent(''' getNode() is only supported when the rendered object is a DOM or composite (class based) component. @@ -192,7 +191,7 @@ class TestJacket { /// > If you are rendering a function component using [mount], calling [getDartInstance] will throw a `StateError`. /// > /// > See [getInstance] for more information about this limitation. - T getDartInstance() { + T? getDartInstance() { // [1] Adding an additional check for dom components here because the current behavior when `_renderedInstance` is // a DOM component (Element) - is to return `null`. While that will most likely cause null exceptions once the // consumer attempts to make a call on the "Dart instance" they have requested - we don't want this change @@ -203,7 +202,7 @@ class TestJacket { 'getDartInstance() is only supported when the rendered object is a composite (class based) component.'); } - return over_react.getDartComponent(_renderedInstance) as T; + return over_react.getDartComponent(_renderedInstance); } /// Returns if the jacket component is mounted or not. @@ -220,20 +219,20 @@ class TestJacket { /// > If you are rendering a function or DOM component using [mount], calling [setState] will throw a `StateError`. /// > /// > See [getInstance] for more information about this limitation. - void setState(newState, [callback()]) { + void setState(newState, [callback()?]) { if (!_isCompositeComponent) { throw StateError( 'setState() is only supported when the rendered object is a composite (class based) component.'); } - getDartInstance().setState(newState, callback); + getDartInstance()!.setState(newState, callback); } /// Unmounts the React component instance and cleans up any attached DOM nodes. void unmount() { _isMounted = false; react_util.unmount(_renderedInstance); - mountNode?.remove(); + mountNode.remove(); react_util.tearDownAttachedNodes(); } } diff --git a/lib/src/over_react_test/props_meta.dart b/lib/src/over_react_test/props_meta.dart index 8089a45a..d3e5b67c 100644 --- a/lib/src/over_react_test/props_meta.dart +++ b/lib/src/over_react_test/props_meta.dart @@ -20,7 +20,7 @@ import 'package:react/react_client/react_interop.dart'; /// /// Returns `null` if [el] does not render a UiComponent2 or does not use the /// new mixin syntax (determined by whether accessing propsMeta throws). -PropsMetaCollection getPropsMeta(ReactElement el) { +PropsMetaCollection? getPropsMeta(ReactElement el) { // ignore: invalid_use_of_protected_member final isComponent2 = ReactDartComponentVersion.fromType(el.type) == '2'; if (!isComponent2) return null; diff --git a/lib/src/over_react_test/react_util.dart b/lib/src/over_react_test/react_util.dart index ecd1c483..74da8c84 100644 --- a/lib/src/over_react_test/react_util.dart +++ b/lib/src/over_react_test/react_util.dart @@ -18,14 +18,13 @@ library over_react_test.react_util; import 'dart:collection'; import 'dart:html'; +import 'package:collection/collection.dart' show IterableNullableExtension; import 'package:js/js.dart'; -import 'package:over_react/over_react.dart'; import 'package:over_react/component_base.dart' as component_base; +import 'package:over_react/over_react.dart'; import 'package:react/react.dart' as react; -import 'package:react/react_dom.dart' as react_dom; -import 'package:react/react_client.dart'; -import 'package:react/react_client/react_interop.dart'; import 'package:react/react_client/js_interop_helpers.dart'; +import 'package:react/react_dom.dart' as react_dom; import 'package:react/react_test_utils.dart' as react_test_utils; import 'package:test/test.dart'; @@ -55,8 +54,8 @@ export 'package:over_react/src/util/react_wrappers.dart'; /// once the component has been unmounted. /* [1] */ render(dynamic component, {bool autoTearDown = true, - Element container, - Callback autoTearDownCallback}) { + Element? container, + Callback? autoTearDownCallback}) { var renderedInstance; component = component is component_base.UiProps ? component() : component; @@ -84,7 +83,7 @@ export 'package:over_react/src/util/react_wrappers.dart'; /// [autoTearDown] to false. /// /// See: . -ReactElement renderShallow(ReactElement instance, {bool autoTearDown = true, Callback autoTearDownCallback}) { +ReactElement renderShallow(ReactElement instance, {bool autoTearDown = true, Callback? autoTearDownCallback}) { var renderer = react_test_utils.createRenderer(); if (autoTearDown) { addTearDown(() { @@ -106,7 +105,7 @@ ReactElement renderShallow(ReactElement instance, {bool autoTearDown = true, Cal void unmount(dynamic instanceOrContainerNode) { if (instanceOrContainerNode == null) return; - Element containerNode; + final Element? containerNode; if (instanceOrContainerNode is Element) { containerNode = instanceOrContainerNode; @@ -136,7 +135,7 @@ void unmount(dynamic instanceOrContainerNode) { /// > If [component] is a function component, calling [renderAndGetDom] will throw a `StateError`. /// > /// > See `TestJacket.getNode` for more information about this limitation. -Element renderAndGetDom(dynamic component, {bool autoTearDown = true, Callback autoTearDownCallback}) { +Element? renderAndGetDom(dynamic component, {bool autoTearDown = true, Callback? autoTearDownCallback}) { final renderedInstance = render(component, autoTearDown: autoTearDown, autoTearDownCallback: autoTearDownCallback); if (!react_test_utils.isCompositeComponent(renderedInstance) && !react_test_utils.isDOMComponent(renderedInstance)) { @@ -152,8 +151,8 @@ Element renderAndGetDom(dynamic component, {bool autoTearDown = true, Callback a /// > If [component] is a function component, calling [renderAndGetComponent] will throw a `StateError`. /// > /// > See `TestJacket.getInstance` for more information about this limitation. -react.Component renderAndGetComponent(dynamic component, - {bool autoTearDown = true, Callback autoTearDownCallback}) { +react.Component? renderAndGetComponent(dynamic component, + {bool autoTearDown = true, Callback? autoTearDownCallback}) { final renderedInstance = render(component, autoTearDown: autoTearDown, autoTearDownCallback: autoTearDownCallback); // [1] Adding an additional check for dom components here because the current behavior when `renderedInstance` is @@ -178,28 +177,28 @@ List _attachedReactContainers = []; /// Returns the rendered component. /* [1] */ renderAttachedToDocument(dynamic component, {bool autoTearDown = true, - Element container, - Callback autoTearDownCallback}) { - container ??= DivElement() + Element? container, + Callback? autoTearDownCallback}) { + final containerElement = container ?? DivElement() // Set arbitrary height and width for container to ensure nothing is cut off. ..style.setProperty('width', '800px') ..style.setProperty('height', '800px'); setComponentZone(); - document.body.append(container); + document.body!.append(containerElement); if (autoTearDown) { addTearDown(() { - react_dom.unmountComponentAtNode(container); - container.remove(); + react_dom.unmountComponentAtNode(containerElement); + containerElement.remove(); if (autoTearDownCallback != null) autoTearDownCallback(); }); } else { - _attachedReactContainers.add(container); + _attachedReactContainers.add(containerElement); } - return react_dom.render(component is component_base.UiProps ? component.build() : component, container); + return react_dom.render(component is component_base.UiProps ? component.build() : component, containerElement); } /// Unmounts and removes the mount nodes for components rendered via [renderAttachedToDocument] that are not @@ -211,7 +210,7 @@ void tearDownAttachedNodes() { } } -typedef void _EventSimulatorAlias(componentOrNode, [Map eventData]); +typedef void _EventSimulatorAlias(componentOrNode, [Map? eventData]); /// Helper function to simulate clicks final _EventSimulatorAlias click = react_test_utils.Simulate.click; @@ -244,11 +243,11 @@ final _EventSimulatorAlias mouseDown = react_test_utils.Simulate.mouseDown; final _EventSimulatorAlias mouseUp = react_test_utils.Simulate.mouseUp; /// Helper function to simulate mouseEnter events. -final _EventSimulatorAlias mouseEnter = (componentOrNode, [Map eventData = const {}]) => +final _EventSimulatorAlias mouseEnter = (componentOrNode, [Map? eventData = const {}]) => Simulate._mouseEnter(componentOrNode, jsifyAndAllowInterop(eventData)); /// Helper function to simulate mouseLeave events. -final _EventSimulatorAlias mouseLeave = (componentOrNode, [Map eventData = const {}]) => +final _EventSimulatorAlias mouseLeave = (componentOrNode, [Map? eventData = const {}]) => Simulate._mouseLeave(componentOrNode, jsifyAndAllowInterop(eventData)); @JS('React.addons.TestUtils.Simulate') @@ -351,9 +350,9 @@ List /* < [1] > */ getAllByTestId(dynamic root, String value, {String key = defa } return react_test_utils.findAllInRenderedTree(root, allowInterop((descendant) { - Map props; + Map? props; if (react_test_utils.isDOMComponent(descendant)) { - props = findDomNode(descendant).attributes; + props = findDomNode(descendant)!.attributes; } else if (react_test_utils.isCompositeComponent(descendant)) { props = getProps(descendant); } @@ -384,7 +383,7 @@ List /* < [1] > */ getAllByTestId(dynamic root, String value, {String key = defa List getAllComponentsByTestId(dynamic root, String value, {String key = defaultTestIdKey}) => getAllByTestId(root, value, key: key) .map((element) => getDartComponent(element)) // ignore: unnecessary_lambdas - .where((component) => component != null) + .whereNotNull() .toList(); /// Returns the [Element] of the first descendant of [root] that has its [key] prop value set to [value]. @@ -415,7 +414,7 @@ List getAllComponentsByTestId(dynamic root, String /// getComponentRootDomByTestId(renderedInstance, 'value'); // returns the `outer` `
` /// /// Related: [queryByTestId]. -Element getComponentRootDomByTestId(dynamic root, String value, {String key = defaultTestIdKey}) { +Element? getComponentRootDomByTestId(dynamic root, String value, {String key = defaultTestIdKey}) { return findDomNode(getByTestId(root, value, key: key)); } @@ -450,7 +449,7 @@ Element getComponentRootDomByTestId(dynamic root, String value, {String key = de /// queryByTestId(renderedInstance, 'value'); // returns the `inner` `
` /// /// Related: [queryAllByTestId], [getComponentRootDomByTestId]. -Element queryByTestId(dynamic root, String value, {String key = defaultTestIdKey, bool searchInShadowDom = false, int shadowDepth}) { +Element? queryByTestId(dynamic root, String value, {String key = defaultTestIdKey, bool searchInShadowDom = false, int? shadowDepth}) { var results = _findDeep(findDomNode(root), _makeTestIdSelector(value, key: key), searchInShadowDom: searchInShadowDom, findMany: false, depth: shadowDepth); return results.isNotEmpty ? results.first : null; } @@ -490,25 +489,34 @@ Element queryByTestId(dynamic root, String value, {String key = defaultTestIdKey ///
/// /// queryAllByTestId(renderedInstance, 'value'); // returns both `inner` `
`s -List queryAllByTestId(dynamic root, String value, {String key = defaultTestIdKey, bool searchInShadowDom = false, int shadowDepth}) { +List queryAllByTestId(dynamic root, String value, {String key = defaultTestIdKey, bool searchInShadowDom = false, int? shadowDepth}) { return _findDeep(findDomNode(root), _makeTestIdSelector(value, key: key), searchInShadowDom: searchInShadowDom, findMany: true, depth: shadowDepth); } String _makeTestIdSelector(String value, {String key = defaultTestIdKey}) => '[$key~="$value"]'; -List _findDeep(Node root, String itemSelector, {bool searchInShadowDom = false, bool findMany = true, int depth}) { +List _findDeep(Node? root, String itemSelector, {bool searchInShadowDom = false, bool findMany = true, int? depth}) { List nodes = []; - void recursiveSeek(Node _root, int _currentDepth) { + void recursiveSeek(Node? _root, int _currentDepth) { // The LHS type prevents `rootQuerySelectorAll` from returning `_FrozenElementList>` instead of `` in DDC - final List Function(String) rootQuerySelectorAll = _root is ShadowRoot ? _root.querySelectorAll : _root is Element ? _root.querySelectorAll : null; + final List Function(String) rootQuerySelectorAll; + if ( _root is ShadowRoot) { + rootQuerySelectorAll = _root.querySelectorAll; + } else if (_root is Element) { + rootQuerySelectorAll = _root.querySelectorAll; + } else { + throw Exception('Unhandled node that is neither a ShadowRoot nor an Element: $_root'); + } nodes.addAll(rootQuerySelectorAll(itemSelector)); if (!findMany && nodes.isNotEmpty) { return; } // This method of finding shadow roots may not be performant, but it's good enough for usage in tests. if (searchInShadowDom && (depth == null || _currentDepth < depth)) { - var foundShadows = rootQuerySelectorAll('*').where((el) => el.shadowRoot != null).map((el) => el.shadowRoot).toList(); - foundShadows.forEach((shadowRoot) => recursiveSeek(shadowRoot, _currentDepth + 1)); + var foundShadows = rootQuerySelectorAll('*').map((el) => el.shadowRoot).whereNotNull().toList(); + for (var shadowRoot in foundShadows) { + recursiveSeek(shadowRoot, _currentDepth + 1); + } } } recursiveSeek(root, 0); @@ -518,7 +526,7 @@ List _findDeep(Node root, String itemSelector, {bool searchInShadowDom /// Returns the [react.Component] of the first descendant of [root] that has its [key] prop value set to [value]. /// /// Returns null if no descendant has its [key] prop value set to [value]. -react.Component getComponentByTestId(dynamic root, String value, {String key = defaultTestIdKey}) { +react.Component? getComponentByTestId(dynamic root, String value, {String key = defaultTestIdKey}) { var instance = getByTestId(root, value, key: key); if (instance != null) { return getDartComponent(instance); @@ -530,7 +538,7 @@ react.Component getComponentByTestId(dynamic root, String value, {String key = d /// Returns the props of the first descendant of [root] that has its [key] prop value set to [value]. /// /// Returns null if no descendant has its [key] prop value set to [value]. -Map getPropsByTestId(dynamic root, String value, {String key = defaultTestIdKey}) { +Map? getPropsByTestId(dynamic root, String value, {String key = defaultTestIdKey}) { var instance = getByTestId(root, value, key: key); if (instance != null) { return getProps(instance); @@ -575,9 +583,9 @@ List findDescendantsWithProp(/* [1] */ root, dynamic propKey) { return false; } - Map props; + Map? props; if (react_test_utils.isDOMComponent(descendant)) { - props = findDomNode(descendant).attributes; + props = findDomNode(descendant)!.attributes; } else if (react_test_utils.isCompositeComponent(descendant)) { props = getProps(descendant); } diff --git a/lib/src/over_react_test/validation_util.dart b/lib/src/over_react_test/validation_util.dart index 68a76b5f..1f9a9e3d 100644 --- a/lib/src/over_react_test/validation_util.dart +++ b/lib/src/over_react_test/validation_util.dart @@ -160,17 +160,23 @@ void verifyValidationWarning(dynamic warningMatcher) { /// [startRecordingValidationWarnings] was first called. /// /// > Related: [clearValidationWarnings] -List getValidationWarnings() => _validationWarnings?.toList(); +List? getValidationWarnings() => _validationWarnings?.toList(); /// Clears the list of [ValidationUtil.warn]ings that have been recorded since /// [startRecordingValidationWarnings] was first called. /// /// > Related: [getValidationWarnings] void clearValidationWarnings() { - _validationWarnings.clear(); + if (_validationWarnings == null) { + throw StateError('Need to call startRecordingValidationWarnings() first.'); + } + _validationWarnings!.clear(); } -List _validationWarnings; +List? _validationWarnings; void _recordValidationWarning(String warningMessage) { - _validationWarnings.add(warningMessage); + if (_validationWarnings == null) { + throw StateError('Need to call startRecordingValidationWarnings() first.'); + } + _validationWarnings!.add(warningMessage); } diff --git a/lib/src/over_react_test/zone_util.dart b/lib/src/over_react_test/zone_util.dart index 94d31373..d8cb97a2 100644 --- a/lib/src/over_react_test/zone_util.dart +++ b/lib/src/over_react_test/zone_util.dart @@ -17,7 +17,7 @@ import 'dart:async'; import 'package:react/react_client.dart'; import 'package:test/test.dart'; -Zone _zone; +Zone? _zone; /// Validates that [storeZone] was called before [zonedExpect] was. void validateZone() { @@ -28,20 +28,17 @@ void validateZone() { /// Store the specified _(or current if none is specified)_ [zone] /// for use within [zonedExpect]. -void storeZone([Zone zone]) { - if (zone == null) { - zone = Zone.current; - } - _zone = zone; +void storeZone([Zone? zone]) { + _zone = zone ?? Zone.current; } /// Calls [expect] in package:test/test.dart in the zone stored in [storeZone]. /// /// Useful for expectations in blocks called in other zones. -void zonedExpect(actual, matcher, {String reason}) { +void zonedExpect(actual, matcher, {String? reason}) { validateZone(); - return _zone.run(() { + return _zone!.run(() { expect(actual, matcher, reason: reason); }); } @@ -73,6 +70,6 @@ void zonedExpect(actual, matcher, {String reason}) { /// renderIntoDocument(( /// TestComponent()..onComponentDidMount = shouldPassTest)()); // Passes /// }); -void setComponentZone([Zone zone]) { +void setComponentZone([Zone? zone]) { componentZone = zone ?? Zone.current; } diff --git a/pubspec.yaml b/pubspec.yaml index ed04a7c0..8984a3e2 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -3,18 +3,25 @@ version: 2.11.7 description: A library for testing OverReact components homepage: https://github.com/Workiva/over_react_test/ environment: - sdk: ">=2.11.0 <3.0.0" + sdk: '>=2.13.0 <3.0.0' dependencies: + collection: ^1.15.0 js: ^0.6.1+1 - matcher: ^0.12.1+4 meta: ^1.8.0 over_react: ^4.1.2 - react: '>=6.0.1 <8.0.0' + react: ^7.0.0 test: ^1.21.1 dev_dependencies: + dart_dev: ^4.1.0 build_runner: ^2.1.2 build_test: ^2.1.3 build_web_compilers: ^3.0.0 dependency_validator: ^3.2.2 pedantic: ^1.8.0 + +dependency_overrides: + over_react: + git: + url: https://github.com/Workiva/over_react + ref: v5_wip diff --git a/test/over_react_test/common_component_util_test.dart b/test/over_react_test/common_component_util_test.dart index b0d77f4b..ff149301 100644 --- a/test/over_react_test/common_component_util_test.dart +++ b/test/over_react_test/common_component_util_test.dart @@ -49,7 +49,7 @@ main() { group('should skip checking for certain props', () { final meta = getPropsMeta(new_boilerplate.TestCommonForwarding()()); - final consumedKeys = meta.forMixin(new_boilerplate.ShouldNotBeForwardedProps).keys; + final consumedKeys = meta!.forMixin(new_boilerplate.ShouldNotBeForwardedProps).keys; final skippedKey = consumedKeys.first; commonComponentTests( @@ -93,7 +93,7 @@ main() { group('does not call `factory` directly within the consuming group', () { void sharedTest( BuilderOnlyUiFactory factory, { - List Function(PropsMetaCollection) getUnconsumedPropKeys, + List Function(PropsMetaCollection)? getUnconsumedPropKeys, }) { var wasFactoryCalled = false; @@ -161,7 +161,7 @@ main() { void expectedFailGroup(String description, void Function() groupBody, {@required dynamic testFailureMatcher}) { group(description, () { int totalTestCount = 0; - List testFailureErrors; + late List testFailureErrors; setUpAll(() => testFailureErrors = []); void testButReAndIgnoreExceptions(name, testBody) { @@ -249,10 +249,10 @@ typedef HelperRenderFunction = dynamic Function(CommonHelperComponent component) const UiFactory arbitraryUiFactory = domProps; UiFactory registerHelperComponent({ - @required HelperRenderFunction render, - Map defaultProps, - Iterable consumedProps, - PropsMetaCollection propsMeta, + required HelperRenderFunction render, + Map? defaultProps, + Iterable? consumedProps, + PropsMetaCollection? propsMeta, }) { final factory = registerComponent2(() { return CommonHelperComponent( @@ -263,7 +263,7 @@ UiFactory registerHelperComponent({ ); }); - return ([Map backingMap]) => arbitraryUiFactory(backingMap)..componentFactory = factory; + return ([Map? backingMap]) => arbitraryUiFactory(backingMap)..componentFactory = factory; } class CommonHelperComponent extends UiComponent2 { @@ -273,10 +273,10 @@ class CommonHelperComponent extends UiComponent2 { final HelperRenderFunction renderValue; CommonHelperComponent({ - @required Map defaultPropsValue, - @required Iterable consumedPropsValue, - @required PropsMetaCollection propsMetaValue, - @required this.renderValue, + required Map? defaultPropsValue, + required Iterable? consumedPropsValue, + required PropsMetaCollection? propsMetaValue, + required this.renderValue, }) : defaultPropsValue = defaultPropsValue ?? {}, propsMetaValue = propsMetaValue ?? const PropsMetaCollection({}), @@ -305,7 +305,7 @@ class CommonHelperComponent extends UiComponent2 { @override typedPropsFactoryJs(map) => typedPropsFactory(map); - UiProps _cachedTypedProps; + late UiProps _cachedTypedProps; @override UiProps get props => _cachedTypedProps; diff --git a/test/over_react_test/console_log_utils_test.dart b/test/over_react_test/console_log_utils_test.dart index c65812d4..3a13c40f 100644 --- a/test/over_react_test/console_log_utils_test.dart +++ b/test/over_react_test/console_log_utils_test.dart @@ -75,7 +75,7 @@ main() { configuration: errorConfig); expect(logs, hasLength(2)); - expect(logs.firstWhere((log) => log.contains('shouldAlwaysBeFalse')), + expect(logs.firstWhere((log) => log?.contains('shouldAlwaysBeFalse') ?? false), contains('set to true')); }); @@ -229,11 +229,12 @@ main() { queryByTestId(jacket.getInstance(), 'ort_sample_component_button'); await Future.delayed(Duration(milliseconds: 5)); - triggerDocumentClick(button); + expect(button, isNotNull); + triggerDocumentClick(button!); }, configuration: warnConfig); expect(logs, hasLength(1)); - expect(logs.first.contains('I have been clicked'), isTrue); + expect(logs.first?.contains('I have been clicked'), isTrue); }); test('handles errors caused when rendering', () async { @@ -301,7 +302,8 @@ main() { queryByTestId(jacket.getInstance(), 'ort_sample_component_button'); await Future.delayed(Duration(milliseconds: 5)); - triggerDocumentClick(button); + expect(button, isNotNull); + triggerDocumentClick(button!); }, configuration: logConfig); expect(logs, hasLength(3)); @@ -315,7 +317,8 @@ main() { queryByTestId(jacket.getInstance(), 'ort_sample_component_button'); await Future.delayed(Duration(milliseconds: 5)); - triggerDocumentClick(button); + expect(button, isNotNull); + triggerDocumentClick(button!); }, configuration: warnConfig); expect(logs, hasLength(5)); diff --git a/test/over_react_test/custom_matchers_test.dart b/test/over_react_test/custom_matchers_test.dart index be71e314..79cba24c 100644 --- a/test/over_react_test/custom_matchers_test.dart +++ b/test/over_react_test/custom_matchers_test.dart @@ -26,7 +26,7 @@ import './helper_components/sample_component2.dart'; /// Main entry point for CustomMatchers testing main() { group('CustomMatcher', () { - Element testElement; + late Element testElement; setUp(() { testElement = Element.div(); @@ -383,14 +383,14 @@ main() { List allAttachedNodes = []; Element makeAttachedNode() { var node = DivElement()..tabIndex = 1; - document.body.append(node); + document.body!.append(node); allAttachedNodes.add(node); return node; } - Element attachedNode; + late Element attachedNode; setUp(() { attachedNode = makeAttachedNode(); @@ -445,7 +445,7 @@ main() { group('LoggingFunctionMatcher', () { group('when passed a List of logs', () { - List logs; + late List logs; setUp(() { logs = [ @@ -607,7 +607,7 @@ main() { group('PropTypeLogMatcher', () { group('when passed a List of logs', () { - List logs; + late List logs; setUp(() { logs = [ @@ -865,7 +865,8 @@ void shouldFail(value, Matcher matcher, expected) { if (expected is String) { expect(_errorString, equalsIgnoringWhitespace(expected)); } else { - expect(_errorString.replaceAll(RegExp(r'[\s\n]+'), ' '), expected); + expect(_errorString, isNotNull); + expect(_errorString!.replaceAll(RegExp(r'[\s\n]+'), ' '), expected); } } diff --git a/test/over_react_test/dom_util_test.dart b/test/over_react_test/dom_util_test.dart index 2aa0d03a..c313e871 100644 --- a/test/over_react_test/dom_util_test.dart +++ b/test/over_react_test/dom_util_test.dart @@ -23,11 +23,11 @@ main() { test('triggerTransitionEnd correctly dispatches a transitionend event', () async { var flag = false; - document.body.onTransitionEnd.listen((Event event) { + document.body!.onTransitionEnd.listen((Event event) { flag = true; }); - await triggerTransitionEnd(document.body); + await triggerTransitionEnd(document.body!); expect(flag, isTrue); }); @@ -42,7 +42,7 @@ main() { test('when the target is attached to the document', () { var renderedInstance = renderAttachedToDocument((Dom.div()..onClick = ((_) => flag = true))()); - triggerDocumentClick(findDomNode(renderedInstance)); + triggerDocumentClick(findDomNode(renderedInstance)!); expect(flag, isTrue); }); @@ -50,7 +50,7 @@ main() { test('and throws when the target is not attached to the document', () { var renderedInstance = render((Dom.div()..onClick = ((_) => flag = true))()); - expect(() => triggerDocumentClick(findDomNode(renderedInstance)), throwsArgumentError); + expect(() => triggerDocumentClick(findDomNode(renderedInstance)!), throwsArgumentError); expect(flag, isFalse); }); }); @@ -65,7 +65,7 @@ main() { test('when the target is attached to the document', () { var renderedInstance = renderAttachedToDocument((Dom.div()..onMouseDown = ((_) => flag = true))()); - triggerDocumentMouseEvent(findDomNode(renderedInstance), 'mousedown'); + triggerDocumentMouseEvent(findDomNode(renderedInstance)!, 'mousedown'); expect(flag, isTrue); }); @@ -73,7 +73,7 @@ main() { test('and throws when the target is not attached to the document', () { var renderedInstance = render((Dom.div()..onMouseDown = ((_) => flag = true))()); - expect(() => triggerDocumentMouseEvent(findDomNode(renderedInstance), 'mousedown'), throwsArgumentError); + expect(() => triggerDocumentMouseEvent(findDomNode(renderedInstance)!, 'mousedown'), throwsArgumentError); expect(flag, isFalse); }); }); @@ -91,7 +91,7 @@ main() { ..tabIndex = '1' )()); - await triggerFocus(findDomNode(renderedInstance)); + await triggerFocus(findDomNode(renderedInstance)!); expect(flag, isTrue); }); @@ -99,7 +99,7 @@ main() { test('and throws when the target is not attached to the document', () { var renderedInstance = render((Dom.div()..onFocus = ((_) => flag = true))()); - expect(() => triggerFocus(findDomNode(renderedInstance)), throwsArgumentError); + expect(() => triggerFocus(findDomNode(renderedInstance)!), throwsArgumentError); expect(flag, isFalse); }); }); diff --git a/test/over_react_test/helper_components/sample_component.dart b/test/over_react_test/helper_components/sample_component.dart index e6f560fc..22988cee 100644 --- a/test/over_react_test/helper_components/sample_component.dart +++ b/test/over_react_test/helper_components/sample_component.dart @@ -7,32 +7,31 @@ part 'sample_component.over_react.g.dart'; UiFactory Sample = castUiFactory(_$Sample); // ignore: undefined_identifier mixin SampleProps on UiProps { - bool shouldNeverBeNull; + bool? shouldNeverBeNull; - bool shouldAlwaysBeFalse; + bool? shouldAlwaysBeFalse; - bool shouldErrorInRender; + bool? shouldErrorInRender; - bool shouldErrorInMount; + bool? shouldErrorInMount; - bool shouldErrorInUnmount; + bool? shouldErrorInUnmount; - bool addExtraLogAndWarn; + bool? addExtraLogAndWarn; - Function() onComponentDidMount; + Function()? onComponentDidMount; - bool shouldLog; + bool? shouldLog; } class SampleComponent extends UiComponent2 { - @override - Map get defaultProps => (newProps() - ..shouldAlwaysBeFalse = false - ..shouldErrorInRender = false - ..shouldErrorInMount = false - ..shouldErrorInUnmount = false - ..addExtraLogAndWarn = false - ..shouldLog = true); + // Prop defaults + bool get shouldAlwaysBeFalse => props.shouldAlwaysBeFalse ?? false; + bool get shouldErrorInRender => props.shouldErrorInRender ?? false; + bool get shouldErrorInMount => props.shouldErrorInMount ?? false; + bool get shouldErrorInUnmount => props.shouldErrorInUnmount ?? false; + bool get addExtraLogAndWarn => props.addExtraLogAndWarn ?? false; + bool get shouldLog => props.shouldLog ?? true; @override get propTypes => { @@ -41,7 +40,7 @@ class SampleComponent extends UiComponent2 { return PropError.required(info.propName, 'shouldNeverBeNull is necessary'); } - if (props.shouldLog == false && props.shouldAlwaysBeFalse == false) { + if (props.shouldLog == false && (props.shouldAlwaysBeFalse ?? false) == false) { return PropError.combination('shouldLog', 'shouldAlwaysBeFalse', 'logging is required'); } @@ -52,7 +51,7 @@ class SampleComponent extends UiComponent2 { return null; }, keyForProp((p) => p.shouldAlwaysBeFalse): (props, info) { - if (props.shouldAlwaysBeFalse) { + if (props.shouldAlwaysBeFalse ?? false) { return PropError.value(props.shouldAlwaysBeFalse, info.propName, 'shouldAlwaysBeFalse should never equal true.'); } @@ -63,22 +62,22 @@ class SampleComponent extends UiComponent2 { @override componentDidMount() { window.console.warn('Just a lil warning'); - if (props.shouldErrorInMount) throw Error(); + if (shouldErrorInMount) throw Error(); props.onComponentDidMount?.call(); } @override render() { window.console.warn('A second warning'); - if (props.shouldErrorInRender) { + if (shouldErrorInRender) { throw Error(); } else { - if (props.addExtraLogAndWarn) { + if (addExtraLogAndWarn) { window.console.log('Extra Log'); window.console.warn('Extra Warn'); } - if (props.shouldLog) window.console.log('Logging a standard log'); + if (shouldLog) window.console.log('Logging a standard log'); window.console.warn('And a third'); return Dom.div()( (Dom.button() @@ -97,6 +96,6 @@ class SampleComponent extends UiComponent2 { void componentWillUnmount() { super.componentWillUnmount(); - if (props.shouldErrorInUnmount) throw Error(); + if (shouldErrorInUnmount) throw Error(); } } diff --git a/test/over_react_test/helper_components/sample_component2.dart b/test/over_react_test/helper_components/sample_component2.dart index 2f6a313e..3c56d124 100644 --- a/test/over_react_test/helper_components/sample_component2.dart +++ b/test/over_react_test/helper_components/sample_component2.dart @@ -7,7 +7,7 @@ part 'sample_component2.over_react.g.dart'; UiFactory Sample2 = castUiFactory(_$Sample2); // ignore: undefined_identifier mixin Sample2Props on UiProps { - bool shouldNeverBeNull; + bool? shouldNeverBeNull; } class SampleComponent2 extends UiComponent2 { diff --git a/test/over_react_test/jacket_test.dart b/test/over_react_test/jacket_test.dart index cc212257..ea603739 100644 --- a/test/over_react_test/jacket_test.dart +++ b/test/over_react_test/jacket_test.dart @@ -25,151 +25,120 @@ part 'jacket_test.over_react.g.dart'; /// Main entry point for TestJacket testing main() { group('mount: renders the given instance', () { + // Helper method to clean up each test. + void umountJacket(TestJacket jacket) { + jacket.unmount(); + expect(document.body!.children, isEmpty); + expect(jacket.isMounted, isFalse); + } + group('attached to the document', () { group('and unmounts after the test is done', () { - TestJacket jacket; - setUp(() { - expect(document.body.children, isEmpty); - }); - - tearDown(() { - expect(document.body.children, isEmpty); - expect(jacket.isMounted, isFalse); - - jacket = null; + expect(document.body!.children, isEmpty); }); test('with the given container', () { var mountNode = DivElement(); - jacket = mount(Sample()(), attachedToDocument: true, mountNode: mountNode); + final jacket = mount(Sample()(), attachedToDocument: true, mountNode: mountNode); - expect(document.body.children[0], mountNode); + expect(document.body!.children[0], mountNode); expect(jacket.isMounted, isTrue); expect(mountNode.children[0], jacket.getNode()); + umountJacket(jacket); }); test('without the given container', () { - jacket = mount(Sample()(), attachedToDocument: true); + final jacket = mount(Sample()(), attachedToDocument: true); expect(jacket.isMounted, isTrue); - expect(document.body.children[0].children[0], jacket.getNode()); + expect(document.body!.children[0].children[0], jacket.getNode()); + umountJacket(jacket); }); }); group('and does not unmount after the test is done', () { - TestJacket jacket; - setUp(() { - expect(document.body.children, isEmpty); - }); - - tearDown(() { - expect(document.body.children, isNotEmpty); - expect(jacket.isMounted, isTrue); - - jacket.unmount(); - - expect(document.body.children, isEmpty); - expect(jacket.isMounted, isFalse); - - jacket = null; + expect(document.body!.children, isEmpty); }); test('with the given container', () { var mountNode = DivElement(); - jacket = mount(Sample()(), + final jacket = mount(Sample()(), attachedToDocument: true, mountNode: mountNode, autoTearDown: false ); - expect(document.body.children[0], mountNode); + expect(document.body!.children[0], mountNode); expect(jacket.isMounted, isTrue); expect(mountNode.children[0], jacket.getNode()); + umountJacket(jacket); }); test('without the given container', () { - jacket = + final jacket = mount(Sample()(), attachedToDocument: true, autoTearDown: false); expect(jacket.isMounted, isTrue); - expect(document.body.children[0].children[0], jacket.getNode()); + expect(document.body!.children[0].children[0], jacket.getNode()); + umountJacket(jacket); }); }); }); group('not attached to the document', () { group('and unmounts after the test is done', () { - TestJacket jacket; - setUp(() { - expect(document.body.children, isEmpty); - }); - - tearDown(() { - expect(document.body.children, isEmpty); - expect(jacket.isMounted, isFalse); - - jacket = null; + expect(document.body!.children, isEmpty); }); test('with the given container', () { var mountNode = DivElement(); - jacket = mount(Sample()(), mountNode: mountNode); + final jacket = mount(Sample()(), mountNode: mountNode); - expect(document.body.children, isEmpty); + expect(document.body!.children, isEmpty); expect(jacket.isMounted, isTrue); expect(mountNode.children[0], jacket.getNode()); + umountJacket(jacket); }); test('without the given container', () { - jacket = mount(Sample()()); + final jacket = mount(Sample()()); expect(jacket.isMounted, isTrue); + umountJacket(jacket); }); }); group('and does not unmount after the test is done', () { - TestJacket jacket; - setUp(() { - expect(document.body.children, isEmpty); - }); - - tearDown(() { - expect(document.body.children, isEmpty); - expect(jacket.isMounted, isTrue); - - jacket.unmount(); - - expect(document.body.children, isEmpty); - expect(jacket.isMounted, isFalse); - - jacket = null; + expect(document.body!.children, isEmpty); }); test('with the given container', () { var mountNode = DivElement(); - jacket = mount(Sample()(), mountNode: mountNode, autoTearDown: false); + final jacket = mount(Sample()(), mountNode: mountNode, autoTearDown: false); - expect(document.body.children.isEmpty, isTrue); + expect(document.body!.children.isEmpty, isTrue); expect(jacket.isMounted, isTrue); expect(mountNode.children[0], jacket.getNode()); + umountJacket(jacket); }); test('without the given container', () { - jacket = mount(Sample()(), autoTearDown: false); + final jacket = mount(Sample()(), autoTearDown: false); - expect(document.body.children, isEmpty); + expect(document.body!.children, isEmpty); expect(jacket.isMounted, isTrue); + umountJacket(jacket); }); }); }); }); group('TestJacket: composite component:', () { - TestJacket jacket; + late TestJacket jacket; setUp(() { var mountNode = DivElement(); @@ -179,7 +148,7 @@ main() { ); expect(Sample(jacket.getProps()).foo, isFalse); - expect(jacket.getDartInstance().state.bar, isFalse); + expect(jacket.getDartInstance()!.state.bar, isFalse); }); test('rerender', () { @@ -201,9 +170,9 @@ main() { }); test('setState', () { - jacket.setState(jacket.getDartInstance().newState()..bar = true); + jacket.setState(jacket.getDartInstance()!.newState()..bar = true); - expect(jacket.getDartInstance().state.bar, isTrue); + expect(jacket.getDartInstance()!.state.bar, isTrue); }); test('unmount', () { @@ -216,8 +185,8 @@ main() { }); group('TestJacket: DOM component:', () { - Element mountNode; - TestJacket jacket; + late Element mountNode; + late TestJacket jacket; setUp(() { mountNode = DivElement(); @@ -231,7 +200,6 @@ main() { tearDown(() { mountNode.remove(); - mountNode = null; }); test('rerender', () { @@ -266,8 +234,8 @@ main() { }); group('TestJacket: function component:', () { - Element mountNode; - TestJacket jacket; + late Element mountNode; + late TestJacket jacket; setUp(() { mountNode = DivElement(); @@ -281,7 +249,6 @@ main() { tearDown(() { mountNode.remove(); - mountNode = null; }); test('rerender', () { @@ -319,11 +286,11 @@ main() { UiFactory Sample = castUiFactory(_$Sample); // ignore: undefined_identifier mixin SampleProps on UiProps { - bool foo; + bool? foo; } mixin SampleState on UiState { - bool bar; + bool? bar; } class SampleComponent extends UiStatefulComponent2 { diff --git a/test/over_react_test/react_util_test.dart b/test/over_react_test/react_util_test.dart index e0159ba7..c70e047b 100644 --- a/test/over_react_test/react_util_test.dart +++ b/test/over_react_test/react_util_test.dart @@ -59,37 +59,37 @@ main() { reason: 'The React instance should have been unmounted.' ); - expect(document.body.children, isEmpty, reason: 'All attached mount points should have been removed.'); + expect(document.body!.children, isEmpty, reason: 'All attached mount points should have been removed.'); }); test('renderAttachedToDocument renders the component into the document', () { - expect(document.body.children, isEmpty); + expect(document.body!.children, isEmpty); renderedInstance = renderAttachedToDocument(Wrapper()); - expect(document.body.children[0].children.contains(findDomNode(renderedInstance)), isTrue, + expect(document.body!.children[0].children.contains(findDomNode(renderedInstance)), isTrue, reason: 'The component should have been rendered into the container div.'); }); test('renderAttachedToDocument renders the component into the document with a given container', () { - expect(document.body.children, isEmpty); + expect(document.body!.children, isEmpty); var container = DivElement(); renderedInstance = renderAttachedToDocument(Wrapper(), container: container); - expect(document.body.children[0].children.contains(findDomNode(renderedInstance)), isTrue, + expect(document.body!.children[0].children.contains(findDomNode(renderedInstance)), isTrue, reason: 'The component should have been rendered into the container div.'); - expect(document.body.children[0], container); + expect(document.body!.children[0], container); }); }); test('renderAttachedToDocument renders the component into the document and tearDownAttachedNodes cleans them up', () { - expect(document.body.children, isEmpty); + expect(document.body!.children, isEmpty); var renderedInstance = renderAttachedToDocument(Wrapper(), autoTearDown: false); - expect(document.body.children[0].children.contains(findDomNode(renderedInstance)), isTrue, + expect(document.body!.children[0].children.contains(findDomNode(renderedInstance)), isTrue, reason: 'The component should have been rendered into the container div.'); tearDownAttachedNodes(); @@ -102,7 +102,7 @@ main() { reason: 'The React instance should have been unmounted.' ); - expect(document.body.children, isEmpty, reason: 'All attached mount points should have been removed.'); + expect(document.body!.children, isEmpty, reason: 'All attached mount points should have been removed.'); }); group('renderAndGetDom', () { @@ -145,7 +145,7 @@ main() { test('simulates a click on a component with additional event data', () { var flag = false; - SyntheticMouseEvent event; + late SyntheticMouseEvent event; var renderedInstance = render((Dom.div() ..onClick = (evt) { flag = true; @@ -253,7 +253,7 @@ main() { }); group('getByTestId returns', () { - sharedTests({bool shallow}) { + sharedTests({required bool shallow}) { testSpecificRender(ReactElement instance) => shallow ? renderShallow(instance) : render(instance); @@ -427,16 +427,6 @@ main() { expect(descendant, isNull); }); - - test('null if the user searches for a test ID of `null` when the test ID is set to \'null\'', () { - var renderedInstance = testSpecificRender(Wrapper()( - (Test()..addTestId('null'))() - )); - - var descendant = getByTestId(renderedInstance, null); - - expect(descendant, isNull); - }); } group('(rendered component)', () { @@ -461,7 +451,7 @@ main() { }); group('getAllByTestId returns', () { - sharedTests({bool shallow}) { + sharedTests({required bool shallow}) { testSpecificRender(ReactElement instance) => shallow ? renderShallow(instance) : render(instance); @@ -640,15 +630,6 @@ main() { expect(descendants, isEmpty); }); - test('an empty list if the user searches for a test ID of `null` when the test ID is set to \'null\'', () { - var renderedInstance = testSpecificRender(Wrapper()( - (Test()..addTestId('null'))(), - )); - - var descendants = getAllByTestId(renderedInstance, null); - expect(descendants, isEmpty); - }); - test('without throwing when text nodes are present in the tree', () { var renderedInstance = render(Wrapper()( Dom.div()(), @@ -750,7 +731,7 @@ main() { group('`data-test-id` html attribute key', () { test('', () { var renderedInstance = render((Nested()..addTestId('value'))()); - var innerNode = findDomNode(renderedInstance).querySelector('[data-test-id~="inner"]'); + var innerNode = findDomNode(renderedInstance)!.querySelector('[data-test-id~="inner"]'); expect(queryByTestId(renderedInstance, 'value'), innerNode); }); @@ -764,7 +745,7 @@ main() { test('custom html attribute key', () { var renderedInstance = render((Nested()..addTestId('value', key: 'data-custom-id'))()); - var innerNode = findDomNode(renderedInstance).querySelector('[data-test-id~="inner"]'); + var innerNode = findDomNode(renderedInstance)!.querySelector('[data-test-id~="inner"]'); expect(queryByTestId(renderedInstance, 'value', key: 'data-custom-id'), innerNode); }); @@ -788,7 +769,7 @@ main() { // Let the shadow dom mount (the test components kinda slow since it does it after adding it to the dom.) await pumpEventQueue(); - var innerNode = shadowHostRef.current.shadowRoot.querySelector('[data-test-id~="$searchId"]'); + var innerNode = shadowHostRef.current!.shadowRoot!.querySelector('[data-test-id~="$searchId"]'); expect(queryByTestId(jacket.mountNode, searchId, searchInShadowDom: true), innerNode); }); @@ -817,7 +798,7 @@ main() { (Nested()..addTestId('value'))(), (Nested()..addTestId('value'))(), )); - var innerNodes = findDomNode(renderedInstance).querySelectorAll('[data-test-id~="inner"]'); + var innerNodes = findDomNode(renderedInstance)!.querySelectorAll('[data-test-id~="inner"]'); expect(queryAllByTestId(renderedInstance, 'value'), innerNodes); }); @@ -834,7 +815,7 @@ main() { (Nested()..addTestId('value'))(), (Nested()..addTestId('value'))(), )); - var innerNodes = findDomNode(renderedInstance).querySelectorAll('[data-test-id~="inner"]'); + var innerNodes = findDomNode(renderedInstance)!.querySelectorAll('[data-test-id~="inner"]'); expect(queryAllByTestId(renderedInstance, 'value'), innerNodes); }); @@ -878,9 +859,9 @@ main() { // Let the shadow dom mount (the test components kinda slow since it does it after adding it to the dom.) await pumpEventQueue(); - var level1 = shadow1Ref.current.shadowRoot.querySelector('.div1'); - var level2 = shadow2Ref.current.shadowRoot.querySelector('.div2'); - var level3 = shadow3Ref.current.shadowRoot.querySelector('.div3'); + var level1 = shadow1Ref.current!.shadowRoot!.querySelector('.div1'); + var level2 = shadow2Ref.current!.shadowRoot!.querySelector('.div2'); + var level3 = shadow3Ref.current!.shadowRoot!.querySelector('.div3'); expect(queryAllByTestId(jacket.mountNode, 'findMe', searchInShadowDom: true), [level1, level2, level3]); }); @@ -919,7 +900,7 @@ main() { // Let the shadow dom mount (the test components kinda slow since it does it after adding it to the dom.) await pumpEventQueue(); - var level1 = shadow1Ref.current.shadowRoot.querySelector('.div1'); + var level1 = shadow1Ref.current!.shadowRoot!.querySelector('.div1'); expect(queryAllByTestId(jacket.mountNode, 'findMe', searchInShadowDom: true, shadowDepth: 1), [level1]); }); @@ -1029,16 +1010,6 @@ main() { expect(descendant, isNull); }); - - test('null if the user searches for `null` when a test ID is set to \'null\'', () { - var renderedInstance = render(Wrapper()( - (Test()..addTestId('null'))() - )); - - var descendant = getComponentByTestId(renderedInstance, null); - - expect(descendant, isNull); - }); }); group('getComponentByTestId returns', () { @@ -1144,16 +1115,6 @@ main() { expect(descendant, isNull); }); - - test('null if the user searches for `null` when a test ID is set to \'null\'', () { - var renderedInstance = render(Wrapper()( - (Test()..addTestId('null'))() - )); - - var descendant = getComponentByTestId(renderedInstance, null); - - expect(descendant, isNull); - }); }); group('getPropsByTestId returns', () { @@ -1265,16 +1226,6 @@ main() { expect(props, isNull); }); - - test('null if the user searches for `null` when a test ID is set to \'null\'', () { - var renderedInstance = render(Wrapper()( - (Test()..addTestId('null'))() - )); - - var props = getPropsByTestId(renderedInstance, null); - - expect(props, isNull); - }); }); group('findDescendantsWithProp', () { diff --git a/test/over_react_test/utils/shadow_nested_component.dart b/test/over_react_test/utils/shadow_nested_component.dart index 01a70efa..1a1f6510 100644 --- a/test/over_react_test/utils/shadow_nested_component.dart +++ b/test/over_react_test/utils/shadow_nested_component.dart @@ -20,8 +20,8 @@ import 'package:over_react/react_dom.dart' as react_dom; part 'shadow_nested_component.over_react.g.dart'; mixin ShadowNestedProps on UiProps { - String shadowRootHostTestId; - String shadowRootFirstChildTestId; + String? shadowRootHostTestId; + String? shadowRootFirstChildTestId; } UiFactory ShadowNested = uiForwardRef( @@ -30,7 +30,7 @@ UiFactory ShadowNested = uiForwardRef( useEffect(() { var shadowRootFirstChild = DivElement()..dataset['test-id'] = props.shadowRootFirstChildTestId ?? 'shadowRootFirstChild'; - divRef.current.attachShadow({'mode':'open'}).append(shadowRootFirstChild); + divRef.current!.attachShadow({'mode':'open'}).append(shadowRootFirstChild); react_dom.render(Fragment()(props.children), shadowRootFirstChild); return () => react_dom.unmountComponentAtNode(shadowRootFirstChild); }, []); diff --git a/test/over_react_test/utils/test_common_component.dart b/test/over_react_test/utils/test_common_component.dart index 8bbfa82b..a116d3a4 100644 --- a/test/over_react_test/utils/test_common_component.dart +++ b/test/over_react_test/utils/test_common_component.dart @@ -57,7 +57,7 @@ abstract class PropsThatShouldBeForwarded { Map get props; - bool foo; + bool? foo; } @PropsMixin() @@ -69,7 +69,7 @@ abstract class PropsThatShouldNotBeForwarded { Map get props; - bool bar; + bool? bar; } // AF-3369 This will be removed once the transition to Dart 2 is complete. diff --git a/test/over_react_test/utils/test_common_component_new_boilerplate.dart b/test/over_react_test/utils/test_common_component_new_boilerplate.dart index 486c0724..b3a36abd 100644 --- a/test/over_react_test/utils/test_common_component_new_boilerplate.dart +++ b/test/over_react_test/utils/test_common_component_new_boilerplate.dart @@ -38,7 +38,7 @@ class TestCommonForwardingComponent extends UiComponent2 TestCommonDomOnlyForwarding = diff --git a/test/over_react_test/utils/test_common_component_required_props.dart b/test/over_react_test/utils/test_common_component_required_props.dart index 43c0d395..fd69a28a 100644 --- a/test/over_react_test/utils/test_common_component_required_props.dart +++ b/test/over_react_test/utils/test_common_component_required_props.dart @@ -23,13 +23,13 @@ UiFactory TestCommonRequired = @Props() class _$TestCommonRequiredProps extends UiProps { @nullableRequiredProp - bool foobar; + bool? foobar; @requiredProp - bool bar; + bool? bar; @nullableRequiredProp - bool defaultFoo; + bool? defaultFoo; } @Component() diff --git a/test/over_react_test/utils/test_common_component_required_props_commponent2.dart b/test/over_react_test/utils/test_common_component_required_props_commponent2.dart index 5b3f095f..d44a44ef 100644 --- a/test/over_react_test/utils/test_common_component_required_props_commponent2.dart +++ b/test/over_react_test/utils/test_common_component_required_props_commponent2.dart @@ -21,13 +21,13 @@ UiFactory TestCommonRequired2 = mixin TestCommonRequired2Props on UiProps { @nullableRequiredProp - bool foobar; + bool? foobar; @requiredProp - bool bar; + bool? bar; @nullableRequiredProp - bool defaultFoo; + bool? defaultFoo; } class TestCommonRequired2Component extends diff --git a/test/over_react_test/validation_util_test.dart b/test/over_react_test/validation_util_test.dart index 9cc9e587..b94f069a 100644 --- a/test/over_react_test/validation_util_test.dart +++ b/test/over_react_test/validation_util_test.dart @@ -34,7 +34,7 @@ main() { assert(ValidationUtil.warn('message1')); expect(getValidationWarnings(), hasLength(1)); - expect(getValidationWarnings().single, 'message1'); + expect(getValidationWarnings()!.single, 'message1'); assert(ValidationUtil.warn('message2')); @@ -48,7 +48,7 @@ main() { assert(ValidationUtil.warn('message1')); expect(getValidationWarnings(), hasLength(1)); - expect(getValidationWarnings().single, 'message1'); + expect(getValidationWarnings()!.single, 'message1'); stopRecordingValidationWarnings(); @@ -94,7 +94,7 @@ main() { assert(ValidationUtil.warn('message1')); expect(getValidationWarnings(), hasLength(1)); - expect(getValidationWarnings().single, 'message1'); + expect(getValidationWarnings()!.single, 'message1'); clearValidationWarnings(); @@ -103,7 +103,7 @@ main() { assert(ValidationUtil.warn('message2')); expect(getValidationWarnings(), hasLength(1)); - expect(getValidationWarnings().single, 'message2'); + expect(getValidationWarnings()!.single, 'message2'); }); }); }, testOn: '!js');