[转]Android平台Gallery2应用分析(三)---StateManager和AbstractGalleryActivity

StateManager

StateManager中有个Stack<StateEntry> mStack,类似于ActivityManager中的ActivityStack。用于控制相册界面的窗口堆栈管理,成员为StateEntry类。再看startState这个函数:

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. public void startState(Class<? extends ActivityState> klass,  
  2.         Bundle data) {  
  3.     Log.v(TAG, "startState " + klass);  
  4.     ActivityState state = null;  
  5.     try {  
  6.         // 用窗口类创建一个ActivityState实例  
  7.         state = klass.newInstance();  
  8.     } catch (Exception e) {  
  9.         throw new AssertionError(e);  
  10.     }  
  11.     // 堆栈非空  
  12.     if (!mStack.isEmpty()) {  
  13.          // 获取栈顶ActivityState  
  14.         ActivityState top = getTopState();  
  15.         top.transitionOnNextPause(top.getClass(), klass,  
  16.                 StateTransitionAnimation.Transition.Incoming);  
  17.         // 调用栈顶ActivityState的onPause  
  18.         if (mIsResumed) top.onPause();  
  19.     }  
  20.     // 初始化当前的ActivityState,这个和startActivity非常相似  
  21.     state.initialize(mActivity, data);  
  22.     // 初始化后入栈  
  23.     mStack.push(new StateEntry(data, state));  
  24.   
  25.     Log.d(TAG, "startState: startState->onCreate");  
  26.     // 调用新ActivityState实例的onCreate,这个和startActivity的流程又好相似  
  27.     state.onCreate(data, null);  
  28.     Log.d(TAG, "startState: startState->resume");  
  29.     // 调用新ActivityState实例的onResume  
  30.     if (mIsResumed) state.resume();  

可以说这个函数和启动应用activity的流程非常相似,只是简化了流程而已。另外,startStateForResult也和startActivityForResult类似。

 

 

AbstractGalleryActivity

在介绍AlbumSetPage、AlbumPage、PhotoPage等页面前,必须先介绍AbstractGalleryActivity,因为以上三个页面的父类ActivityState中有成员变量AbstractGalleryActivity mActivity。相册的Activity实际上只有一个,可以说就是这个mActivity,通过StateManager和DataManager来控制显示不同的页面。
继承自Activity,其中有几个重要成员:GLRootView mGLRootView,StateManager mStateManager,DataManager,GalleryActionBar mActionBar,OrientationManager mOrientationManager。
其中GalleryActionBar就是相册各个界面顶部的ActionBar控件。按照Activity的生命周期,在onCreate中创建OrientationManager对象赋予mOrientationManager,调用toggleStatusBarByOrientation在屏幕横竖屏时对window的全屏标记做增加和清除。

 

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. // Shows status bar in portrait view, hide in landscape view  
  2. private void toggleStatusBarByOrientation() {  
  3.     if (mDisableToggleStatusBar) return;  
  4.   
  5.     Window win = getWindow();  
  6.     if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_PORTRAIT) {  
  7.         win.clearFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);  
  8.     } else {  
  9.         win.addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);  
  10.     }  
  11. }  

在setContentView()中加载GLRootView。在onStart()中注册AlertDialog的onClick处理。在onResume(),onPause()中都会分别对StateManager,DataManager做对应的onResume, onPause操作。例如:

 

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. @Override  
  2. protected void onResume() {  
  3.     super.onResume();  
  4.     mGLRootView.lockRenderThread();  
  5.     try {  
  6.         getStateManager().resume();  
  7.         getDataManager().resume();  
  8.     } finally {  
  9.         mGLRootView.unlockRenderThread();  
  10.     }  
  11.     mGLRootView.onResume();  
  12.     mOrientationManager.resume();  
  13. }  

其中 mGLRootView会调用 lockRenderThread(),在执行完xxx.resume()后会再调用unlockRenderThread()。下面是GLRootView的lockXXX()和unlockXXX()函数。

 

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. @Override  
  2. public void lockRenderThread() {  
  3.     mRenderLock.lock();  
  4. }  
  5.   
  6. @Override  
  7. public void unlockRenderThread() {  
  8.     mRenderLock.unlock();  
  9. }  

其中 mRenderLock是ReentrantLock对象。那么ReentrantLock(可重入锁)的作用是什么?
因为Gallery的每个页面AlbumSet、AlbumPage、PhotoPage、PhotoView都是同一个Activity,界面的控制是通过StateManager以及DataManager,切换进入每个状态页面,都会获取不同的GLRootView,所以必须保证GLRootView是锁定情况下,才能对StateManager以及DataManager做处理。

 

欢迎转载和技术交流,转载请帮忙注明出处,http://blog.csdn.net/discovery_by_joseph,谢谢!

posted @ 2017-02-16 16:37  得即高歌失即休  阅读(229)  评论(0)    收藏  举报