web-dev-qa-db-ja.com

リストビューのEdittext android

Editextとtextviewを備えたリストビューがあります。

Edittextをタッチすると、edittextがフォーカスを失いました!

Android:windowSoftInputMode="adjustPan"(AndroidManifest.xml)を設定してこの問題を解決しました。今私はeditextよりedittextにタッチしてフォーカスを取得しますが、アプリケーションラベルとリストビューのいくつかのrawが消えます(上部)。

ユーザーがアプリケーションラベルやリストビューの生を失うことなくedittextにタッチしたときにフォーカスを取得したいと思います。

私が実装したコード:

以下のコーディングは、ユーザーがedittextにタッチしたときにフォーカスを取得しますが、アプリケーションキーとソフトキーパッドがポップアップすると、rawのリストビューが消えます。

1)AndroidManifest.xml

<application Android:icon="@drawable/icon" Android:label="@string/app_name">
        <activity Android:name=".MyListViewDemoActivity"
                  Android:label="@string/app_name"
                  Android:windowSoftInputMode="adjustPan"
                  >
            <intent-filter>
                <action Android:name="Android.intent.action.MAIN" />
                <category Android:name="Android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

</application>

2)raw_layout.xml

<LinearLayout
  xmlns:Android="http://schemas.Android.com/apk/res/Android"
  Android:orientation="vertical"
  Android:layout_width="fill_parent"
  Android:layout_height="fill_parent">
  <EditText Android:id="@+id/mEditText"
  Android:layout_width="fill_parent"
  Android:layout_height="wrap_content"
  />  
</LinearLayout>

3)main.xml

<LinearLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
    Android:orientation="vertical"
    Android:layout_width="fill_parent"
    Android:layout_height="fill_parent"
    >
<ListView Android:id="@+id/mListView"
Android:layout_width="fill_parent"
Android:layout_height="wrap_content"
/>
</LinearLayout>

4)MyListViewDemoActivity

public class MyListViewDemoActivity extends Activity {
    private ListView mListView;
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        mListView=(ListView)findViewById(R.id.mListView);
        mListView.setAdapter(new MyAdapter(this));
    }
}

class MyAdapter extends BaseAdapter {

    private Activity mContext;
    private String character[]={"a","b","c","d","e","f","g","h","i","j"};
    public MyAdapter(Activity context)
    {
        mContext=context;
    }

    @Override
    public int getCount() {
        // TODO Auto-generated method stub
        return character.length;
    }

    @Override
    public Object getItem(int position) {
        // TODO Auto-generated method stub
        return null;
    }

    @Override
    public long getItemId(int position) {
        // TODO Auto-generated method stub
        return 0;
    }
private class Holder
{
    EditText mEditText;
}
    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        // TODO Auto-generated method stub
        final Holder holder;
        if (convertView == null) {
            holder = new Holder();
            LayoutInflater inflater =mContext.getLayoutInflater();
            convertView = inflater.inflate(R.layout.raw_layout, null);
            holder.mEditText = (EditText) convertView
                    .findViewById(R.id.mEditText);
            convertView.setTag(holder);
        } else {
            holder = (Holder) convertView.getTag();
        }
        holder.mEditText.setText(character[position]);
        holder.mEditText.setOnFocusChangeListener(new OnFocusChangeListener() {

            @Override
            public void onFocusChange(View v, boolean hasFocus) {
                // TODO Auto-generated method stub
                if (!hasFocus){
                    final EditText etxt = (EditText) v;
                    holder.mEditText.setText(etxt.getText().toString());
                }

            }
        });
        return convertView;
    }

}
21
user1206620

私も同じ問題を抱えていました。私の数字キーボードは、qwertyキーボードに置き換わる前に一時的に表示され、EditTextがフォーカスを失います。

問題は、キーボードが表示されるとEditTextがフォーカスを失うことです。これを防ぐには、適切なアクティビティ(1つまたは複数)のAndroidManifest.xmlに以下を追加します。

Android:windowSoftInputMode="adjustPan"

Android ドキュメント を参照してください:

入力メソッドが画面に表示されると、アプリのUIに使用できるスペースの量が減少します。システムは、UIの表示部分をどのように調整するかを決定しますが、正しく調整されない場合があります。アプリの最適な動作を確保するには、残りのスペースにシステムがUIを表示する方法を指定する必要があります。

アクティビティで優先する処理を宣言するには、マニフェストのAndroid:windowSoftInputMode要素で<activity>属性を使用し、「調整」値の1つを指定します。

たとえば、システムがレイア​​ウトを利用可能なスペースにサイズ変更することを保証するには(おそらくスクロールが必要な場合でも)すべてのレイアウトコンテンツにアクセスできることを保証するには、"adjustResize"を使用します。

7
Cyrille

コードを確認せずに、問題の解決策を提案するにはどうすればよいでしょうか。したがって、質問するときはいつでも可能なコードを投稿する練習を続けてください。

ただし、ここでは実装のためのチュートリアルを1つ見つけました ListView内のAndroid Focusable EditText 。例を見て、自分のやり方で実装するか、問題の解決策を見つけてください。

4
Paresh Mayani

最近、この「ListViewのEditTextをアイテムとして配置する」問題を解決しました。私は英語があまり上手ではありません。だから、はっきりと説明していないことがあったら教えてください。

ListViewを垂直方向にスクロールできることがわかっているので、EditTextを項目としてListViewに配置します。

