Xamarin.Forms读取并展示Android和iOS通讯录 - TerminalMACS客户端

本文同步更新地址:

  • https://dotnet9.com/11520.html

  • https://terminalmacs.com/861.html

阅读导航:

  • 一、功能说明

  • 二、代码实现

  • 三、源码获取

  • 四、参考资料

  • 五、后面计划

一、功能说明

完整思维导图:https://github.com/dotnet9/TerminalMACS/blob/master/docs/TerminalMACS.xmind 

本文介绍图中右侧画红圈处的功能,即使用Xamarin.Forms获取和展示Android和iOS的通讯录信息,下面是最终效果,由于使用的是真实手机,所以联系人姓名及电话号码打码显示。

并简单的进行了搜索功能处理,之所以说简单,是因为通讯录列表是全部读取出来了,搜索是直接从此列表进行过滤的。

下图来自:https://www.xamboy.com/2019/10/10/getting-phone-contacts-in-xamarin-forms/,本功能是参考此文所写,所以直接引用文中的图片。


二、代码实现

1、共享库工程创建联系人实体类:Contacts.cs

  1. namespace TerminalMACS.Clients.App.Models

  2. {

  3.   /// <summary>

  4.   /// 通讯录

  5.   /// </summary>

  6.    public class Contact

  7.    {

  8.       /// <summary>

  9.       /// 获取或者设置名称

  10.       /// </summary>

  11.        public string Name { get; set; }

  12.       /// <summary>

  13.       /// 获取或者设置 头像

  14.       /// </summary>

  15.        public string Image { get; set; }

  16.       /// <summary>

  17.       /// 获取或者设置 邮箱地址

  18.       /// </summary>

  19.        public string[] Emails { get; set; }

  20.       /// <summary>

  21.       /// 获取或者设置 手机号码

  22.       /// </summary>

  23.        public string[] PhoneNumbers { get; set; }

  24.    }

  25. }

2、共享库创建通讯录服务接口:IContactsService.cs

包括:

  • 一个通讯录获取请求接口:RetrieveContactsAsync

  • 一个读取一条通讯结果通知事件:OnContactLoaded

  1. using System;

  2. using System.Collections.Generic;

  3. using System.Threading;

  4. using System.Threading.Tasks;

  5. using TerminalMACS.Clients.App.Models;

  6. namespace TerminalMACS.Clients.App.Services

  7. {

  8.   /// <summary>

  9.   /// 通讯录事件参数

  10.   /// </summary>

  11.    public class ContactEventArgs:EventArgs

  12.    {

  13.        public Contact Contact { get; }

  14.        public ContactEventArgs(Contact contact)

  15.        {

  16.            Contact = contact;

  17.        }

  18.    }

  19.   /// <summary>

  20.   /// 通讯录服务接口,android和iOS终端具体的通讯录获取服务需要继承此接口

  21.   /// </summary>

  22.    public interface IContactsService

  23.    {

  24.       /// <summary>

  25.       /// 读取一条数据通知

  26.       /// </summary>

  27.        event EventHandler<ContactEventArgs> OnContactLoaded;

  28.       /// <summary>

  29.       /// 是否正在加载

  30.       /// </summary>

  31.        bool IsLoading { get; }

  32.       /// <summary>

  33.       /// 尝试获取所有通讯录

  34.       /// </summary>

  35.       /// <param name="token"></param>

  36.       /// <returns></returns>

  37.        Task<IList<Contact>> RetrieveContactsAsync(CancellationToken? token = null);

  38.    }

  39. }

