Guest Dean Ellis Posted September 12, 2024 Posted September 12, 2024 We have introducing a new way to generate asset packs for your .NET & .NET MAUI Android applications in .NET 9 that you can try out today. What are Asset Packs? Why should you use them? How to get started? Let’s get into it! [HEADING=1]What is an Asset Pack?[/HEADING] Back in 2018 Google introduced a new package format for deploying Android applications to the Google Play Store. Support for this new format, Android App Bundles (AAB), has been in .NET Android since .NET 6. There are many feature of Android App Bundles and we prioritized the top features that developers needed. One advanced features that is part of the AAB format is Asset Packs which will now be part of .NET 9. So what are asset packs? Part of the new package format was the ability to place [iCODE]Assets[/iCODE] into a separate package. This would allow developers to upload games and apps which would normally be larger that the basic package size allowed by Google Play. By putting these assets into a separate package you get the ability to upload a package which is up to 2 gigabytes in size. The basic package size is 200 megabytes. There is a condition, asset packs can ONLY contain [iCODE]Assets[/iCODE]. In the case of .NET Android this means items which have the [iCODE]AndroidAsset[/iCODE] build action. Asset Packs can can have different delivery options. This controls when your assets will install on the device. Install Time packs are installed at the same time as the application. This type pack can be up to 1 Gigabyte in size, but you can only have one of them. The Fast Follow type of pack will install at some point shortly AFTER the app has finished installing. The app will be able to start while this type of pack is being installed so developers should check it has finished installing before trying to use the assets. This kind of asset pack can be up to 512 Megabytes in size. The final type is the On Demand type. These will never be downloaded to the device unless the application specifically requests it. As mentioned the total size of ALL your asset packs cannot exceed 2 Gigabytes, and you can have up to 50 separate asset packs. If you have an application with allot of Assets such as a game, you can see how these types of pack would be very useful. Assets like music, movies or textures can take up allot of space inside your application package. Having the ability to split them out gives you more freedom to put code based features in your actual game or app. For more information you can read up on Asset Delivery at Android Developers [HEADING=1]The Problem: Building Asset Packs[/HEADING] With the .NET 8 version of .NET Android building an asset pack was not supported by the .NET build system. There is an MSBuild ItemGroup ‘AndroidAppBundleModules’ which can be used to add additional [iCODE]zip[/iCODE] files to the [iCODE]aab[/iCODE] package. However users would still need to figure out a way to build the asset pack manually using [iCODE]gradle[/iCODE] or [iCODE]aapt2[/iCODE]. Some community based hacks are available but it would be nice to be able to use this feature directly in .NET Android. Most of these hacks involve setting up a separate project and adding custom build targets to your solution in order to build the appropriate zip files. This makes it a bit awkward to work with. [HEADING=1]The Solution: AssetPack Metadata[/HEADING] We thought long and hard on the best way to implement asset pack support. Ideally we wanted something which would allow not only .NET Android developers to easily create asset packs, but also allow .NET Maui developers to make use of it as well. We decided on was to make use of MSBuild Metadata to control the creation of the asset packs. There will be no need to create any extra projects or mess about with custom target. Adding some simple Metadata to [iCODE]AndroidAsset[/iCODE] Items will allow users to define asset packs. Given the following files in an example project MyProject.csprojMainActivity.csAssets/ MyLargeAsset.mp4 MySmallAsset.jsonAndroidManifest.xml The [iCODE]MyLargeAsset.mp4[/iCODE] and [iCODE]MySmallAsset.json[/iCODE] will both end up in the main [iCODE].aab[/iCODE] file when we publish the application. Lets say that [iCODE]MyLargeAsset.mp4[/iCODE] is over the 200 Megabyte limit, so having it in the main app bundle is not an option. So what we want to do is place this asset in an [iCODE]Install Time[/iCODE] asset pack. This way we can make sure it is in place as soon as the app is installed on a device. With the new system we can make use of two new Metadata attributes we are supporting for the [iCODE]AndroidAsset[/iCODE] ItemGroup. The first piece of Metadata is [iCODE]AssetPack[/iCODE]. If present this attribute will control which asset pack the asset will end up in. If not present the asset will end up in the main app bundle by default. The AssetPack name will be in the form of [iCODE]$(AndroidPackage).%(AssetPack)[/iCODE], so if the package name of your project is [iCODE]com.foo.myproject[/iCODE] and the [iCODE]AssetPack[/iCODE] value was [iCODE]bar[/iCODE], the asset pack would be [iCODE]com.foo.myproject.bar[/iCODE]. There is a special value for [iCODE]AssetPack[/iCODE] which is [iCODE]base[/iCODE]. This can be used to make sure certain assets end up in the main app bundle rather than an asset pack. More on this later. The other is [iCODE]DeliveryType[/iCODE]. If present you can use this to control what type of asset pack is created. Valid values for this are [iCODE]InstallTime[/iCODE], [iCODE]FastFollow[/iCODE] and [iCODE]OnDemand[/iCODE]. If not present the default value is [iCODE]InstallTime[/iCODE]. So in our example if we wanted to move the [iCODE]MyLargeAsset.mp4[/iCODE] asset into its own asset pack we can use the following in the [iCODE]MyProject.csproj[/iCODE]. Note we are using [iCODE]Update[/iCODE] rather than include since .NET Android supports auto import of assets. <ItemGroup> <AndroidAsset Update="Assets/MyLargeAsset.mp4" AssetPack="myassets" /></ItemGroup> This will cause the .NET Android build system to create a new asset pack called [iCODE]com.foo.myproject.myassets[/iCODE] and include the [iCODE]MyLargeAsset.mp4[/iCODE] in that pack. This pack will be automatically included in the final [iCODE].aab[/iCODE] file. There is no need to manually create this pack at all. Because the default value of [iCODE]DeliveryType[/iCODE] is [iCODE]InstallTime[/iCODE] by default, there is no additional work needed in this case. [HEADING=1]A More Complete Example[/HEADING] Now there are probably use cases where you need to control which assets go into a pack and which ones do not. But you also have hundreds of assets. In this case you will probably want to use wildcards to update the auto imported items. <ItemGroup> <AndroidAsset Update="Assets/*" AssetPack="myassets" DeliveryType="FastFollow" /></ItemGroup> The above snippet will make ALL the assets picked up from the Assets folder go into the [iCODE]myassets[/iCODE] asset pack which is [iCODE]FastFollow[/iCODE]. However if you have one or two items that you need in the main package, because the asset pack might not be installed when they are needed, we need a way to do that. This is where the previously mentioned [iCODE]base[/iCODE] value for [iCODE]AssetPack[/iCODE] comes in. <ItemGroup> <AndroidAsset Update="Assets/*" AssetPack="myassets" DeliveryType="FastFollow" /> <AndroidAsset Update="Assets/myimportantfile.json" AssetPack="base" /></ItemGroup> In the updated example above, the [iCODE]myimportantfile.json[/iCODE] will now end up in the main app bundle, rather than the [iCODE]myassets[/iCODE] asset pack. This is a good way to control which specific assets end up in certain locations. [HEADING=2]Checking the Status of FastFollow Asset Packs[/HEADING] If you are using [iCODE]FastFollow[/iCODE] based packs, you will need to check it is installed before trying to access its contents. First add a PackageReference to the [iCODE]Xamarin.Google.Android.Play.Asset.Delivery[/iCODE] NuGet package. <ItemGroup><PackageReference Include="Xamarin.Google.Android.Play.Asset.Delivery" Version="2.0.5.0" /></ItemGroup> This will pull in the required API’s to work with Asset Packs. Next we need to make use of the [iCODE]Google.Play.Core.Assets.AssetPackManager[/iCODE] to query the location of the [iCODE]FastFollow[/iCODE] asset pack. We can use the [iCODE]GetPackLocation[/iCODE] method to do this. If it returns [iCODE]null[/iCODE] for the [iCODE]AssetsPath[/iCODE] method on the returned [iCODE]AssetPackLocation[/iCODE], then the pack has not yet been installed. If it returns anything else that is the install location of the pack. var assetPackManager = AssetPackManagerFactory.GetInstance (this);AssetPackLocation assetPackPath = assetPackManager.GetPackLocation("myfastfollowpack");string assetsFolderPath = assetPackPath?.AssetsPath() ?? null;if (assetsFolderPath is null) { // FastFollow Pack is not installed.} [HEADING=2]Downloading OnDemand Asset Packs[/HEADING] If you want to use [iCODE]OnDemand[/iCODE] packs you will need to download these manually. In order to do that you must use the [iCODE]Google.Play.Core.Assets.IAssetPackManager[/iCODE]. In order to monitor the progress of the download you need to make use of the [iCODE]AssetPackStateUpdateListener[/iCODE] type. However because this type is a Java Generic you cannot use it directly. To work around this the .NET Binding provides a [iCODE]AssetPackStateUpdateListenerWrapper[/iCODE] class which can be used to hook up a event to monitor the progress. First we need to declare the fields we need. // This is the underlying IAssetPackManagerIAssetPackManager assetPackManager;// This is the wrapper AssetPackStateUpdateListener which allows us// to provide an event to get updates from the IAssetPackManager.AssetPackStateUpdateListenerWrapper listener; Next up we declare the delegate we want to use to monitor the download. Note the [iCODE]AssetPackStateEventArgs[/iCODE] is a nested class of [iCODE]AssetPackStateUpdateListenerWrapper[/iCODE]. You can find out what [iCODE]AssetPackStatus[/iCODE] values to you can use over at the android documentation. void Listener_StateUpdate(object sender, AssetPackStateUpdateListenerWrapper.AssetPackStateEventArgs e){ var status = e.State.Status(); switch (status) { case AssetPackStatus.Downloading: long downloaded = e.State.BytesDownloaded(); long totalSize = e.State.TotalBytesToDownload (); double percent = 100.0 * downloaded / totalSize; Android.Util.Log.Info ("Listener_StateUpdate", $"Downloading {percent}"); break; case AssetPackStatus.Completed: break; case AssetPackStatus.WaitingForWifi: assetPackManager.ShowCellularDataConfirmation (this); break; }} Next thing to do is to create an instance of the [iCODE]IAssetPackManager[/iCODE] via the [iCODE]AssetPackManagerFactory.GetInstance[/iCODE] method. We then need to create the [iCODE]AssetPackStateUpdateListenerWrapper[/iCODE] instance and hookup the [iCODE]StateUpdate[/iCODE] delegate. assetPackManager = AssetPackManagerFactory.GetInstance (this);// Create our Wrapper and set up the event handler.listener = new AssetPackStateUpdateListenerWrapper();listener.StateUpdate += Listener_StateUpdate; We need to make sure to [iCODE]RegisterListener[/iCODE] and [iCODE]UnregisterListener[/iCODE] the listener with the [iCODE]assetPackManager[/iCODE] when the application resumes or pauses. protected override void OnResume(){ // register our Listener Wrapper with the SplitInstallManager so we get feedback. assetPackManager.RegisterListener(listener.Listener); base.OnResume();}protected override void OnPause(){ assetPackManager.UnregisterListener(listener.Listener); base.OnPause();} Before we download the [iCODE]OnDemand[/iCODE] asset pack we need to check to see if it has been installed first. We do this using the [iCODE]assetPackManager.GetPackLocation[/iCODE] method like we did for [iCODE]FastFollow[/iCODE] packs. In this case if we get a [iCODE]null[/iCODE] for [iCODE]AssetsPath()[/iCODE] the pack is not installed. We can then call [iCODE]assetPackManager.Fetch[/iCODE] to start the download. You can also use the C# extension method [iCODE]AsAsync<T>[/iCODE] which returns a [iCODE]Task<T>[/iCODE] so it can be [iCODE]awaited[/iCODE] on. This extension method is available in the [iCODE]Android.Gms.Extensions[/iCODE] namespace. // Try to install the new feature.var assetPackPath = assetPackManager.GetPackLocation ("myondemandpack");string assetsFolderPath = assetPackPath?.AssetsPath() ?? null;if (assetsFolderPath is null) { await assetPackManager.Fetch(new string[] { "myondemandpack" }).AsAsync<AssetPackStates>();} From this point on we can monitor the status via the [iCODE]AssetPackStateUpdateListenerWrapper[/iCODE] we setup earlier. [HEADING=1]Debugging and Testing[/HEADING] To test your asset packs locally you need to make sure you are using the [iCODE]aab[/iCODE] [iCODE]AndroidPackageFormat[/iCODE]. By default .NET Android will use [iCODE]apk[/iCODE] for debugging. If the [iCODE]AndroidPackageFormat[/iCODE] is left as [iCODE]apk[/iCODE] all the assets will be packaged into the [iCODE]apk[/iCODE] as they normally would be. The following settings in your csproj will turn on the required settings for debugging your asset packs. <PropertyGroup Condition=" '$(Configuration)' == 'Debug' "><AndroidPackageFormat>aab</AndroidPackageFormat><EmbedAssembliesIntoApk>true</EmbedAssembliesIntoApk><AndroidBundleToolExtraArgs>--local-testing</AndroidBundleToolExtraArgs></PropertyGroup> The [iCODE]--local-testing[/iCODE] flag is required for testing on your local device. It tells the [iCODE]bundletool[/iCODE] app that ALL the asset packs should be installed in a cached location on the device. It also sets up the [iCODE]IAssetPackManager[/iCODE] to use a mock downloader which will use the cache. This allows you to test installing [iCODE]OnDemand[/iCODE] and [iCODE]FastFollow[/iCODE] packs in a [iCODE]Debug[/iCODE] environment. [HEADING=1]Can I use this in my Maui Application?[/HEADING] Yes absolutely! When building your Maui application for Android you are using the .NET Android Sdk. So all the features available for .NET Android users can be used by Maui developers. Maui does have its own way to define what an [iCODE]Asset[/iCODE] is, this is via the [iCODE]MauiAsset[/iCODE] build action. Adding the additional [iCODE]AssetPack[/iCODE] and [iCODE]DeliveryType[/iCODE] meta data to the [iCODE]MauiAsset[/iCODE] Items will produce the same results. The additional metadata will be ignored by other platforms. <MauiAsset Include="Resources\Raw\**" LogicalName="%(RecursiveDir)%(Filename)%(Extension)" AssetPack="myassetpack"/> If you have specific items you want to place in an Asset Pack you can use the [iCODE]Update[/iCODE] method to define the [iCODE]AssetPack[/iCODE] metadata. [iCODE]<MauiAsset Update="Resources\Raw\MyLargeAsset.txt" AssetPack="myassetpack" />[/iCODE][HEADING=1]Conclusion[/HEADING] Asset Packs provide a great way to increase the size of your application without compromising on the amount of media or data you ship with it. Moving some or all of your Assets into an Asset Pack will allow you to put more of the base package size towards code and user interface. Be sure to read more about what is new for .NET MAUI in .NET 9. The post Android Asset Packs for .NET & .NET MAUI Android Apps appeared first on .NET Blog. Continue reading... Quote
Recommended Posts
Join the conversation
You can post now and register later. If you have an account, sign in now to post with your account.