Android - 使用默认文本选择CAB找到的潜在错误

时间:2014-04-10 16:39:18

标签: android android-fragments

我开发了一个应用程序,它使用一个视图并将两个片段中的一个加载到其中,具体取决于方向,即纵向/横向。

两个片段都具有相同的UI TextView / EditText / Button组件。

当应用程序处于纵向模式并且按下文本字段很长(ish)时,将使用选择范围标记选择文本,并使用默认文本选择CAB替换应用程序栏。

一切如预期。

但是,当我转换为横向模式时,虽然我的应用程序仍然按预期工作,但默认文本选择行为不起作用。我仍然可以通过长按字段来选择文本但是虽然文本突出显示,但没有选择范围标记,并且不显示默认文本选择CAB。我对所选文字无能为力。

当我回到纵向模式时,一切都按预期工作了。

我的目标是最低sdk为16,并以19为目标。

这是一个已知的错误还是我从纵向翻转到风景时错过了一步?

编辑:

我在沙盒中做了一些进一步的调查,发现了什么似乎是一个bug,虽然不确定它是在Android还是手机本身。似乎在纵向和横向模式下都会触发此错误。

我已在以下小应用程序中隔离了该问题。

的AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="org.example.foo"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="16"
        android:targetSdkVersion="19" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="org.example.foo.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

的strings.xml

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

    <string name="app_name">Foo</string>
    <string name="action_settings">Settings</string>
    <string name="select_me">Select Me</string>

</resources>

ActivityMain.java

package org.example.foo;

import android.app.Activity;
import android.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup;

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main);

        if (savedInstanceState == null) {
            getFragmentManager().beginTransaction()
                    .add(R.id.container, new PlaceholderFragment())
                    .commit();
        }
    }


    @Override
    public boolean onCreateOptionsMenu(Menu menu) {

        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();
        if (id == R.id.action_settings) {
            return true;
        }
        return super.onOptionsItemSelected(item);
    }

    public static class PlaceholderFragment extends Fragment
    {
        public PlaceholderFragment()
        {
        }

        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
        {
            View rootView = inflater.inflate(R.layout.fragment_main, container, false);

            return rootView;
        }
    }
}

activity_main.xml中

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="org.example.foo.MainActivity"
    tools:ignore="MergeRootFrame" />

fragment_main.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:orientation="vertical"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="org.example.foo.PlaceholderFragment" >

    <TextView
        android:id="@+id/tv"
        android:textIsSelectable="true"
        android:layout_width="200dp" 
        android:layout_height="50dp"
        android:text="@string/select_me"
/>

</LinearLayout>

所以看起来layout_width和layout_height属性的组合正在推动默认的文本选择CAB功能。

使用这些参数,我可以运行应用程序并以纵向和横向模式选择文本,并按预期显示默认文本选择CAB。

如果我将layout_width或layout_height增加一个或多个,则会触发该错误。要么增加仍然允许我选择一个单词,但它会停止默认文本选择CAB的操作,包括文本选择范围标记。

进一步的调查表明,该bug可能对当前的Android状态产生一些永久性影响,导致内部内存泄漏或类似情况。

考虑一下:

1)使用我在此处列出的属性运行应用程序,并且应用程序按预期工作

2)将layout_width或layout_height属性中的任何一个增加1,重建应用程序并且它不再按预期运行,它现在显示错误

3)重置(2)中的调整属性,所以它现在处于原始状态(1),重建应用程序,它仍然显示错误

4)重新启动手机,(3)中发布的应用程序现在按预期工作

此时我无法想到其他任何事情,除非将此报告为Android的错误,然后尝试重新设计我的应用以解决此问题。

最后一个想法是,这可能是与硬件相关的错误,所以也许有人可以在不同的设备上为我测试这个?

我目前在运行Android 4.1.1的华为Y300-100手机上遇到此错误

修改

我已向谷歌报告此错误,发行号为68435

2 个答案:

答案 0 :(得分:0)

我认为问题不仅仅是一些维度参数。以下工作没有任何问题:

import android.app.Activity;
import android.graphics.Color;
import android.os.Bundle;
import android.util.TypedValue;
import android.view.Gravity;
import android.widget.FrameLayout;
import android.widget.TextView;

public class MyActivity extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        FrameLayout holder = new FrameLayout(this);

        int width = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP,
                201,
                getResources().getDisplayMetrics());
        int hieght = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP,
                51,
                getResources().getDisplayMetrics());

        FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(width,hieght);
        params.gravity = Gravity.CENTER;

        TextView textView = new TextView(this);
        textView.setTextIsSelectable(true);
        textView.setText("Select Me");
        textView.setBackgroundColor(Color.GRAY);

        holder.addView(textView,params);

        setContentView(holder);
    }
}

答案 1 :(得分:0)

AndroidManifest.xml中,删除orientation中的android:configChanges="keyboardHidden|screenSize|orientation",以避免重新计算光标的偏移量