3、iOS工程中添加通讯录服务,实现IContactsService接口:

  1. using Contacts;

  2. using Foundation;

  3. using System;

  4. using System.Collections.Generic;

  5. using System.IO;

  6. using System.Linq;

  7. using System.Threading;

  8. using System.Threading.Tasks;

  9. using TerminalMACS.Clients.App.Models;

  10. using TerminalMACS.Clients.App.Services;

  11. namespace TerminalMACS.Clients.App.iOS.Services

  12. {

  13.   /// <summary>

  14.   /// 通讯录获取服务

  15.   /// </summary>

  16.    public class ContactsService : NSObject, IContactsService

  17.    {

  18.        const string ThumbnailPrefix = "thumb";

  19.        bool requestStop = false;

  20.        public event EventHandler<ContactEventArgs> OnContactLoaded;

  21.        bool _isLoading = false;

  22.        public bool IsLoading => _isLoading;

  23.       /// <summary>

  24.       /// 异步请求权限

  25.       /// </summary>

  26.       /// <returns></returns>

  27.        public async Task<bool> RequestPermissionAsync()

  28.        {

  29.            var status = CNContactStore.GetAuthorizationStatus(CNEntityType.Contacts);

  30.            Tuple<bool, NSError> authotization = new Tuple<bool, NSError>(status == CNAuthorizationStatus.Authorized, null);

  31.            if (status == CNAuthorizationStatus.NotDetermined)

  32.            {

  33.                using (var store = new CNContactStore())

  34.                {

  35.                    authotization = await store.RequestAccessAsync(CNEntityType.Contacts);

  36.                }

  37.            }

  38.            return authotization.Item1;

  39.        }

  40.       /// <summary>

  41.       /// 异步请求通讯录,此方法由界面真正调用

  42.       /// </summary>

  43.       /// <param name="cancelToken"></param>

  44.       /// <returns></returns>

  45.        public async Task<IList<Contact>> RetrieveContactsAsync(CancellationToken? cancelToken = null)

  46.        {

  47.            requestStop = false;

  48.            if (!cancelToken.HasValue)

  49.                cancelToken = CancellationToken.None;

  50.           // 我们创建了一个十进制的TaskCompletionSource

  51.            var taskCompletionSource = new TaskCompletionSource<IList<Contact>>();

  52.           // 在cancellationToken中注册lambda

  53.            cancelToken.Value.Register(() =>

  54.            {

  55.               // 我们收到一条取消消息,取消TaskCompletionSource.Task

  56.                requestStop = true;

  57.                taskCompletionSource.TrySetCanceled();

  58.            });

  59.            _isLoading = true;

  60.            var task = LoadContactsAsync();

  61.           // 等待两个任务中的第一个任务完成

  62.            var completedTask = await Task.WhenAny(task, taskCompletionSource.Task);

  63.            _isLoading = false;

  64.            return await completedTask;

  65.        }

  66.       /// <summary>

  67.       /// 异步加载通讯录,具体的通讯录读取方法

  68.       /// </summary>

  69.       /// <returns></returns>

  70.        async Task<IList<Contact>> LoadContactsAsync()

  71.        {

  72.            IList<Contact> contacts = new List<Contact>();

  73.            var hasPermission = await RequestPermissionAsync();

  74.            if (hasPermission)

  75.            {

  76.                NSError error = null;

  77.                var keysToFetch = new[] { CNContactKey.PhoneNumbers, CNContactKey.GivenName, CNContactKey.FamilyName, CNContactKey.EmailAddresses, CNContactKey.ImageDataAvailable, CNContactKey.ThumbnailImageData };

  78.                var request = new CNContactFetchRequest(keysToFetch: keysToFetch);

  79.                request.SortOrder = CNContactSortOrder.GivenName;

  80.                using (var store = new CNContactStore())

  81.                {

  82.                    var result = store.EnumerateContacts(request, out error, new CNContactStoreListContactsHandler((CNContact c, ref bool stop) =>

  83.                    {

  84.                        string path = null;

  85.                        if (c.ImageDataAvailable)

  86.                        {

  87.                            path = path = Path.Combine(Path.GetTempPath(), $"{ThumbnailPrefix}-{Guid.NewGuid()}");

  88.                            if (!File.Exists(path))

  89.                            {

  90.                                var imageData = c.ThumbnailImageData;

  91.                                imageData?.Save(path, true);

  92.                            }

  93.                        }

  94.                        var contact = new Contact()

  95.                        {

  96.                            Name = string.IsNullOrEmpty(c.FamilyName) ? c.GivenName : $"{c.GivenName} {c.FamilyName}",

  97.                            Image = path,

  98.                            PhoneNumbers = c.PhoneNumbers?.Select(p => p?.Value?.StringValue).ToArray(),

  99.                            Emails = c.EmailAddresses?.Select(p => p?.Value?.ToString()).ToArray(),

  100.                        };

  101.                        if (!string.IsNullOrWhiteSpace(contact.Name))

  102.                        {

  103.                            OnContactLoaded?.Invoke(this, new ContactEventArgs(contact));

  104.                            contacts.Add(contact);

  105.                        }

  106.                        stop = requestStop;

  107.                    }));

  108.                }

  109.            }

  110.            return contacts;

  111.        }

  112.    }

  113. }

4、在iOS工程中的Info.plist文件添加通讯录权限使用说明

5、在Android工程中添加读取通讯录权限配置:AndroidManifest.xml

  1. <uses-permission android:name="android.permission.READ_CONTACTS"/>

完整权限配置如下

  1. <?xml version="1.0" encoding="utf-8"?>

  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android" android:versionCode="1" android:versionName="1.0" package="com.companyname.terminalmacs.clients.app">

  3.  <uses-sdk android:minSdkVersion="21" android:targetSdkVersion="28" />

  4.  <application android:label="TerminalMACS.Clients.App.Android"></application>

  5.  <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

  6.  <uses-permission android:name="android.permission.READ_CONTACTS"/>

  7.  <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

  8. </manifest>

