2022年8月8日 星期一

動手學 .NET MAUI 開發 : 平台特性 Platform Feature 設計

平台特性 Platform Feature 設計

對相依性注入容器宣告要註冊 地理位置 服務

在 .NET MAUI 中,想要取得裝置內的 GPS 座標,可以透過 .NET MAUI 所提供的 地理位置 物件來實現。要使用這個物件,可以透過相依性注入的方式來取得。

  • 在專案根目錄下
  • 找到並且打開 [MauiProgram.cs] 檔案
  • 找到 #region 使用 Microsoft.Extensions.DependencyInjection 套件,註冊相關要用到的服務
  • 在其下方加入底下程式碼
builder.Services.AddSingleton<IGeolocation>(Geolocation.Default);

底下為完成後的 [MauiProgram.cs] 檔案內容

namespace PrismMonkey;

public static class MauiProgram
{
    public static MauiApp CreateMauiApp()
    {
        var builder = MauiApp.CreateBuilder();
        builder
            .UsePrismApp<App>(PrismStartup.Configure)
            .ConfigureFonts(fonts =>
            {
                fonts.AddFont("OpenSans-Regular.ttf", "OpenSansRegular");
                fonts.AddFont("OpenSans-Semibold.ttf", "OpenSansSemibold");
            });

        #region 使用 Microsoft.Extensions.DependencyInjection 套件,註冊相關要用到的服務
        builder.Services.AddSingleton<IConnectivity>(Connectivity.Current);
        builder.Services.AddSingleton<IGeolocation>(Geolocation.Default);
        #endregion

        return builder.Build();
    }
}

找出離你最近的猴子 : 修正 ViewModel

宣告要注入的 地理位置 欄位

  • 在 [ViewModels] 資料夾下
  • 找到並且打開 [MonkeyListPageViewModel.cs] 檔案
  • 找到 #region 透過建構式注入的服務
  • 在其下方加入底下程式碼
private readonly IGeolocation geolocation;

在建構式來注入 地理位置 服務物件

  • 找到 MonkeyListPageViewModel 建構式方法
  • 在其方法上加入 IGeolocation 參數,使用這樣程式碼 IGeolocation geolocation
  • 完成後的該建構式簽章為
public MonkeyListPageViewModel(INavigationService navigationService,
            IPageDialogService dialogService,
            MonkeyService monkeyService, IConnectivity connectivity,
            IGeolocation geolocation) { ... }
  • 找到 this.connectivity = connectivity; 敘述
  • 在其下方加入底下程式碼
this.geolocation = geolocation;

建立一個命令物件,用於執行找出最近猴子的程式碼

  • 找到 #region 在此設計要進行命令物件綁定的屬性
  • 在其下方加入底下程式碼
public DelegateCommand GetClosestMonkeyCommand { get; set; }

設計找出最近猴子的商業邏輯方法

  • 找到 #region 在此設計該 ViewModel 的其他商業邏輯程式碼
  • 在其下方加入底下程式碼
// 發現離你最近的猴子
private async Task FindClosestMokey()
{
    if (IsBusy || Monkeys.Count == 0)
        return;

    try
    {
        // 裝置可能已快取裝置的最新位置,取得最後一個已知位置
        var location = await geolocation.GetLastKnownLocationAsync();
        if (location == null)
        {
            // 查詢裝置的目前位置
            location = await geolocation.GetLocationAsync(new GeolocationRequest
            {
                DesiredAccuracy = GeolocationAccuracy.Medium,
                Timeout = TimeSpan.FromSeconds(30)
            });
        }

        // 發現離你最近的猴子
        // 方法 Location.CalculateDistance 會計算兩個地理位置之間的距離
        var first = Monkeys.OrderBy(m => location.CalculateDistance(
            new Location(m.Latitude, m.Longitude), DistanceUnits.Miles))
            .FirstOrDefault();

        await dialogService.DisplayAlertAsync("", first.Name + " " +
            first.Location, "OK");

    }
    catch (Exception ex)
    {
        Debug.WriteLine($"Unable to query location: {ex.Message}");
        await dialogService.DisplayAlertAsync("Error!", ex.Message, "OK");
    }
}

在建構式內,為 GetClosestMonkeyCommand 物件進行初始化

  • 找到 #region 在此將命令屬性進行初始化,建立命令物件與指派委派方法
  • 在其下方加入底下程式碼
