Attempt is going well. I use this in 2 of my production apps now.
For now this project builds and can be added to a solution. If you want to use it as is just clone it and add it to your solution. then in your app go to dependencies and add it as a project dependency.
You may need to add to your project.csproj file
- Basic Notifications
- Notifications with Data
- Notifications with user actions like in in the crossgeeks library
- Silent notifications in foreground and background
- Notifications when the app is closed/killed. still have to test this / start on it.
- Basic Notifications
- Notifications with Data
- Notifications with user actions like in in the crossgeeks library
- Silent notifications in foreground and background
Apple displays these weird, to me at least. you have to swipe down the notification to see the actions. just clicking on it doesnt work and gets treated as the notification being opened. You can, like I did, check in your OnNotificationOpened if the notification has a click_action and use a DisplayAlert to give the user a chance to respond. Same goes on android. if you click on the notification instead of on one of the actions it gets treated as OnNotificationOpened.
My app is a default maui app without blazor
<TargetFrameworks>net7.0-android33.0;net7.0-ios</TargetFrameworks>
<UseMaui>true</UseMaui>
<SingleProject>true</SingleProject>
<ImplicitUsings>enable</ImplicitUsings>
<OutputType>Exe</OutputType>
These are my other references
<ItemGroup Condition="'$(TargetFramework)' == 'net7.0-android33.0'">
<PackageReference Include="Xamarin.AndroidX.Activity" Version="1.7.2.1" />
<PackageReference Include="Xamarin.AndroidX.Activity.Ktx" Version="1.7.2.1" />
</ItemGroup>
<ItemGroup>
<PackageReference Include="Camera.MAUI" Version="1.4.4" />
<PackageReference Include="CClarke.Plugin.Calendars" Version="1.1.0" />
<PackageReference Include="CommunityToolkit.Maui" Version="5.2.0" />
<PackageReference Include="Geolocator.Plugin" Version="1.0.2" />
<PackageReference Include="Microsoft.AppCenter" Version="5.0.2">
<TreatAsUsed>true</TreatAsUsed>
</PackageReference>
<PackageReference Include="Microsoft.AppCenter.Analytics" Version="5.0.2" />
<PackageReference Include="Microsoft.AppCenter.Crashes" Version="5.0.2" />
<PackageReference Include="Microsoft.CSharp" Version="4.7.0" />
<PackageReference Include="NdefLibrary" Version="4.1.0" />
<PackageReference Include="Newtonsoft.Json" Version="13.0.3">
<TreatAsUsed>true</TreatAsUsed>
</PackageReference>
<PackageReference Include="Plugin.LocalNotification" Version="10.0.3" />
<PackageReference Include="Plugin.Maui.Audio" Version="1.0.0" />
<PackageReference Include="Plugin.NFC" Version="0.1.26" />
<PackageReference Include="PropertyChanged.Fody" Version="3.4.0" />
<PackageReference Include="sqlite-net-pcl" Version="1.8.116" />
<PackageReference Include="ZXing.Net" Version="0.16.9" />
<PackageReference Include="sqlite-net-pcl" Version="1.8.116" />
<PackageReference Include="SQLiteNetExtensions.Async" Version="2.1.0" />
<PackageReference Include="SQLitePCLRaw.core" Version="2.1.0-pre20220207221914">
<TreatAsUsed>true</TreatAsUsed>
</PackageReference>
<PackageReference Include="SQLitePCLRaw.bundle_green" Version="2.1.0-pre20220207221914">
<TreatAsUsed>true</TreatAsUsed>
</PackageReference>
<PackageReference Include="SQLitePCLRaw.provider.dynamic_cdecl" Version="2.1.0-pre20220207221914" />
<PackageReference Include="SQLitePCLRaw.provider.sqlite3" Version="2.1.0-pre20220207221914">
<TreatAsUsed>true</TreatAsUsed>
</PackageReference>
</ItemGroup>
Some of the setup takes place in your MauiProgram and some takes place in your app.cs. First off your google-services.json You stick this in your root folder and set its build action to GoogleServicesJson
Then your MauiProgram This is the same as how it is in the platforms of crossgeeks on your builder add the following
private static MauiAppBuilder RegisterFirebaseServices(this MauiAppBuilder builder)
{
builder.ConfigureLifecycleEvents(events => {
#if IOS
events.AddiOS(iOS => iOS.WillFinishLaunching((_, options) =>
{
var usercategories = new NotificationUserCategory[]
{
new NotificationUserCategory("StillAlive", new List<NotificationUserAction>
{
new NotificationUserAction("Yes","Yes", NotificationActionType.Foreground),
new NotificationUserAction("No","No", NotificationActionType.Foreground)
})
};
//If you dont want useractions call one of the other initialize options
FirebasePushNotificationManager.Initialize(options, usercategories, true);
return false;
}));
#elif ANDROID
var usercategories = new NotificationUserCategory[]
{
new NotificationUserCategory("StillAlive", new List<NotificationUserAction>
{
new NotificationUserAction("Yes","Yes", NotificationActionType.Foreground),
new NotificationUserAction("No","No", NotificationActionType.Foreground)
})
};
events.AddAndroid(android => android.OnCreate((activity, _) =>
//If you dont want useractions call one of the other initialize options
FirebasePushNotificationManager.Initialize(usercategories, true, false, true)
));
#endif
});
builder.Services.AddSingleton<IPushNotificationHandler, DefaultPushNotificationHandler>();
return builder;
}
This initializes firebase notifications. Now in your app.cs or wherever you want you can add your handler functions. like
BetterFirebasePushNotification.Current.OnNotificationReceived += async (s, p) =>
{
//p = your notification
}
Or maybe you want to listen for token refreshes and send the new token to a backend
BetterFirebasePushNotification.Current.OnTokenRefresh += async (s, p) =>
{
if (p.Token != null && p.Token != "")
{
var request = new RegistrationUpdate()
{
RegToken = p.Token,
SerialKey = Helpers.Settings.License.Replace("/", "").Replace("\\", "").Replace('"', ' ').Trim()
};
var stringContent = new StringContent(JsonConvert.SerializeObject(request), Encoding.UTF8, "application/json");
await RestManager.Instance.ExecuteRest("SetPushtoken", stringContent);
}
}
I ran into an issue where on ios 18 my willPresentNotification triggered twice. So I looked around for solutions and found one to implement in this project. Credit to https://github.com/thomasgalliker/Plugin.FirebasePushNotifications His project makes it a toggelable option if it should check for it. With mine you can only change the ratelimit time. Default is 3 seconds. You can modify this by calling FirebasePushNotificationManager.SetRateLimit and pass it a timespan. I do this in iOS.WillFinishLaunching.
Check if you uploaded your apns key to firebase console. You can find this by going to your Firebase console > Project Settings > Cloud Messaging > Scroll down to Apple app configuration. and select your app. Then Upload your APNs auth key. You get this APNS key from your apple developer env.
Another issue could be you have this in your info.plist
<key>FirebaseAppDelegateProxyEnabled</key>
<false/>
remove this.
Since most scenarios work now im going to convert this to a Nuget package.