6、在Android工程中添加通讯录服务,实现IContactServer接口:ContactsService.cs

  1. using Acr.UserDialogs;

  2. using Android;

  3. using Android.App;

  4. using Android.Content;

  5. using Android.Content.PM;

  6. using Android.Database;

  7. using Android.Provider;

  8. using Android.Runtime;

  9. using Android.Support.V4.App;

  10. using Plugin.CurrentActivity;

  11. using System;

  12. using System.Collections.Generic;

  13. using System.IO;

  14. using System.Linq;

  15. using System.Threading;

  16. using System.Threading.Tasks;

  17. using TerminalMACS.Clients.App.Models;

  18. using TerminalMACS.Clients.App.Services;

  19. namespace TerminalMACS.Clients.App.Droid.Services

  20. {

  21.   /// <summary>

  22.   /// 通讯录获取服务

  23.   /// </summary>

  24.    public class ContactsService : IContactsService

  25.    {

  26.        const string ThumbnailPrefix = "thumb";

  27.        bool stopLoad = false;

  28.        static TaskCompletionSource<bool> contactPermissionTcs;

  29.        public string TAG

  30.        {

  31.            get

  32.            {

  33.                return "MainActivity";

  34.            }

  35.        }

  36.        bool _isLoading = false;

  37.        public bool IsLoading => _isLoading;

  38.       //权限请求状态码

  39.        public const int RequestContacts = 1239;

  40.       /// <summary>

  41.       /// 获取通讯录需要的请求权限

  42.       /// </summary>

  43.        static string[] PermissionsContact = {

  44.            Manifest.Permission.ReadContacts

  45.        };

  46.        public event EventHandler<ContactEventArgs> OnContactLoaded;

  47.       /// <summary>

  48.       /// 异步请求通讯录权限

  49.       /// </summary>

  50.        async void RequestContactsPermissions()

  51.        {

  52.           //检查是否可以弹出申请读、写通讯录权限

  53.            if (ActivityCompat.ShouldShowRequestPermissionRationale(CrossCurrentActivity.Current.Activity, Manifest.Permission.ReadContacts)

  54.                || ActivityCompat.ShouldShowRequestPermissionRationale(CrossCurrentActivity.Current.Activity, Manifest.Permission.WriteContacts))

  55.            {

  56.               // 如果未授予许可,请向用户提供其他理由用户将从使用权限的附加上下文中受益。

  57.               // 例如,如果请求先前被拒绝。

  58.                await UserDialogs.Instance.AlertAsync("通讯录权限", "此操作需要“通讯录”权限", "确定");

  59.            }

  60.            else

  61.            {

  62.               // 尚未授予通讯录权限。直接请求这些权限。

  63.                ActivityCompat.RequestPermissions(CrossCurrentActivity.Current.Activity, PermissionsContact, RequestContacts);

  64.            }

  65.        }

  66.       /// <summary>

  67.       /// 收到用户响应请求权限操作后的结果

  68.       /// </summary>

  69.       /// <param name="requestCode"></param>

  70.       /// <param name="permissions"></param>

  71.       /// <param name="grantResults"></param>

  72.        public static void OnRequestPermissionsResult(int requestCode, string[] permissions, [GeneratedEnum] Android.Content.PM.Permission[] grantResults)

  73.        {

  74.            if (requestCode == RequestContacts)

  75.            {

  76.               // 我们请求了多个通讯录权限,因此需要检查相关的所有权限

  77.                if (PermissionUtil.VerifyPermissions(grantResults))

  78.                {

  79.                   // 已授予所有必需的权限,显示联系人片段。

  80.                    contactPermissionTcs.TrySetResult(true);

  81.                }

  82.                else

  83.                {

  84.                    contactPermissionTcs.TrySetResult(false);

  85.                }

  86.            }

  87.        }

  88.       /// <summary>

  89.       /// 异步请求权限

  90.       /// </summary>

  91.       /// <returns></returns>

  92.        public async Task<bool> RequestPermissionAsync()

  93.        {

  94.            contactPermissionTcs = new TaskCompletionSource<bool>();

  95.           // 验证是否已授予所有必需的通讯录权限。

  96.            if (Android.Support.V4.Content.ContextCompat.CheckSelfPermission(CrossCurrentActivity.Current.Activity, Manifest.Permission.ReadContacts) != (int)Permission.Granted

  97.                || Android.Support.V4.Content.ContextCompat.CheckSelfPermission(CrossCurrentActivity.Current.Activity, Manifest.Permission.WriteContacts) != (int)Permission.Granted)

  98.            {

  99.               // 尚未授予通讯录权限。

  100.                RequestContactsPermissions();

  101.            }

  102.            else

  103.            {

  104.               // 已授予通讯录权限。

  105.                contactPermissionTcs.TrySetResult(true);

  106.            }

  107.            return await contactPermissionTcs.Task;

  108.        }

  109.       /// <summary>

  110.       /// 异步请求通讯录,此方法由界面真正调用

  111.       /// </summary>

  112.       /// <param name="cancelToken"></param>

  113.       /// <returns></returns>

  114.        public async Task<IList<Contact>> RetrieveContactsAsync(CancellationToken? cancelToken = null)

  115.        {

  116.            stopLoad = false;

  117.            if (!cancelToken.HasValue)

  118.                cancelToken = CancellationToken.None;

  119.           // 我们创建了一个十进制的TaskCompletionSource

  120.            var taskCompletionSource = new TaskCompletionSource<IList<Contact>>();

  121.           // 在cancellationToken中注册lambda

  122.            cancelToken.Value.Register(() =>

  123.            {

  124.               // 我们收到一条取消消息,取消TaskCompletionSource.Task

  125.                stopLoad = true;

  126.                taskCompletionSource.TrySetCanceled();

  127.            });

  128.            _isLoading = true;

  129.            var task = LoadContactsAsync();

  130.           // 等待两个任务中的第一个任务完成

  131.            var completedTask = await Task.WhenAny(task, taskCompletionSource.Task);

  132.            _isLoading = false;

  133.            return await completedTask;

  134.        }

  135.       /// <summary>

  136.       /// 异步加载通讯录,具体的通讯录读取方法

  137.       /// </summary>

  138.       /// <returns></returns>

  139.        async Task<IList<Contact>> LoadContactsAsync()

  140.        {

  141.            IList<Contact> contacts = new List<Contact>();

  142.            var hasPermission = await RequestPermissionAsync();

  143.            if (!hasPermission)

  144.            {

  145.                return contacts;

  146.            }

  147.            var uri = ContactsContract.Contacts.ContentUri;

  148.            var ctx = Application.Context;

  149.            await Task.Run(() =>

  150.            {

  151.               // 暂时只请求通讯录Id、DisplayName、PhotoThumbnailUri,可以扩展

  152.                var cursor = ctx.ApplicationContext.ContentResolver.Query(uri, new string[]

  153.                {

  154.                        ContactsContract.Contacts.InterfaceConsts.Id,

  155.                        ContactsContract.Contacts.InterfaceConsts.DisplayName,

  156.                        ContactsContract.Contacts.InterfaceConsts.PhotoThumbnailUri

  157.                }, null, null, $"{ContactsContract.Contacts.InterfaceConsts.DisplayName} ASC");

  158.                if (cursor.Count > 0)

  159.                {

  160.                    while (cursor.MoveToNext())

  161.                    {

  162.                        var contact = CreateContact(cursor, ctx);

  163.                        if (!string.IsNullOrWhiteSpace(contact.Name))

  164.                        {

  165.                           // 读取出一条,即通知界面展示

  166.                            OnContactLoaded?.Invoke(this, new ContactEventArgs(contact));

  167.                            contacts.Add(contact);

  168.                        }

  169.                        if (stopLoad)

  170.                            break;

  171.                    }

  172.                }

  173.            });

  174.            return contacts;

  175.        }

  176.       /// <summary>

  177.       /// 读取一条通讯录数据

  178.       /// </summary>

  179.       /// <param name="cursor"></param>

  180.       /// <param name="ctx"></param>

  181.       /// <returns></returns>

  182.        Contact CreateContact(ICursor cursor, Context ctx)

  183.        {

  184.            var contactId = GetString(cursor, ContactsContract.Contacts.InterfaceConsts.Id);

  185.            var numbers = GetNumbers(ctx, contactId);

  186.            var emails = GetEmails(ctx, contactId);

  187.            var uri = GetString(cursor, ContactsContract.Contacts.InterfaceConsts.PhotoThumbnailUri);

  188.            string path = null;

  189.            if (!string.IsNullOrEmpty(uri))

  190.            {

  191.                try

  192.                {

  193.                    using (var stream = Android.App.Application.Context.ContentResolver.OpenInputStream(Android.Net.Uri.Parse(uri)))

  194.                    {

  195.                        path = Path.Combine(Path.GetTempPath(), $"{ThumbnailPrefix}-{Guid.NewGuid()}");

  196.                        using (var fstream = new FileStream(path, FileMode.Create))

  197.                        {

  198.                            stream.CopyTo(fstream);

  199.                            fstream.Close();

  200.                        }

  201.                        stream.Close();

  202.                    }

  203.                }

  204.                catch (Exception ex)

  205.                {

  206.                    System.Diagnostics.Debug.WriteLine(ex);

  207.                }

  208.            }

  209.            var contact = new Contact

  210.            {

  211.                Name = GetString(cursor, ContactsContract.Contacts.InterfaceConsts.DisplayName),

  212.                Emails = emails,

  213.                Image = path,

  214.                PhoneNumbers = numbers,

  215.            };

  216.            return contact;

  217.        }

  218.       /// <summary>

  219.       /// 读取联系人电话号码

  220.       /// </summary>

  221.       /// <param name="ctx"></param>

  222.       /// <param name="contactId"></param>

  223.       /// <returns></returns>

  224.        string[] GetNumbers(Context ctx, string contactId)

  225.        {

  226.            var key = ContactsContract.CommonDataKinds.Phone.Number;

  227.            var cursor = ctx.ApplicationContext.ContentResolver.Query(

  228.                ContactsContract.CommonDataKinds.Phone.ContentUri,

  229.                null,

  230.                ContactsContract.CommonDataKinds.Phone.InterfaceConsts.ContactId + " = ?",

  231.                new[] { contactId },

  232.                null

  233.            );

  234.            return ReadCursorItems(cursor, key)?.ToArray();

  235.        }

  236.       /// <summary>

  237.       /// 读取联系人邮箱地址

  238.       /// </summary>

  239.       /// <param name="ctx"></param>

  240.       /// <param name="contactId"></param>

  241.       /// <returns></returns>

  242.        string[] GetEmails(Context ctx, string contactId)

  243.        {

  244.            var key = ContactsContract.CommonDataKinds.Email.InterfaceConsts.Data;

  245.            var cursor = ctx.ApplicationContext.ContentResolver.Query(

  246.                ContactsContract.CommonDataKinds.Email.ContentUri,

  247.                null,

  248.                ContactsContract.CommonDataKinds.Email.InterfaceConsts.ContactId + " = ?",

  249.                new[] { contactId },

  250.                null);

  251.            return ReadCursorItems(cursor, key)?.ToArray();

  252.        }

  253.        IEnumerable<string> ReadCursorItems(ICursor cursor, string key)

  254.        {

  255.            while (cursor.MoveToNext())

  256.            {

  257.                var value = GetString(cursor, key);

  258.                yield return value;

  259.            }

  260.            cursor.Close();

  261.        }

  262.        string GetString(ICursor cursor, string key)

  263.        {

  264.            return cursor.GetString(cursor.GetColumnIndex(key));

  265.        }

  266.    }

  267. }