#region 取得當前 GPS 位置,並且找出最接近猴子的命令
GetClosestMonkeyCommand = new DelegateCommand(async () =>
{
    await FindClosestMokey();
});
#endregion

這裡將會宣告一個可以進行資料綁定的命令物件,該物件準備設計用來在 View 中的某個按鈕來進行綁定,一旦,使用者點選這個按鈕之後,將會開始執行這個按鈕所指定的委派方法。

底下為完成後的 [MonkeyListPageViewModel.cs] 檔案內容

namespace PrismMonkey.ViewModels
{
    using System.Collections.ObjectModel;
    using System.ComponentModel;
    using System.Diagnostics;
    using Prism.Navigation;
    using Prism.Services.Dialogs;
    using PrismMonkey.Helpers;
    using PrismMonkey.Models;
    using PrismMonkey.Services;

    public class MonkeyListPageViewModel : INotifyPropertyChanged, INavigationAware
    {
        // 這裡是實作 INotifyPropertyChanged 介面需要用到的事件成員
        // 這是要用於屬性變更的時候,將會觸發這個事件通知
        public event PropertyChangedEventHandler PropertyChanged;

        #region 透過建構式注入的服務
        // 這是透過建構式注入的頁面導航的實作執行個體
        private readonly INavigationService navigationService;
        private readonly IPageDialogService dialogService;
        private readonly MonkeyService monkeyService;
        private readonly IConnectivity connectivity;
        private readonly IGeolocation geolocation;
        #endregion

        #region 在此設計要進行資料綁定的屬性
        /// <summary>
        /// 要瀏覽的所有猴子集合紀錄
        /// </summary>
        public ObservableCollection<Monkey> Monkeys { get; set; } = new();
        /// <summary>
        /// 是否有觸發 下拉更新 手勢條件
        /// </summary>
        public bool IsRefreshing { get; set; }
        /// <summary>
        /// 是否正在忙碌要從網路上下載猴子清單的 JSON 內容
        /// </summary>
        public bool IsBusy { get; set; }
        /// <summary>
        /// 是否沒有正在忙碌要從網路上下載猴子清單的 JSON 內容
        /// </summary>
        public bool IsNotBusy => !IsBusy;
        #endregion

        #region 在此設計要進行命令物件綁定的屬性
        public DelegateCommand GetClosestMonkeyCommand { get; set; }
        public DelegateCommand<Monkey> GoToDetailsCommand { get; set; }
        public DelegateCommand GetMonkeysCommand { get; set; }
        #endregion

        public MonkeyListPageViewModel(INavigationService navigationService,
            IPageDialogService dialogService,
            MonkeyService monkeyService, IConnectivity connectivity,
            IGeolocation geolocation)
        {
            #region 將透過建構式注入進來的物件,指派給這個類別內的欄位或者屬性
            this.navigationService = navigationService;
            this.dialogService = dialogService;
            this.monkeyService = monkeyService;
            this.connectivity = connectivity;
            this.geolocation = geolocation;
            #endregion

            #region 在此將命令屬性進行初始化,建立命令物件與指派委派方法

            #region 取得當前 GPS 位置,並且找出最接近猴子的命令
            GetClosestMonkeyCommand = new DelegateCommand(async () =>
            {
                await FindClosestMokey();
            });
            #endregion

            #region 點選某個猴子之後,要進行頁面切換的命令
            GoToDetailsCommand = new DelegateCommand<Monkey>(async monkey =>
            {
                // 若沒有取得猴子資訊,則不會有任何動作
                if (monkey == null)
                    return;

                NavigationParameters parameters = new();
                parameters.Add(ConstantHelper.NavigationKeyMonkey, monkey);

                #region 舊的頁面導航用法
                //await navigationService.NavigateAsync(ConstantHelper.MonkeyDetailPage, parameters);
                #endregion

                #region 採用 Navigation Builder 的用法
                // 參考文章 : https://github.com/PrismLibrary/Prism/issues/2283
                await navigationService.CreateBuilder()
                .WithParameters(parameters)
                .AddNavigationSegment(ConstantHelper.MonkeyDetailPage)
                .NavigateAsync();
                #endregion
            });
            #endregion

            #region 取得網路上最新猴子清單資訊的命令
            GetMonkeysCommand = new DelegateCommand(async () =>
            {
                await ReloadMonkey();
            });
            #endregion

            #endregion
        }

