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;
}
}
私も同じ問題を抱えていました。私の数字キーボードは、qwertyキーボードに置き換わる前に一時的に表示され、EditTextがフォーカスを失います。
問題は、キーボードが表示されるとEditTextがフォーカスを失うことです。これを防ぐには、適切なアクティビティ(1つまたは複数)のAndroidManifest.xmlに以下を追加します。
Android:windowSoftInputMode="adjustPan"
Android ドキュメント を参照してください:
入力メソッドが画面に表示されると、アプリのUIに使用できるスペースの量が減少します。システムは、UIの表示部分をどのように調整するかを決定しますが、正しく調整されない場合があります。アプリの最適な動作を確保するには、残りのスペースにシステムがUIを表示する方法を指定する必要があります。
アクティビティで優先する処理を宣言するには、マニフェストの
Android:windowSoftInputMode
要素で<activity>
属性を使用し、「調整」値の1つを指定します。たとえば、システムがレイアウトを利用可能なスペースにサイズ変更することを保証するには(おそらくスクロールが必要な場合でも)すべてのレイアウトコンテンツにアクセスできることを保証するには、
"adjustResize"
を使用します。
コードを確認せずに、問題の解決策を提案するにはどうすればよいでしょうか。したがって、質問するときはいつでも可能なコードを投稿する練習を続けてください。
ただし、ここでは実装のためのチュートリアルを1つ見つけました ListView内のAndroid Focusable EditText 。例を見て、自分のやり方で実装するか、問題の解決策を見つけてください。
最近、この「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を参照してください。
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);
}
}