Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> Android開發實例 >> Mono for Android實現高效的導航實例

Mono for Android實現高效的導航實例

編輯:Android開發實例

 Android 4.0 系統定義了一系列的高效導航方式 (Effective Navigation), 主要包括標簽、下拉列表、以及向上和返回等, 本文介紹如何用 Mono for Android 實現這些的導航方式。

准備 Android 4.0 ICS 項目

新建 Android ICS 項目

打開 MonoDevelop , 新建一個 Mono for Android 項目, 並在項目的屬性頁將 Target Framework 設置為 Android 4.0.3 (Ice Cream Sandwich) , 如下圖所示:

添加 Mono.Android.Support.v4 引用項

在解決方案窗口, 選中項目的引用節點, 右擊選擇編輯引用, 添加對 Mono.Android.Support.v4.dll 的引用, 如圖所示:

在項目中新建一個目錄 SupportLib , 並添加對 android-support-v4.jar 文件(位於 android-sdk/extras/android/support/v4 目錄, 如果沒有, 需要用 SDK Manager 安裝)的引用, 並將 jar 文件的編譯動作 (BuildAction) 設置為 AndroidJavaLibrary , 如下圖所示:

本文提到的導航都是根據 Android 4.0 設計規范中推薦的 ActionBar 實現的, 因此整個應用程序啟用帶 ActionBar 的主題, 如果使用 Java 的話, 需要手工編輯 AppManifest.xml 文件的設置, 而用 Mono for Android 的話, 基本上不需要手工編輯這個文件。

Mono for Android 的做法是, 新建一個 App 類, 繼承自 Android.App.Application 類, 並添加 Android.App.ApplicationAttribute 標記, 在編譯時, Mono for Android 會根據這些標記自動生成一個 AppManifest.xml 文件並打包到最終的 apk 文件中。

App 類的代碼如下:

[Application(Label = "@string/AppName", Icon = "@drawable/ic_launcher", Theme = "@android:style/Theme.Holo.Light.DarkActionBar")] public class App : Application { public App(IntPtr javaReference, JniHandleOwnership transfer) : base(javaReference, transfer) { } }

添加這個類之後, 項目中的每個 Activity 將默認都是用這個主題, 如果有 Activity 要使用其它的主題, 才需要添加自己的主題屬性。

標簽導航

Android 的標簽用 ActionBar 實現, 用戶既可以點擊標簽切換視圖, 也可以水平滑動切換視圖, 如下圖所示:

用戶既可以點擊上面的 ‘SECTION 0’、 ‘SECTION 1’、 ‘SECTION 2’ 標簽切換視圖, 也可以在視圖上水平拖動切換視圖, 同時標簽選中項也要同步選中, 實現的代碼如下:

  1. [Activity (Label = "@string/AppName", Icon = "@drawable/ic_launcher", MainLauncher = true)] 
  2. public class MainActivity : FragmentActivity { 
  3.  
  4.     /// <summary> 
  5.     /// AppSectionsPagerAdapter 提供要顯示的視圖, 繼承自 
  6.     /// Mono.Android.Support.V4.View.PagerAdapter, 所有加載過視圖都保存在內存中,  
  7.     /// 如果視圖占用內存過多, 考慮替換成 FragmentStatePagerAdapter 。 
  8.     /// </summary> 
  9.     AppSectionsPagerAdapter _appSectionsPagerAdapter; 
  10.  
  11.     /// <summary> 
  12.     /// 用 ViewPager 來顯示視圖三個主視圖, 每次只顯示一個。 
  13.     /// </summary> 
  14.     ViewPager _viewPager; 
  15.  
  16.     protected override void OnCreate(Bundle bundle) { 
  17.         base.OnCreate(bundle); 
  18.  
  19.         this.SetContentView(Resource.Layout.MainActivity); 
  20.  
  21.         // 創建 Adapter 
  22.         this._appSectionsPagerAdapter = new AppSectionsPagerAdapter(this.SupportFragmentManager); 
  23.         // 設置 ActionBar 
  24.         var actionBar = this.ActionBar; 
  25.         // 首頁不需要向上的 Home 按鈕 
  26.         actionBar.SetHomeButtonEnabled(false); 
  27.         // 設置標簽導航模式 
  28.         actionBar.NavigationMode = ActionBarNavigationMode.Tabs; 
  29.         // 設置 ViewPager 的 Adapter , 這樣用戶就可以水平滑動切換視圖了 
  30.         this._viewPager = this.FindViewById<ViewPager>(Resource.Id.Pager); 
  31.         this._viewPager.Adapter = this._appSectionsPagerAdapter; 
  32.         // 當水平滑動切換視圖時, 設置選中的標簽 
  33.         this._viewPager.PageSelected += delegate(object sender, ViewPager.PageSelectedEventArgs e) { 
  34.             actionBar.SetSelectedNavigationItem(e.P0); 
  35.         }; 
  36.  
  37.         // 依次添加三個標簽, 並添加標簽的選中事件處理函數, 設置當前的視圖。 
  38.         for (var i = 0; i < this._appSectionsPagerAdapter.Count; i++) { 
  39.             var tab = actionBar.NewTab().SetText(this._appSectionsPagerAdapter.GetPageTitle(i)); 
  40.             tab.TabSelected += delegate(object sender, Android.App.ActionBar.TabEventArgs e) { 
  41.                 this._viewPager.CurrentItem = tab.Position; 
  42.             }; 
  43.             actionBar.AddTab(tab); 
  44.         } 
  45.     } 

左右導航

標簽導航並不適合所有的場景, 有時僅僅需要顯示視圖的標題即可, 但是同樣可以水平滑動切換視圖, 如下圖所示:

這種導航方式相當於標簽式導航的簡化版, 用戶只可以左右滑動切換視圖, 實現的代碼如下:

  1. protected override void OnCreate(Bundle bundle) { 
  2.     base.OnCreate(bundle); 
  3.     this.SetContentView(Resource.Layout.CollectionDemoActivity); 
  4.     // 創建 Adapter 
  5.     this._demoCollectionPagerAdapter = new DemoCollectionPagerAdapter(this.SupportFragmentManager); 
  6.  
  7.     // 設置 ViewPager 的 Adapter 
  8.     this._viewPager = this.FindViewById<ViewPager>(Resource.Id.Pager); 
  9.     this._viewPager.Adapter = this.mDemoCollectionPagerAdapter; 

因為要顯示標題, 所以這個 Activity 的 Layout 添加了一個 PagerTitleStrip , Layout 源代碼如下:

<android.support.v4.view.ViewPager     xmlns:android="http://schemas.android.com/apk/res/android" 	android:id="@+id/Pager" 	android:orientation="vertical" 	android:layout_width="match_parent" 	android:layout_height="match_parent" 	> 	<!-- 	PaterTitleStrip 即可顯示選中頁面的標題, 也顯示臨近選中的幾個視圖的標題 	--> 	<android.support.v4.view.PagerTitleStrip android:id="@+id/PagerTitleStrip" 		android:layout_width="match_parent" 		android:layout_height="wrap_content" 		android:layout_gravity="top" 		android:background="#33b5e5" 		android:textColor="#fff" 		android:paddingTop="4dp" 		android:paddingBottom="4dp" />  </android.support.v4.view.ViewPager>

下拉列表

下拉列表導航是在 ActionBar 中顯示一個下拉列表 (Spinner), 就像一個菜單, 只顯示選中的菜單項對應的視圖, 如下圖所示:

將 ActionBar 設置為下拉列表導航時, 一般不顯示 Activity 自身的標題, 因此需要將 Activity 的 Label 標記為空字符串, 並且 Activity 需要實現接口 ActionBar.IOnNavigationListener , ListNavigationActivity 的部分實現代碼如下:

  1. [Activity (Label = "")] 
  2. public class ListNavigationActivity 
  3.         : FragmentActivity, ActionBar.IOnNavigationListener { 
  4.  
  5.     ListNavSectionsPagerAdapter _navSectionsPagerAdapter; 
  6.      
  7.     protected override void OnCreate(Bundle bundle) { 
  8.         base.OnCreate(bundle); 
  9.         /* 其他代碼省略 … */ 
  10.  
  11.         // 設置 ActionBar 
  12.         var actionBar = this.ActionBar; 
  13.         // 將 Home 設置為向上 
  14.         actionBar.SetDisplayHomeAsUpEnabled(true); 
  15.         // 設置 ActionBar 的導航模式為下拉列表 
  16.         actionBar.NavigationMode = ActionBarNavigationMode.List; 
  17.          
  18.         var titles = new string[this._navSectionsPagerAdapter.Count]; 
  19.         for (var i = 0; i < titles.Length; i++) { 
  20.             titles[i] = this._navSectionsPagerAdapter.GetPageTitle(i); 
  21.         } 
  22.         // 設置列表導航的回調參數 
  23.         actionBar.SetListNavigationCallbacks( 
  24.             new ArrayAdapter( 
  25.                 actionBar.ThemedContext, 
  26.                 Resource.Layout.ListNavigationActivityActionbarListItem, 
  27.                 Android.Resource.Id.Text1, 
  28.                 titles 
  29.             ), 
  30.             this 
  31.         ); 
  32.         // 設置 ViewPager 
  33.         this._viewPager = this.FindViewById<ViewPager>(Resource.Id.Pager); 
  34.         this._viewPager.Adapter = this._navSectionsPagerAdapter; 
  35.         // 當 ViewPager 的選中頁切換時, 同步 actionBar 的選中項。 
  36.         this._viewPager.PageSelected += delegate(object sender, ViewPager.PageSelectedEventArgs e) { 
  37.             actionBar.SetSelectedNavigationItem(e.P0); 
  38.         }; 
  39.     } 
  40.      
  41.     // ActionBar.IOnNavigationListener 
  42.     public bool OnNavigationItemSelected(int itemPosition, long itemId) { 
  43.         this._viewPager.CurrentItem = itemPosition; 
  44.         return true; 
  45.     } 

向上導航

所謂的向上導航, 就是在 Activity 的圖標上顯示一個向左的箭頭, 點擊圖標返回應用程序的上一級 Activity , 注意是上一級 Activity , 不是上一個 Activity , 關於向上與返回的區別, 可以看看 Android SDK 中的 Providing Ancestral and Temporal Navigation 一文, 將向上和返回講解的非常清楚, 在這裡只討論 Mono for Android 的實現方式。

要顯示向上導航的按鈕, 需要在 OnCreate 方法中對 ActionBar 做如下設置:

// 設置 ActionBar var actionBar = this.ActionBar; // 將 Home 按鈕顯示為向上, 提示用戶點擊這個按鈕可以返回應用程序的上一級。 actionBar.SetDisplayHomeAsUpEnabled(true);

同時還需要重寫 OnOptionsItemSelected 方法, 當用戶點擊 Home 按鈕時, 做相應的處理, 實現向上導航的代碼如下:

  1. public override bool OnOptionsItemSelected(Android.Views.IMenuItem item) { 
  2.     // 作為示例, 只處理用戶點擊 Home 按鈕的情況。 
  3.     if (item.ItemId == Android.Resource.Id.Home) { 
  4.         // 當 Home 按鈕被點擊時會調用到這裡 
  5.         // 創建啟動上級 Activity 的 Intent 
  6.         var upIntent = new Intent(this, typeof(MainActivity)); 
  7.         // 使用 Suport Package 中的 NavUtils 來正確處理向上導航 
  8.         if (NavUtils.ShouldUpRecreateTask(this, upIntent)) { 
  9.             // 上級 Activity 沒有起動過, 需要創建一個新的導航棧道 
  10.             TaskStackBuilder.Create(this) 
  11.                 // If there are ancestor activities, they should be added here. 
  12.                 .AddNextIntent(upIntent) 
  13.                 .StartActivities(); 
  14.             this.Finish(); 
  15.         } 
  16.         else { 
  17.             // 上級 Activity 已經創建過了, 直接導航就行。 
  18.             NavUtils.NavigateUpTo(this, upIntent); 
  19.         } 
  20.         return true; 
  21.     } 
  22.     return base.OnOptionsItemSelected(item); 

總結

Android 系統的導航與 iOS 相比復雜很多, 實現起來也相對麻煩一些, 好在有 Google 的 Support Package 已經多大部分操作提供了比較好的封裝, 還是比較容易掌握的。 文中的完整的源代碼已經提交的 Github 上, 地址是 https://github.com/beginor/MonoDroid/tree/master/EffectiveNavigation 。

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved