From 13c3ae6a6484e724ec1f74f261228e6ac5d34de5 Mon Sep 17 00:00:00 2001 From: Reko Paananen Date: Thu, 12 Dec 2024 14:18:38 +0200 Subject: [PATCH 1/2] enable custom gesture recognizers --- .../google_mobile_ads/lib/src/ad_containers.dart | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/packages/google_mobile_ads/lib/src/ad_containers.dart b/packages/google_mobile_ads/lib/src/ad_containers.dart index 83a038b7f..f55bd2723 100644 --- a/packages/google_mobile_ads/lib/src/ad_containers.dart +++ b/packages/google_mobile_ads/lib/src/ad_containers.dart @@ -651,11 +651,18 @@ class AdWidget extends StatefulWidget { /// Default constructor for [AdWidget]. /// /// [ad] must be loaded before this is added to the widget tree. - const AdWidget({Key? key, required this.ad}) : super(key: key); + const AdWidget({ + Key? key, + required this.ad, + this.gestureRecognizers, + }) : super(key: key); /// Ad to be displayed as a widget. final AdWithView ad; + /// Gesture recognizers to enable custom interactions with ads + final Set>? gestureRecognizers; + @override _AdWidgetState createState() => _AdWidgetState(); } @@ -714,7 +721,8 @@ class _AdWidgetState extends State { (BuildContext context, PlatformViewController controller) { return AndroidViewSurface( controller: controller as AndroidViewController, - gestureRecognizers: const >{}, + gestureRecognizers: widget.gestureRecognizers ?? + const >{}, hitTestBehavior: PlatformViewHitTestBehavior.opaque, ); }, @@ -736,6 +744,7 @@ class _AdWidgetState extends State { viewType: '${instanceManager.channel.name}/ad_widget', creationParams: instanceManager.adIdFor(widget.ad), creationParamsCodec: StandardMessageCodec(), + gestureRecognizers: widget.gestureRecognizers, ); } } From 187323c4d898f09968a2d8739f8bcfdcdaf5bf73 Mon Sep 17 00:00:00 2001 From: Reko Paananen Date: Thu, 12 Dec 2024 15:07:56 +0200 Subject: [PATCH 2/2] comment and test --- .../lib/src/ad_containers.dart | 4 ++ .../test/ad_containers_test.dart | 40 +++++++++++++++++++ 2 files changed, 44 insertions(+) diff --git a/packages/google_mobile_ads/lib/src/ad_containers.dart b/packages/google_mobile_ads/lib/src/ad_containers.dart index f55bd2723..abd1f25f4 100644 --- a/packages/google_mobile_ads/lib/src/ad_containers.dart +++ b/packages/google_mobile_ads/lib/src/ad_containers.dart @@ -645,6 +645,10 @@ abstract class AdWithoutView extends Ad { /// (e.g. [BannerAd] and [NativeAd]) and allows them to be added to the Flutter /// widget tree. /// +/// Optionally, you can provide [gestureRecognizers] to enable custom gesture +/// interactions with the ad content. For example, this can allow users to +/// swipe or perform other gestures based on your app's requirements. +/// /// Must call `load()` first before showing the widget. Otherwise, a /// [PlatformException] will be thrown. class AdWidget extends StatefulWidget { diff --git a/packages/google_mobile_ads/test/ad_containers_test.dart b/packages/google_mobile_ads/test/ad_containers_test.dart index f67ed656f..6b8e240c6 100644 --- a/packages/google_mobile_ads/test/ad_containers_test.dart +++ b/packages/google_mobile_ads/test/ad_containers_test.dart @@ -15,6 +15,7 @@ import 'dart:async'; import 'package:flutter/foundation.dart'; +import 'package:flutter/gestures.dart'; import 'package:flutter/material.dart'; import 'package:google_mobile_ads/src/ad_instance_manager.dart'; import 'package:google_mobile_ads/google_mobile_ads.dart'; @@ -412,6 +413,45 @@ void main() { } }); + testWidgets('passes gestureRecognizers to PlatformView on iOS', + (WidgetTester tester) async { + final gestureRecognizers = >{ + Factory(() => PanGestureRecognizer()), + Factory(() => TapGestureRecognizer()), + }; + + final NativeAd ad = NativeAd( + adUnitId: 'test-ad-unit', + factoryId: '0', + listener: NativeAdListener(), + request: AdRequest(), + ); + + await ad.load(); + + debugDefaultTargetPlatformOverride = TargetPlatform.iOS; + + await tester.pumpWidget( + MaterialApp( + home: Material( + child: AdWidget( + ad: ad, + gestureRecognizers: gestureRecognizers, + ), + ), + ), + ); + + final uiKitViewFinder = find.byType(UiKitView); + expect(uiKitViewFinder, findsOneWidget); + + final uiKitView = tester.widget(uiKitViewFinder); + expect(uiKitView.gestureRecognizers, gestureRecognizers); + + debugDefaultTargetPlatformOverride = null; + await ad.dispose(); + }); + test('load show rewarded', () async { RewardedAd? rewarded; AdRequest request = AdRequest();