需要添加 Plugin.CurrentActivity 和 Acr.UserDialogs 包。

7、Android工程添加权限处理判断类

Permission.Util.cs

  1. using Android.Content.PM;

  2. namespace TerminalMACS.Clients.App.Droid

  3. {

  4.    public static class PermissionUtil

  5.    {

  6.        /**

  7.    * 通过验证给定数组中的每个条目的值是否为Permission.Granted,检查是否已授予所有给定权限。

  8.    *

  9.    * See Activity#onRequestPermissionsResult (int, String[], int[])

  10.    */

  11.        public static bool VerifyPermissions(Permission[] grantResults)

  12.        {

  13.           // 必须至少检查一个结果.

  14.            if (grantResults.Length < 1)

  15.                return false;

  16.           // 验证是否已授予每个必需的权限,否则返回false.

  17.            foreach (Permission result in grantResults)

  18.            {

  19.                if (result != Permission.Granted)

  20.                {

  21.                    return false;

  22.                }

  23.            }

  24.            return true;

  25.        }

  26.    }

  27. }

MainActivity.OnRequestPermissionResult是权限申请结果处理函数,在此函数中调用ContactsService.OnRequestPermissionsResult通知通讯录服务权限处理结果。

MainActivity.cs

  1. using Acr.UserDialogs;

  2. using Android.App;

  3. using Android.Content.PM;

  4. using Android.OS;

  5. using Android.Runtime;

  6. using TerminalMACS.Clients.App.Droid.Services;

  7. using TerminalMACS.Clients.App.Services;

  8. namespace TerminalMACS.Clients.App.Droid

  9. {

  10.    [Activity(Label = "TerminalMACS.Clients.App", Icon = "@mipmap/icon", Theme = "@style/MainTheme", MainLauncher = true, ConfigurationChanges = ConfigChanges.ScreenSize | ConfigChanges.Orientation)]

  11.    public class MainActivity : global::Xamarin.Forms.Platform.Android.FormsAppCompatActivity

  12.    {

  13.        IContactsService contactsService = new ContactsService();

  14.        protected override void OnCreate(Bundle savedInstanceState)

  15.        {

  16.            TabLayoutResource = Resource.Layout.Tabbar;

  17.            ToolbarResource = Resource.Layout.Toolbar;

  18.            base.OnCreate(savedInstanceState);

  19.            Xamarin.Essentials.Platform.Init(this, savedInstanceState);

  20.            global::Xamarin.Forms.Forms.Init(this, savedInstanceState);

  21.            UserDialogs.Init(() => this);

  22.           // 将通讯录服务实例传递给共享库,由共享库使用读取通讯录接口

  23.            LoadApplication(new App(contactsService));

  24.        }

  25.        public override void OnRequestPermissionsResult(int requestCode, string[] permissions, [GeneratedEnum] Android.Content.PM.Permission[] grantResults)

  26.        {

  27.            Xamarin.Essentials.Platform.OnRequestPermissionsResult(requestCode, permissions, grantResults);

  28.           // 通讯录服务处理权限请求结果

  29.            ContactsService.OnRequestPermissionsResult(requestCode, permissions, grantResults);

  30.            

  31.            base.OnRequestPermissionsResult(requestCode, permissions, grantResults);

  32.        }

  33.    }

  34. }

