Shiny.Music
1.0.0-beta-0005
Prefix Reserved
dotnet add package Shiny.Music --version 1.0.0-beta-0005
NuGet\Install-Package Shiny.Music -Version 1.0.0-beta-0005
<PackageReference Include="Shiny.Music" Version="1.0.0-beta-0005" />
<PackageVersion Include="Shiny.Music" Version="1.0.0-beta-0005" />
<PackageReference Include="Shiny.Music" />
paket add Shiny.Music --version 1.0.0-beta-0005
#r "nuget: Shiny.Music, 1.0.0-beta-0005"
#:package Shiny.Music@1.0.0-beta-0005
#addin nuget:?package=Shiny.Music&version=1.0.0-beta-0005&prerelease
#tool nuget:?package=Shiny.Music&version=1.0.0-beta-0005&prerelease
Shiny.Music
A .NET library for accessing the device music library on Android and iOS. Provides a unified API for:
- 🔐 Requesting permissions to access music
- 🎵 Querying metadata about music on the device
- ▶️ Playing music files from the device library
- 📁 Copying music files (where permitted)
Installation
Add a project reference to Shiny.Music from your .NET MAUI or platform-specific app.
Quick Start
// Register in MauiProgram.cs
builder.Services.AddShinyMusic();
// Use via dependency injection
public class MyPage
{
readonly IMediaLibrary _library;
readonly IMusicPlayer _player;
public MyPage(IMediaLibrary library, IMusicPlayer player)
{
_library = library;
_player = player;
}
async Task Example()
{
// 1. Request permission
var status = await _library.RequestPermissionAsync();
if (status != PermissionStatus.Granted) return;
// 2. Get all tracks
var tracks = await _library.GetAllTracksAsync();
// 3. Play a track
await _player.PlayAsync(tracks[0]);
// 4. Copy a track
var dest = Path.Combine(FileSystem.AppDataDirectory, "copy.m4a");
var success = await _library.CopyTrackAsync(tracks[0], dest);
}
}
Platform Configuration
Android
Required Permissions
Add these to your AndroidManifest.xml:
<uses-permission android:name="android.permission.READ_MEDIA_AUDIO" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"
android:maxSdkVersion="32" />
Notes
- Minimum API Level: 24 (Android 7.0)
- Target API 33+: Uses
READ_MEDIA_AUDIOgranular media permission - Target API < 33: Falls back to
READ_EXTERNAL_STORAGE - The library requests runtime permissions via the MAUI Permissions API
- Music is queried through
MediaStore.Audio.Media - Playback uses
Android.Media.MediaPlayerwith content URIs - Copy: Reads from the
ContentResolverinput stream. Works for all locally stored music files.
iOS
Required Info.plist Entry
<key>NSAppleMusicUsageDescription</key>
<string>This app needs access to your music library to browse and play your music.</string>
⚠️ This is mandatory. Your app will crash on launch if you attempt to access the music library without this key.
Notes
- Minimum iOS Version: 15.0
- Permission is requested via
MPMediaLibrary.RequestAuthorization - Music metadata is queried using
MPMediaQueryfrom theMediaPlayerframework - Playback uses
AVAudioPlayerfromAVFoundationwith the item'sAssetURL - Copy Limitations:
- ✅ Locally synced / purchased (non-DRM) tracks can be exported via
AVAssetExportSession - ❌ Apple Music subscription (DRM-protected) tracks cannot be copied. The
AssetURLis empty for these items, and iOS does not provide filesystem access to DRM content. - The
CopyTrackAsyncmethod returnsfalsefor tracks that cannot be exported. - Exported format is Apple M4A (
.m4a)
- ✅ Locally synced / purchased (non-DRM) tracks can be exported via
Entitlements
No special entitlements are required beyond the Info.plist usage description. The MediaPlayer and AVFoundation frameworks are standard iOS frameworks.
API Reference
IMediaLibrary
| Method | Description |
|---|---|
RequestPermissionAsync() |
Prompts the user for music library access |
CheckPermissionAsync() |
Checks current permission status without prompting |
GetAllTracksAsync() |
Returns all music tracks on the device |
SearchTracksAsync(query) |
Searches tracks by title, artist, or album |
CopyTrackAsync(track, destPath) |
Copies a track to the specified path; returns false if not possible |
IMusicPlayer
| Member | Description |
|---|---|
PlayAsync(track) |
Loads and plays the specified track |
Pause() |
Pauses current playback |
Resume() |
Resumes after pausing |
Stop() |
Stops playback and releases the track |
Seek(position) |
Seeks to a position in the track |
State |
Current PlaybackState (Stopped/Playing/Paused) |
CurrentTrack |
The currently loaded MusicMetadata |
Position / Duration |
Current position and total duration |
StateChanged |
Event fired when playback state changes |
PlaybackCompleted |
Event fired when a track finishes |
MusicMetadata
| Property | Type | Description |
|---|---|---|
Id |
string |
Platform-specific unique identifier |
Title |
string |
Track title |
Artist |
string |
Artist name |
Album |
string |
Album name |
Genre |
string? |
Genre (may be null) |
Duration |
TimeSpan |
Track duration |
AlbumArtUri |
string? |
Album art URI (Android only; null on iOS) |
IsExplicit |
bool? |
Whether the track is marked as explicit content. iOS only via MPMediaItem.IsExplicitItem; always null on Android. |
ContentUri |
string |
URI used for playback and file operations |
Sample App
The sample/MusicSample project is a .NET MAUI app that demonstrates all library features:
- Permission Request — Tap "Request Permission" to prompt for music access
- Browse — Tap "Load All" to list all music on the device
- Search — Type a query and tap "Search" to filter by title/artist/album
- Play/Pause/Stop — Select a track and use the playback controls
- Copy — Select a track and tap "Copy" to export it to app storage
Running the Sample
# Android
dotnet build sample/MusicSample -f net9.0-android -t:Run
# iOS (requires Mac with Xcode)
dotnet build sample/MusicSample -f net9.0-ios -t:Run
Note: Music library access requires a physical device. Simulators/emulators typically have no music content.
License
MIT
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net10.0 is compatible. net10.0-android was computed. net10.0-android36.0 is compatible. net10.0-browser was computed. net10.0-ios was computed. net10.0-ios26.0 is compatible. net10.0-maccatalyst was computed. net10.0-macos was computed. net10.0-tvos was computed. net10.0-windows was computed. |
-
net10.0
- Microsoft.Extensions.DependencyInjection.Abstractions (>= 10.0.0)
- Microsoft.Maui.Essentials (>= 10.0.41)
-
net10.0-android36.0
- Microsoft.Extensions.DependencyInjection.Abstractions (>= 10.0.0)
- Microsoft.Maui.Essentials (>= 10.0.41)
-
net10.0-ios26.0
- Microsoft.Extensions.DependencyInjection.Abstractions (>= 10.0.0)
- Microsoft.Maui.Essentials (>= 10.0.41)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
| Version | Downloads | Last Updated |
|---|---|---|
| 1.0.0-beta-0005 | 38 | 3/2/2026 |
| 1.0.0-beta-0004 | 40 | 3/2/2026 |
| 1.0.0-beta-0003 | 41 | 3/2/2026 |