        #region 在此設計該 ViewModel 的其他商業邏輯程式碼

        // 發現離你最近的猴子
        private async Task FindClosestMokey()
        {
            if (IsBusy || Monkeys.Count == 0)
                return;

            try
            {
                // 裝置可能已快取裝置的最新位置,取得最後一個已知位置
                var location = await geolocation.GetLastKnownLocationAsync();
                if (location == null)
                {
                    // 查詢裝置的目前位置
                    location = await geolocation.GetLocationAsync(new GeolocationRequest
                    {
                        DesiredAccuracy = GeolocationAccuracy.Medium,
                        Timeout = TimeSpan.FromSeconds(30)
                    });
                }

                // 發現離你最近的猴子
                // 方法 Location.CalculateDistance 會計算兩個地理位置之間的距離
                var first = Monkeys.OrderBy(m => location.CalculateDistance(
                    new Location(m.Latitude, m.Longitude), DistanceUnits.Miles))
                    .FirstOrDefault();

                await dialogService.DisplayAlertAsync("", first.Name + " " +
                    first.Location, "OK");

            }
            catch (Exception ex)
            {
                Debug.WriteLine($"Unable to query location: {ex.Message}");
                await dialogService.DisplayAlertAsync("Error!", ex.Message, "OK");
            }
        }

        // 使用注入的 MonkeyService 物件,來取得遠端所有猴子 JSON 紀錄
        private async Task ReloadMonkey()
        {
            // 若已經觸發這個命令委派方法,則無需繼續往下執行
            if (IsBusy)
                return;

            // 保持良好習慣,對於使用 await 呼叫非同步方法,要捕捉例外異常,避免程式崩潰
            try
            {
                // 透過 .NET MAUI Essentials 的 Connectivity 類別可讓您監視裝置網路狀況的變更、
                // 檢查目前的網路存取,以及目前連線方式。
                // 若現在無法連上 Internet ,則顯示與通知使用者,操作錯誤訊息
                if (connectivity.NetworkAccess != NetworkAccess.Internet)
                {
                    // 使用 Prism.Maui 提供的對話窗警訊物件,顯示此錯誤訊息
                    await dialogService.DisplayAlertAsync("No connectivity!",
                         $"Please check internet and try again.", "OK");
                    return;
                }

                IsBusy = true;

                // 透過之前設計的猴子讀取遠端服務端點服務類別
                // 取得網路上的所有猴子 JSON 內容
                var monkeys = await monkeyService.GetMonkeysAsync();

                if (Monkeys.Count != 0)
                    Monkeys.Clear();

                foreach (var monkey in monkeys)
                    Monkeys.Add(monkey);

            }
            catch (Exception ex)
            {
                Debug.WriteLine($"Unable to get monkeys: {ex.Message}");
                await dialogService.DisplayAlertAsync("Error!", ex.Message, "OK");
            }
            finally
            {
                IsBusy = false;
                IsRefreshing = false;
            }
        }
        #endregion

        #region 頁面導航將會觸發的方法
        // 因為實作 INavigationAware 介面,需要建立這個方法
        // 該方法將會用於當要離開此頁面的時候,會被觸發執行
        public void OnNavigatedFrom(INavigationParameters parameters)
        {
        }

        // 因為實作 INavigationAware 介面,需要建立這個方法
        // 該方法將會用於當要導航到此頁面的時候,會被觸發執行
        public void OnNavigatedTo(INavigationParameters parameters)
        {
        }
        #endregion

    }
}

找出離你最近的猴子 : 修正 View

  • 在 [Views] 資料夾下
  • 找到並且打開 [MonkeyListPage.xaml] 檔案
  • 找到 Text="Get Monkeys" />
  • 在其下方加入底下 XAML 標記
<Button
    Grid.Row="1" Grid.Column="1"
    Margin="8"
    Command="{Binding GetClosestMonkeyCommand}"
    IsEnabled="{Binding IsNotBusy}"
    Text="Find Closest" />

底下為完成後的 [MonkeyListPage.xaml] 檔案內容