8、创建通讯录ViewModel,并使用通讯录服务

  1. using System;

  2. using System.Collections;

  3. using System.Collections.Generic;

  4. using System.Collections.ObjectModel;

  5. using System.Linq;

  6. using System.Threading.Tasks;

  7. using System.Windows.Input;

  8. using TerminalMACS.Clients.App.Models;

  9. using TerminalMACS.Clients.App.Services;

  10. using Xamarin.Forms;

  11. namespace TerminalMACS.Clients.App.ViewModels

  12. {

  13.   /// <summary>

  14.   /// 通讯录ViewModel

  15.   /// </summary>

  16.    public class ContactViewModel : BaseViewModel

  17.    {

  18.       /// <summary>

  19.       /// 通讯录服务接口

  20.       /// </summary>

  21.        IContactsService _contactService;

  22.       /// <summary>

  23.       /// 标题

  24.       /// </summary>

  25.        public new string Title => "通讯录";

  26.        private string _SearchText;

  27.       /// <summary>

  28.       /// 搜索关键字

  29.       /// </summary>

  30.        public string SearchText

  31.        {

  32.            get { return _SearchText; }

  33.            set

  34.            {

  35.                SetProperty(ref _SearchText, value);

  36.            }

  37.        }

  38.       /// <summary>

  39.       /// 通讯录搜索命令

  40.       /// </summary>

  41.        public ICommand RaiseSearchCommand { get; }

  42.       /// <summary>

  43.       /// 通讯录列表

  44.       /// </summary>

  45.        public ObservableCollection<Contact> Contacts { get; set; }

  46.        private List<Contact> _FilteredContacts;

  47.       /// <summary>

  48.       /// 通讯录过滤列表

  49.       /// </summary>

  50.        public List<Contact> FilteredContacts

  51.        {

  52.            get { return _FilteredContacts; }

  53.            set

  54.            {

  55.                SetProperty(ref _FilteredContacts, value);

  56.            }

  57.        }

  58.        public ContactViewModel(IContactsService contactService)

  59.        {

  60.            _contactService = contactService;

  61.            Contacts = new ObservableCollection<Contact>();

  62.            Xamarin.Forms.BindingBase.EnableCollectionSynchronization(Contacts, null, ObservableCollectionCallback);

  63.            _contactService.OnContactLoaded += OnContactLoaded;

  64.            LoadContacts();

  65.            RaiseSearchCommand = new Command(RaiseSearchHandle);

  66.        }

  67.       /// <summary>

  68.       /// 过滤通讯录

  69.       /// </summary>

  70.        void RaiseSearchHandle()

  71.        {

  72.            if (string.IsNullOrEmpty(SearchText))

  73.            {

  74.                FilteredContacts = Contacts.ToList();

  75.                return;

  76.            }

  77.            Func<Contact, bool> checkContact = (s) =>

  78.            {

  79.                if (!string.IsNullOrWhiteSpace(s.Name) && s.Name.ToLower().Contains(SearchText.ToLower()))

  80.                {

  81.                    return true;

  82.                }

  83.                else if (s.PhoneNumbers.Length > 0 && s.PhoneNumbers.ToList().Exists(cu => cu.ToString().Contains(SearchText)))

  84.                {

  85.                    return true;

  86.                }

  87.                return false;

  88.            };

  89.            FilteredContacts = Contacts.ToList().Where(checkContact).ToList();

  90.        }

  91.       /// <summary>

  92.       /// BindingBase.EnableCollectionSynchronization 为集合启用跨线程更新

  93.       /// </summary>

  94.       /// <param name="collection"></param>

  95.       /// <param name="context"></param>

  96.       /// <param name="accessMethod"></param>

  97.       /// <param name="writeAccess"></param>

  98.        void ObservableCollectionCallback(IEnumerable collection, object context, Action accessMethod, bool writeAccess)

  99.        {

  100.           // `lock` ensures that only one thread access the collection at a time

  101.            lock (collection)

  102.            {

  103.                accessMethod?.Invoke();

  104.            }

  105.        }

  106.       /// <summary>

  107.       /// 收到事件通知,读取一条通讯录信息

  108.       /// </summary>

  109.       /// <param name="sender"></param>

  110.       /// <param name="e"></param>

  111.        private void OnContactLoaded(object sender, ContactEventArgs e)

  112.        {

  113.            Contacts.Add(e.Contact);

  114.            RaiseSearchHandle();

  115.        }

  116.       /// <summary>

  117.       /// 异步读取终端通讯录

  118.       /// </summary>

  119.       /// <returns></returns>

  120.        async Task LoadContacts()

  121.        {

  122.            try

  123.            {

  124.                await _contactService.RetrieveContactsAsync();

  125.            }

  126.            catch (TaskCanceledException)

  127.            {

  128.                Console.WriteLine("任务已经取消");

  129.            }

  130.        }

  131.    }

  132. }

