nowindxdw
11/2/2017 - 7:49 AM

Fragment 横竖屏切换问题

Fragment 横竖屏切换问题

发表于3年前(2012-09-03 19:07)   阅读(6100) | 评论(0) 5人收藏此文章, 我要收藏
赞0
让“码农”快速实现财务自由,通联量化实验室激发你的投资潜力!
android fragment 横竖屏切换

在默认情况下当发生横竖屏切换时,当前Activity中的fragment都会通过Fragment.instantiate重新生成,该方法将使用默认的构造函数来生成相应的Fragment,所以如果没有默认构造函数的话将会报错,例如:
?
1
2
3
4
5
6
7
8
9
10
classMyFragment {
publicMyFragment(inttitle, String message){
 
}
publicstaticfinalMyFragment newInstance(inttitle, String message)
{
    MyFragment f = newMyFragment (inttitle, String message);
    returnf;
}
}
这时候当横竖屏切换时由于MyFragment没有默认构造函数,将抛出java.lang.InstantiationException异常,正确的方式为使用Bundle来进行参数传递修改如下:
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
classMyFragment {
privatestaticfinalString TITLE="title",MESSAGE="message";
privateString title,message;
publicvoidonCreate(Bundle saveInstanceState){
    title = getArguments().getString(TITLE);
    message = getArguments().getString(MESSAGE);
}
publicstaticfinalMyFragment newInstance(inttitle, String message)
{
    MyFragment f = newMyFragment (inttitle, String message);
    Bundle bdl = newBundle(2);
    bdl.setString(TITLE,title);
    bdl.setString(MESSAGE,message);
    f.setArguments(bdl);
    returnf;
}
}
 通过这种方式创建的Fragment在横竖屏切换时通过getArguments依然能够获得之前设置的数据,其原理是在FragmentActivity切换时会调用onRetainNonConfigurationInstance方法将FragmentManager中管理的所有Fragment及其状态数据(其中就包括了这个设置的Bundle)保存在一个FragmentActivity.NonConfigurationInstances对象实例中,这样在新的FragmentActivity启动时在onCreate方法中可以使用Activity.getLastNonConfigurationInstance()方法来获取这个对象,然后通过FragmentManager.restoreAllState方法还原所有Fragment及其状态,需要注意的是在这种情况下可能会出现之前的Fragment没有detach而处于活动状态导致该Fragment的视图生成,可能会造成两个Fragment视图重叠的情况,在官方的Support.v4的例子中FragmentTabs.TabManager.addTab里有一段检查Fragment是否detach的代码正是用于解决这个问题。
通过上面的处理基本是没问题了,但因为默认情况下横竖屏切换后整个FragmentActivity会被销毁并重建,所有Fragment中的成员变量也会丢失,但所有的Fragment状态数据如上所述会被保留并还原,这个时候所有的视图都会重新创建。
解决方法一:在相应的Activity配置中加上android:configChanges="orientation|keyboardHidden"设置,这样切换时就不会销毁FragmentActivity,所有的Fragment的状态及视图也就会保持。


解决方法二:在使用FragmentTransaction.add()方法添加fragment时设置第三个tag参数,随后在还原时可通过FragmentManager.findFragmentByTag()方法找回还原的fragment.

Android 禁止屏幕旋转 & 旋转屏幕时保持Activity内容

 
1.在应用中固定屏幕方向。
 
     在AndroidManifest.xml的activity中加入:
           android:screenOrientation=”landscape”
     属性即可(landscape是横向,portrait是纵向)。
 
OK
 
 
2.随屏幕旋转时,不重新调用onCreate。
 
当将手机屏幕旋转时,系统会被强制重置启动onCreate方法。
 
1)修改AndroidManifest.xml
 
     在activity属性中加入:
          android:configChanges=”orientation|keyboardHidden”
android:configChanges,这个方法主要是负责列出清单,当清单上用户指定的设置改变时,Activity会自己处理这些变化。
orientation,屏幕界面旋转(可能是用户手动旋转的),【注意:如果你的开发API等级等于或高于13,你还需要设置screenSize,因为screenSize会在屏幕旋转时改变】
keyboardHidden,键盘辅助功能改变
2)在相对应的Activity中继承重写onConfigurationChanged方法,这个方法将会在我们的应用发生变化时,让我们能随心所谓地进行监听处理。
 
public void onConfigurationChanged(Configuration newConfig) {
        // TODO Auto-generated method stub
super.onConfigurationChanged(newConfig);
        if (newConfig.orientation==Configuration.ORIENTATION_LANDSCAPE) {
           // Nothing need to be done here
            
        } else {
           // Nothing need to be done here
        }
        
       
    }
还有界面设计方面的问题,Android手机大部分是HVGA、WVGA的分辨率,屏幕视觉上比较“狭长”。往往竖着看很合适的布局,当屏幕横向翻转以后 显示会变得很别扭。当屏幕由竖直方向改变为横向时,我们可以把界面中的控件由本来的垂直线性布局修改为横向线性布局,这样布局会更合理一些。我们可以自己 写一个布局类集成LinearLayout布局,通过覆盖onMeasure方法来实现这种自动布局。当屏幕的宽高发生改变时,系统会调用 onMeasure方法。通过这个方法,我们可以获得改变以后的宽高尺寸,从而来实现屏幕翻转的自动布局,主要代码如下:

Java代码 
 /**  
  * 屏幕改变时自动调用  
* @param widthMeasureSpec 改变后的宽度  
* @param heightMeasureSpec 改变后的高度  
*/  
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)   
 {   
     /*宽度*/  
     int screenWith = View.MeasureSpec.getSize(widthMeasureSpec);   
     /*高度*/  
     int screenHeight = View.MeasureSpec.getSize(heightMeasureSpec);   
        
     /*竖直布局*/  
     if (screenWith < screenHeight)   
     {   
            
         this.setOrientation(VERTICAL);   
         for (int i = 0; i < getChildCount(); i++)   
         {   
             View childView = getChildAt(i);   
             if (childView instanceof CakyCanvas)   
             {   
                 /*该控件占布局的2/5*/  
                 LayoutParams params = new LayoutParams(screenWith,   
                         screenHeight * 2/ 5  
                 updateViewLayout(childView, params);   
             }   
             else if (childView instanceof CakyExplainCanvas)   
             {   
                 /*该控件占布局的3/5*/  
                 LayoutParams params = new LayoutParams(screenWith,   
                         screenHeight * 3/ 5  
                 updateViewLayout(childView, params);   
             }   
         }   
     }   
     /*横向布局*/  
     else  
     {   
            
         this.setOrientation(HORIZONTAL);   
         for (int i = 0; i < getChildCount(); i++)   
         {   
             View childView = getChildAt(i);   
             if (childView instanceof CakyCanvas)   
             {   
                 LayoutParams params = new LayoutParams(   
                         screenWith * 2/ 5  
                         screenHeight);   
                 updateViewLayout(childView, params);   
             }   
             else if (childView instanceof CakyExplainCanvas)   
             {   
                 LayoutParams params = new LayoutParams(   
                         screenWith * 3/ 5  
                         screenHeight);   
                 updateViewLayout(childView, params);   
             }   
         }   
     }   
     super.onMeasure(widthMeasureSpec, heightMeasureSpec);   
 }