<?xml version="1.0" encoding="utf-8" ?>
<ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
             xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
             xmlns:viewmodel="clr-namespace:PrismMonkey.ViewModels"
             xmlns:model="clr-namespace:PrismMonkey.Models"
             x:DataType="viewmodel:MonkeyListPageViewModel"
             x:Class="PrismMonkey.Views.MonkeyListPage"
             Title="所有猴子清單">

  <Grid
    ColumnDefinitions="*,*"
    ColumnSpacing="5"
    RowDefinitions="*,Auto"
    RowSpacing="0"
    >
    <CollectionView
      Grid.ColumnSpan="2"
      ItemsSource="{Binding Monkeys}"
      BackgroundColor="Transparent"
      SelectionMode="None">

      <CollectionView.ItemTemplate>
        <DataTemplate x:DataType="model:Monkey">
          <Grid Padding="10">
            <Frame HeightRequest="125" >
              <Frame.GestureRecognizers>
                <TapGestureRecognizer
                    Command="{Binding Source={RelativeSource AncestorType={x:Type viewmodel:MonkeyListPageViewModel}},
                    Path=GoToDetailsCommand}" CommandParameter="{Binding .}"/>
              </Frame.GestureRecognizers>
              <Grid Padding="0" ColumnDefinitions="125,*">
                <Image
                    Aspect="AspectFill"
                    HeightRequest="125" WidthRequest="125"
                    Source="{Binding Image}"
                    />
                <VerticalStackLayout
                    Grid.Column="1"
                    Padding="10">
                  <Label Text="{Binding Name}" />
                  <Label Text="{Binding Location}" />
                </VerticalStackLayout>
              </Grid>
            </Frame>
          </Grid>
        </DataTemplate>
      </CollectionView.ItemTemplate>
    </CollectionView>

    <Button
      Grid.Row="1" Grid.Column="0"
      Margin="8"
      Command="{Binding GetMonkeysCommand}"
      IsEnabled="{Binding IsNotBusy}"
      Text="Get Monkeys" />

    <Button
      Grid.Row="1" Grid.Column="1"
      Margin="8"
      Command="{Binding GetClosestMonkeyCommand}"
      IsEnabled="{Binding IsNotBusy}"
      Text="Find Closest" />

    <ActivityIndicator
      Grid.RowSpan="2" Grid.ColumnSpan="2"
      HorizontalOptions="FillAndExpand"
      VerticalOptions="CenterAndExpand"
      IsRunning="{Binding IsBusy}" IsVisible="{Binding IsBusy}"
      />
  </Grid>

</ContentPage>

修正 Android 需要使用定位權限

  • 滑鼠右擊資料夾 [Platforms] > [Android]
  • 從彈出功能表內點選 [加入] > [類別]
  • 建立名稱為 [AssemblyInfo.cs] 檔案
  • 使用底下 C# 程式碼,替換掉這個檔案內容
using Android.App;

#region 在這裡宣告 Android 系統中需要用到的權限 Permission
[assembly: UsesPermission(Android.Manifest.Permission.AccessCoarseLocation)]
[assembly: UsesPermission(Android.Manifest.Permission.AccessFineLocation)]
[assembly: UsesFeature("android.hardware.location", Required = false)]
[assembly: UsesFeature("android.hardware.location.gps", Required = false)]
[assembly: UsesFeature("android.hardware.location.network", Required = false)]
[assembly: UsesPermission(Android.Manifest.Permission.Internet)]
#endregion