9、添加通讯录页面展示通讯录数据

  1. <?xml version="1.0" encoding="utf-8" ?>

  2. <ContentPage xmlns="http://xamarin.com/schemas/2014/forms"

  3.             xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"

  4.             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"

  5.             xmlns:d="http://xamarin.com/schemas/2014/forms/design"

  6.             xmlns:ios="clr-namespace:Xamarin.Forms.PlatformConfiguration.iOSSpecific;assembly=Xamarin.Forms.Core"

  7.             mc:Ignorable="d"

  8.             Title="{Binding Title}"

  9.             x:Class="TerminalMACS.Clients.App.Views.ContactPage"

  10.             ios:Page.UseSafeArea="true">

  11.    <ContentPage.Content>

  12.        <StackLayout>

  13.            <SearchBar x:Name="filterText"

  14.                        HeightRequest="40"

  15.                        Text="{Binding SearchText}"

  16.                       SearchCommand="{Binding RaiseSearchCommand}"/>

  17.            <ListView   ItemsSource="{Binding FilteredContacts}"

  18.                        HasUnevenRows="True">

  19.                <ListView.ItemTemplate>

  20.                    <DataTemplate>

  21.                        <ViewCell>

  22.                            <StackLayout Padding="10"

  23.                                         Orientation="Horizontal">

  24.                                <Image  Source="{Binding Image}"

  25.                                        VerticalOptions="Center"

  26.                                        x:Name="image"

  27.                                        Aspect="AspectFit"

  28.                                        HeightRequest="60"/>

  29.                                <StackLayout VerticalOptions="Center">

  30.                                    <Label Text="{Binding Name}"

  31.                                       FontAttributes="Bold"/>

  32.                                    <Label Text="{Binding PhoneNumbers[0]}"/>

  33.                                    <Label Text="{Binding Emails[0]}"/>

  34.                                </StackLayout>

  35.                            </StackLayout>

  36.                        </ViewCell>

  37.                    </DataTemplate>

  38.                </ListView.ItemTemplate>

  39.            </ListView>

  40.        </StackLayout>

  41.    </ContentPage.Content>

  42. </ContentPage>

三、源码获取

  • 1.完整源码:https://github.com/dotnet9/TerminalMACS

  • 2.Android客户端可成功取得通讯录数据,并可查询;

已编译的Android客户端:https://terminalmacs.com/terminalmacs-clients-app-android

  • 3.iOS读取通讯录功能代码也已添加,但由于本人没有iOS测试环境,所以未验证,有条件的朋友可以测试下iOS的通讯录读取功能,如果代码不起作用,可参考本文参考的文章检查iOS代码。

四、参考资料

Getting phone contacts in Xamarin Forms:https://www.xamboy.com/2019/10/10/getting-phone-contacts-in-xamarin-forms/

参考文章末尾有源代码链接。

五、后面计划

Xamarin.Forms客户端基本信息获取,比如IMEI、IMSI、本机号码、Mac地址等。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/310815.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

paragon+ntfs+linux,NTFS For Mac 超强兼容性

NTFS For Mac是为解决Windows和Mac OS X不兼容问题而开发的低级别档案系统驱动&#xff0c;提供在Mac OS X下完全读/写访问NTFS档案系统的任何版本。兼容mac OS X所有版本、32/64位内核模式&#xff0c;及其它第三方软件。不仅如此&#xff0c;NTFS For Mac 超强兼容性支持更多…

Asp.Net Core Ocelot Consul 微服务

做一个简单的微服务架构如下图&#xff1a;这个图表示的是一个网关代理Consul的两个服务&#xff0c;consul每个服务注册集群安装 Consul的服务&#xff0c;这里安装单机版的&#xff0c;集群版配置最低要求&#xff08;3个Consul server&#xff09;的需要三台虚拟机&#xff…

.Neter们,你真的应该了解下EFCore3.x

本期导读&#xff1a;技术文&#xff0c;带你了解关于EntityFrameworkCore3.x的那些事&#xff0c;本文共1493个字&#xff0c;阅读大约需要3分钟。文末福利不要错过哦&#xff01;是的各位.Neter&#xff0c;不用怀疑&#xff0c;使用O/RM的开发者越来越多了&#xff0c;从风起…

LeetCode 111二叉树的最小深度-简单

给定一个二叉树&#xff0c;找出其最小深度。 最小深度是从根节点到最近叶子节点的最短路径上的节点数量。 说明&#xff1a;叶子节点是指没有子节点的节点。 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;2 示例 2&#xff1a; 输…

istio回归「单体应用」对我们的启发

大家好&#xff0c;我是Z哥。这次分享给大家的是一篇与技术相关的文章&#xff0c;但是我想表达的核心观点并不仅限于技术范围。我们中国有句古话&#xff0c;分久必合&#xff0c;合久必分。很多事物的发展都逃不开这个规律。如今&#xff0c;这件事也正在分布式、微服务概念大…

LeetCode 110平衡二叉树-简单

给定一个二叉树&#xff0c;判断它是否是高度平衡的二叉树。 本题中&#xff0c;一棵高度平衡二叉树定义为&#xff1a; 一个二叉树每个节点 的左右两个子树的高度差的绝对值不超过 1 。示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;t…

c语言中如何使用面向对象编程,如何使用C语言的面向对象

我们都知道&#xff0c;C才是面向对象的语言&#xff0c;但是C语言是否能使用面向对象的功能&#xff1f;(1)继承性typedef struct _parent{int data_parent;}Parent;typedef struct _Child{struct _parent parent;int data_child;}Child;在设计C语言继承性的时候&#xff0c;我…

LeetCode 112路径总和-简单

给你二叉树的根节点 root 和一个表示目标和的整数 targetSum &#xff0c;判断该树中是否存在 根节点到叶子节点 的路径&#xff0c;这条路径上所有节点值相加等于目标和 targetSum 。 叶子节点 是指没有子节点的节点。 示例 1&#xff1a; 输入&#xff1a;root [5,4,8,11,…

ASP.NET MVC升级到ASP.NET Core MVC踩坑小结

写在前面ASP.NET Core是微软新推出的支持跨平台、高性能、开源的开发框架&#xff0c;它的优势不必多说&#xff0c;因为已经说得太多了。当然&#xff0c;现在依然有着数量庞大的系统运行于.NET Framework上&#xff0c;由于有大量的Break Changes&#xff0c;很多项目项目团队…

用函数求C15的值C语言,南开19春学期(1503、1509、1603、1609、1703)《C语言程序设计》在线作业-1辅导资料.docx...

南开19春学期(1503、1509、1603、1609、1703)《C语言程序设计》在线作业-1辅导资料.docx 南开19春学期(1503、1509、1603、1609、1703)C语言程序设计在线作业-11、D 2、B 3、C 4、A 5、D 一、单选题共40题&#xff0c;80分1、以下对一维整型数组 a 的正确说明是 Aint a10 ;Bint…

LeetCode 563二叉树的坡度-简单

给定一个二叉树&#xff0c;计算 整个树 的坡度 。 一个树的 节点的坡度 定义即为&#xff0c;该节点左子树的节点之和和右子树节点之和的 差的绝对值 。如果没有左子树的话&#xff0c;左子树的节点之和为 0 &#xff1b;没有右子树的话也是一样。空结点的坡度是 0 。 整个树…

CIO/CTO都应该掌握和了解的EA(企业架构)

我们已进入数字化技术推动的第四次工业革命&#xff0c;是以工业互联网建设为标志。单纯从IT的视角管理信息化系统让许多企业深陷管理困境&#xff0c;解决问题也是按下葫芦浮起瓢。实际上&#xff0c;IT的服务对象是企业的战略、组织、管理、流程等一系列的要素&#xff0c;因…

扛并发主力军,引入应用层缓存

1.背景缓存的使用一定是今后开发中100%会用到的技术&#xff0c;尤其是Redis相关的问题&#xff0c;如果面试官不问我我几个缓存相关的问题&#xff0c;那我觉得我可能是去了个假的互联网公司。这里考虑到有些初学者刚刚出校园或者自学中&#xff0c;准许我多费口舌介绍下关于缓…

LeetCode 783二叉搜索树节点最小距离-简单

给你一个二叉搜索树的根节点 root &#xff0c;返回 树中任意两不同节点值之间的最小差值 。 示例 1&#xff1a; 输入&#xff1a;root [4,2,6,1,3] 输出&#xff1a;1 示例 2&#xff1a; 输入&#xff1a;root [1,0,48,null,null,12,49] 输出&#xff1a;1 提示&…

使用 VMware + win10 + vs2019 从零搭建双机内核调试环境

我在前面的文章——《使用 VMware win10 VirtualKD windbg 从零搭建双机内核调试环境》分享了使用 windbg 进行双机内核调试的环境搭建的步骤。有小伙伴儿留言说&#xff1a;在使用 vs 进行双机内核调试的时候&#xff0c;总是连不上。希望能发一篇使用 vs 进行双机内核调试…

C#中的9个“黑魔法”与“骚操作”

C#中的9个“黑魔法”与“骚操作”我们知道 C#是非常先进的语言&#xff0c;因为是它很有远见的“语法糖”。这些“语法糖”有时过于好用&#xff0c;导致有人觉得它是 C#编译器写死的东西&#xff0c;没有道理可讲的——有点像“黑魔法”。那么我们可以看看 C#这些高级语言功能…

LeetCode 872叶子相似的树-简单

请考虑一棵二叉树上所有的叶子&#xff0c;这些叶子的值按从左到右的顺序排列形成一个 叶值序列 。 举个例子&#xff0c;如上图所示&#xff0c;给定一棵叶值序列为 (6, 7, 4, 9, 8) 的树。 如果有两棵二叉树的叶值序列是相同&#xff0c;那么我们就认为它们是 叶相似 的。 …

.NET Core开发实战(第35课:MediatR:让领域事件处理更加优雅)--学习笔记

35 | MediatR&#xff1a;让领域事件处理更加优雅核心对象IMediatorINotificationINotificationHandler这两个与之前的 Request 的行为是不一样的&#xff0c;接下来看一下代码internal class MyEvent : INotification {public string EventName { get; set; } }internal class…

LeetCode 559N叉树的最大深度-简单

给定一个 N 叉树&#xff0c;找到其最大深度。 最大深度是指从根节点到最远叶子节点的最长路径上的节点总数。 N 叉树输入按层序遍历序列化表示&#xff0c;每组子节点由空值分隔&#xff08;请参见示例&#xff09;。 示例 1&#xff1a; 输入&#xff1a;root [1,null,3,…

android 5.0状态栏下载地址,Android沉浸式状态栏(5.0以上系统)

Android沉浸式状态栏(5.0以上系统)沉浸式状态栏可以分为两种:1.直接给状态栏设置颜色 (如下图:)这里写图片描述java代码形式:if (Build.VERSION.SDK_INT > Build.VERSION_CODES.LOLLIPOP) {Window window activity.getWindow();window.addFlags(WindowManager.LayoutParams…