最初:追加

Android:windowSoftInputMode = "adjustResize"

アクティビティノードのAndroidManifest.xml内。

2番目:EditText状態を制御するモデルデータソースとしてpojoを作成します

Line.Java

public class Line{
    int num;
    String text;
    boolean focus;

    get set method and so on...
}

3番目:EditTextをListViewに適合させるアダプタを作成します。

アイテムitem_line.xml:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout
    xmlns:Android="http://schemas.Android.com/apk/res/Android"
    Android:id="@+id/frameLayout"
    Android:layout_width="match_parent"
    Android:layout_height="80dp">

    <EditText
        Android:id="@+id/etLine"
        Android:focusable="true"
        Android:layout_width="match_parent"
        Android:layout_height="wrap_content"
        Android:layout_gravity="bottom"/>
</FrameLayout>

アダプター:

@Override
public View getView(final int position, View convertView, final ViewGroup parent) {
    final ViewHolder holder;
    if (convertView == null) {
        holder = new ViewHolder();
        convertView = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_line, parent, false);
        holder.etLine = (EditText) convertView.findViewById(R.id.etLine);
        convertView.setTag(holder);
    } else {
        holder = (ViewHolder) convertView.getTag();
    }

    final Line line = lines.get(position);

    // step 1: remove Android.text.TextWatcher added in step 5 to make sure Android.text.TextWatcher 
    //         don't trigger in step 2;
    // why?
    // 
    // note: When an object of a type is attached to an Editable, 
    //       TextWatcher's methods will be called when the EidtText's text is changed.
    //       
    //       EditText use a ArrayList<TextWatcher> type object to store the listener, so we must
    //       make sure there's only one TextWatcher object in this list;
    // 
    // Avoid triggering TextWatcher's method in step 2 we remove it at first time.
    // 
    if (holder.etLine.getTag() instanceof TextWatcher) {
        holder.etLine.removeTextChangedListener((TextWatcher) (holder.etLine.getTag()));
    }

    // step 2: set text and focus after remove Android.text.TextWatcher(step 1);
    holder.etLine.setHint(position + ".");

    // set text
    if (TextUtils.isEmpty(line.getText())) {
        holder.etLine.setTextKeepState("");
    } else {
        holder.etLine.setTextKeepState(line.getText());
    }

    // set focus status
    // why?
    //
    // note: ListView has a very elegant recycle algorithm. So views in ListView is not reliable.
    //       Especially in this case, EditText is an item of ListView. Software input window may cause
    //       ListView relayout leading adapter's getView() invoke many times.
    //       Above all if we change EditText's focus state directly in EditText level(not in Adapter). 
    //       The focus state may be messed up when the particularly view reused in other position. 
    //       
    //       So using data source control View's state is the core to deal with this problem.   
    if (line.isFocus()) {
        holder.etLine.requestFocus();
    } else {
        holder.etLine.clearFocus();
    }

    // step 3: set an OnTouchListener to EditText to update focus status indicator in data source
    // why?
    // 
    // in step 2, we know we must control view state through data source. We use OnTouchListener
    // to watch the state change and update the data source when user move up fingers(ACTION_UP).
    // We don't want to consume the touch event, simply return false in method onTouch().
    holder.etLine.setOnTouchListener(new View.OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent event) {
            if (event.getAction() == MotionEvent.ACTION_UP) {
                check(position);
            }
            return false;
        }
    });

    // step 4: set TextWatcher to EditText to listen text changes in EditText to updating the text in data source
    // why?
    // 
    // again, use data source to control view state.
    // When user edit the text in one EditText item and scroll the ListView. The particularly EditText item will be
    // reuse in adapter's getView(), this may lead text messed up in ListView.
    // How to deal with this problem?
    // Easy! We update the text in data source at the same time when user is editing. TextWatcher is the best way to
    // do this.
    final TextWatcher watcher = new SimpeTextWather() {

        @Override
        public void afterTextChanged(Editable s) {
            if (TextUtils.isEmpty(s)) {
                line.setText(null);
            } else {
                line.setText(String.valueOf(s));
            }
        }
    };
    holder.etLine.addTextChangedListener(watcher);

    // step 5: Set watcher as a tag of EditText.
    // so we can remove the same object which was setted to EditText in step 4;
    // Make sure only one callback is attached to EditText
    holder.etLine.setTag(watcher);

    return convertView;
}

/**
 * change focus status in data source
 */
private void check(int position) {
    for (Line l : lines) {
        l.setFocus(false);
    }
    lines.get(position).setFocus(true);
}

static class ViewHolder {
    EditText etLine;
}

全部終わった!

詳細については、私のgithubを参照してください。

デモ: https://github.com/Aspsine/EditTextInListView

3
Aspsine

ListViewビューを再作成し、ScrollView内でLinearLayoutを使用してから、コードでrunOnUiThreadを使用して、このような他のスレッドでビューを埋めます

public void fillDataTask(Context context, final LinearLayout listView) {

        getActivity().runOnUiThread(new Runnable() {
            @Override
            public void run() {

                fillView(context, listView);
            }
        });
    }



    private void fillView(Context context, LinearLayout listView) {
            MyAdapter adapter = new MyAdapter(context);
            final int adapterCount = adapter.getCount();
            for (int i = 0; i < adapterCount; i++) {
                View item = adapter.getView(i, null, null);
                listView.addView(item);
            }

    }
0