在 Android 平台執行專案

  • 點選中間上方工具列的 [Windows Machine] 這個工具列按鈕旁的下拉選單三角形

  • 從彈出功能表中,找到 [Android Emulators] 內的任何一個模擬器

  • 接者,開始執行這個專案,讓他可以在 Android 模擬器出現

  • 當出現 [所有猴子清單] 這個頁面後

  • 點選下方的 [Get Monkeys] 按鈕

  • 稍微等候一段時間,將會看到所有猴子清單物件出現在畫面上

  • 點選下方的 [Get Monkeys] 按鈕

  • 此時,畫面上將會出現 [Allow PrismMonkey to access this device's location?] 的對話窗

  • 在此,點選 [While using the app]

  • 接下來就會看到離你最近的猴子了

對相依性注入容器宣告要註冊 地圖 服務

在 .NET MAUI 中,想要將某個 GPS 座標使用手機上的地圖軟體,將其顯示出來,可以透過 .NET MAUI 所提供的 地圖 物件來實現。要使用這個物件,可以透過相依性注入的方式來取得。

  • 在專案根目錄下
  • 找到並且打開 [MauiProgram.cs] 檔案
  • 找到 #region 使用 Microsoft.Extensions.DependencyInjection 套件,註冊相關要用到的服務
  • 在其下方加入底下程式碼
builder.Services.AddSingleton<IMap>(Map.Default);

底下為完成後的 [MauiProgram.cs] 檔案內容

namespace PrismMonkey;

public static class MauiProgram
{
    public static MauiApp CreateMauiApp()
    {
        var builder = MauiApp.CreateBuilder();
        builder
            .UsePrismApp<App>(PrismStartup.Configure)
            .ConfigureFonts(fonts =>
            {
                fonts.AddFont("OpenSans-Regular.ttf", "OpenSansRegular");
                fonts.AddFont("OpenSans-Semibold.ttf", "OpenSansSemibold");
            });

        #region 使用 Microsoft.Extensions.DependencyInjection 套件,註冊相關要用到的服務
        builder.Services.AddSingleton<IConnectivity>(Connectivity.Current);
        builder.Services.AddSingleton<IGeolocation>(Geolocation.Default);
        builder.Services.AddSingleton<IMap>(Map.Default);
        #endregion

        return builder.Build();
    }
}

顯示猴子所在地地圖 : 修正 ViewModel

宣告要注入的 地理位置 欄位

  • 在 [ViewModels] 資料夾下
  • 找到並且打開 [MonkeyDetailPageViewModel.cs] 檔案
  • 找到 #region 透過建構式注入的服務
  • 在其下方加入底下程式碼
private readonly IMap map;

在建構式來注入 地理位置 服務物件

  • 找到 MonkeyDetailPageViewModel 建構式方法
  • 在其方法上加入 IMap 參數,使用這樣程式碼 IMap map
  • 完成後的該建構式簽章為
public MonkeyDetailPageViewModel(INavigationService navigationService,
    IPageDialogService dialogService, IMap map) { ... }
  • 找到 this.dialogService = dialogService; 敘述
  • 在其下方加入底下程式碼
this.map = map;

建立一個命令物件,用於執行顯示猴子所在地地圖的程式碼

  • 找到 #region 在此設計要進行命令物件綁定的屬性
  • 在其下方加入底下程式碼
public DelegateCommand OpenMapCommand { get; set; }

在建構式內,為 GetClosestMonkeyCommand 物件進行初始化

  • 找到 #region 在此將命令屬性進行初始化,建立命令物件與指派委派方法
  • 在其下方加入底下程式碼
#region 根據猴子 GPS 位置,顯示此位置地圖在螢幕上
OpenMapCommand = new DelegateCommand(async () =>
{
    try
    {
        // 開啟並且顯示地圖
        await map.OpenAsync(Monkey.Latitude, Monkey.Longitude, new MapLaunchOptions
        {
            Name = Monkey.Name,
            NavigationMode = Microsoft.Maui.ApplicationModel.NavigationMode.None
        });
    }
    catch (Exception ex)
    {
        Debug.WriteLine($"Unable to launch maps: {ex.Message}");
        await dialogService.DisplayAlertAsync("Error, no Maps app!", ex.Message, "OK");
    }
});
#endregion

顯示猴子所在地地圖 : 修正 View

  • 在 [Views] 資料夾下
  • 找到並且打開 [MonkeyDetailPage.xaml] 檔案
  • 找到 <VerticalStackLayout Padding="10" Spacing="10">
  • 在其下方加入底下 XAML 標記
<Button
    Grid.Row="1" Grid.Column="1"
    Margin="8"
    Command="{Binding GetClosestMonkeyCommand}"
    IsEnabled="{Binding IsNotBusy}"
    Text="Find Closest" />

在 Android 平台執行專案

  • 點選中間上方工具列的 [Windows Machine] 這個工具列按鈕旁的下拉選單三角形

  • 從彈出功能表中,找到 [Android Emulators] 內的任何一個模擬器

  • 接者,開始執行這個專案,讓他可以在 Android 模擬器出現

  • 當出現 [所有猴子清單] 這個頁面後

  • 點選下方的 [Get Monkeys] 按鈕

  • 稍微等候一段時間,將會看到所有猴子清單物件出現在畫面上

  • 點選下方的 [Get Monkeys] 按鈕

  • 當所有猴子出現在螢幕上後

  • 隨選點選任何一個猴子

  • 在畫面上將會看到一個 [Show on Map] 按鈕

  • 請點選此按鈕

  • 接下來就會看到猴子會出現在哪個地方的地圖了