问题描述
我正在使用此库: : 来检测何时打开或关闭键盘,这取决于将android:windowSoftInputMode="adjustResize"
输入到Android清单中。
该库可以完美地检测软键盘的打开和关闭事件,但是由于使用了AdjustResize参数,我的内容无法显示。
Java:
KeyboardVisibilityEvent.setEventListener( AddActivity.this, new KeyboardVisibilityEventListener() { @Override public void onVisibilityChanged(boolean isOpen) { // some code depending on keyboard visiblity status if (noteEditText.isFocused()) { if (isOpen) { Log.d("KB", "Keyboard is open"); noteEditText.setLines(12); noteEditText.setCursorVisible(true); } else { Log.d("KB", "Keyboard is closed"); noteEditText.setLines(50); noteEditText.setCursorVisible(false); } } } }); noteEditText.setOnFocusChangeListener(new View.OnFocusChangeListener() { @Override public void onFocusChange(View v, boolean hasFocus) { Log.d("KB", "onFocusChange"); if (firstStart) { noteEditText.setLines(12); noteEditText.setCursorVisible(true); firstStart = false; } } });
XML:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/add_record" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" android:padding="20dp" android:windowSoftInputMode="stateHidden"> <EditText android:id="@+id/title_edittext" android:layout_width="match_parent" android:layout_height="wrap_content" android:ems="10" android:hint="@string/enter_title" android:inputType="textCapSentences" android:textColor="@color/fontPrimary" android:theme="@style/EditTextCustomCursor"> <requestFocus /> </EditText> <EditText android:id="@+id/note_edittext" android:layout_width="match_parent" android:layout_height="wrap_content" android:background="@null" android:ellipsize="end" android:gravity="top|left" android:hint="@string/enter_note" android:inputType="textCapSentences|textMultiLine" android:lines="50" android:maxLines="20" android:minLines="5" android:paddingLeft="5dp" android:scrollHorizontally="false" android:scrollbars="vertical" android:textColor="@color/fontPrimary" android:theme="@style/EditTextCustomCursor" />
因此,通过调整第二个EditText的行可以很好地工作,因此我在键盘上方键入内容,但是当我关闭键盘时,滚动到该EditText的底部并单击底部,将EditText光标放在我单击的位置,但是然后将第一个EditText和Support ActionBar推出视线,并在底部(如下面的图像(图2所示)中所示)处留出较大的间隙,因为选择了“ F”,即底部EditText)。
所需效果(均处于正确位置)
实际效果(Support ActionBar和顶部EditText移出了视图)
我也尝试过使用'adjustNothing'并执行以下操作,但这似乎也不起作用,因为EditText的高度在行数更改之前不会更改,并且仅在知道是否更改行数时才更改键盘是否打开或关闭。
private void setupListeners() {
final View activityRootView = getWindow().getDecorView().findViewById(android.R.id.content);
activityRootView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
View mainView = (LinearLayout) findViewById(R.id.add_record);
int heightDiff = mainView.getHeight() - noteEditText.getHeight();
Log.d("KB", "HeightDiff: " + heightDiff);
if (heightDiff > 1000 || keyboardShown) { // 99% of the time the height diff will be due to a keyboard.
Log.d("KB", "Keyboard is open");
if (isKeyboardVisible) {
noteEditText.setLines(12);
noteEditText.setCursorVisible(true);
noteEditText.requestLayout();
isKeyboardVisible = false;
}
} else {
Log.d("KB", "Keyboard is closed");
if (!isKeyboardVisible) {
noteEditText.setLines(50);
noteEditText.setCursorVisible(false);
noteEditText.requestLayout();
isKeyboardVisible = true;
}
}
}
});
noteEditText.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
numTimesClicked++;
Log.d("KB", "onClick: " + numTimesClicked);
if (clicked) {
// Run function
Log.d("KB", "clicked");
InputMethodManager imm = (InputMethodManager) AddActivity.this.getSystemService(Context.INPUT_METHOD_SERVICE);
if (imm.isAcceptingText()) {
Log.d("KB", "Software Keyboard was shown");
isKeyboardVisible = true;
keyboardShown = true;
} else {
Log.d("KB", "Software Keyboard was not shown");
isKeyboardVisible = false;
keyboardShown = false;
}
} else {
Log.d("KB", "scroll");
clicked = true;
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
clicked = false;
}
}, 3 * 1000);
}
}
});
noteEditText.setOnKeyListener(new View.OnKeyListener() {
@Override
public boolean onKey(View v, int keyCode, KeyEvent event) {
if (event.getKeyCode() == KeyEvent.KEYCODE_BACK) {
Log.d("KB", "closeKeyboard");
noteEditText.setLines(50);
noteEditText.setCursorVisible(false);
noteEditText.requestLayout();
isKeyboardVisible = false;
}
return false;
}
});
因此,我将如何实现调整EditText行(打开和关闭键盘时)并且不将其他内容移出视线的预期效果? 那么,当出现软键盘时,Support ActionBar和第一个EditText始终保持在同一位置,并且仅调整第二个EditText吗?
1楼
我不确定,但是我相信如果将布局内容保留在ScrollView
并且每次用户输入新行或新单词时,都可以设置scrollView.scrollTo(0,0)
。
另外,您可以使用ViewTreeObserver
代替此库,请参考
另外,您可以使用onFocusChangeListener()
进行管理,这是更好的做法。
2楼
我现在设法解决了这个问题(不是最好的解决方案),但是它确实起作用。 现在,无论选择哪一行,顶部的EditText和Support ActionBar都不会从视图中移出,而第二个EditText会自动调整其大小。
表现:
<activity android:theme="@style/AppTheme"
android:name=".activities.AddActivity"
android:label="@string/add_record"
android:windowSoftInputMode="stateVisible|adjustResize"
android:parentActivityName=".MainActivity"
android:excludeFromRecents="true"/>
<activity android:theme="@style/AppTheme"
android:name=".activities.ModifyActivity"
android:label="@string/modify_record"
android:windowSoftInputMode="stateAlwaysHidden|adjustResize"
android:parentActivityName=".MainActivity"
android:excludeFromRecents="true"/>
Java:
private void setupListeners() {
final LinearLayout linearLayout = (LinearLayout) findViewById(R.id.add_record);
if (linearLayout != null) {
linearLayout.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Log.d(TAG, "Clicking ll");
noteEditText.requestFocus();
InputMethodManager inputMethodManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
inputMethodManager.toggleSoftInputFromWindow(linearLayout.getApplicationWindowToken(),
InputMethodManager.SHOW_FORCED, 0);
}
});
}
KeyboardVisibilityEvent.setEventListener(
AddActivity.this,
new KeyboardVisibilityEventListener() {
@Override
public void onVisibilityChanged(boolean isOpen) {
// some code depending on keyboard visiblity status
Log.d(TAG, "Keyboard visibility changed");
int currentLine = getCurrentCursorLine(noteEditText);
lineCount = noteEditText.getLineCount();
if (isOpen && keyboardActuallyOpen) {
Log.d(TAG, "Keyboard is open");
//keyboardActuallyClosed = false;
/*
scrollView.fullScroll(View.FOCUS_UP);
noteEditText.requestFocus();
*/
if (currentLine < 25) {
getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_ADJUST_NOTHING);
scrollView.scrollTo(0, 0);
noteEditText.setMinLines(12);
noteEditText.setLines(12);
noteEditText.setMaxLines(12);
scrollView.scrollTo(0, 0);
noteEditText.setVerticalScrollBarEnabled(true);
scrollView.setVerticalScrollBarEnabled(false);
noteEditText.setCursorVisible(true);
scrollView.requestLayout();
noteEditText.requestLayout();
} else {
getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE);
scrollView.scrollTo(0, 0);
noteEditText.setMinLines(12);
noteEditText.setLines(12);
noteEditText.setMaxLines(12);
scrollView.scrollTo(0, 0);
noteEditText.setVerticalScrollBarEnabled(true);
scrollView.setVerticalScrollBarEnabled(false);
noteEditText.setCursorVisible(true);
}
} else {
if (!keyboardActuallyOpen) {
lineCount = noteEditText.getLineCount();
Log.d(TAG, "Keyboard is closed: " + lineCount);
noteEditText.setVerticalScrollBarEnabled(false);
scrollView.setVerticalScrollBarEnabled(true);
noteEditText.setCursorVisible(false);
noteEditText.setMinLines(lineCount);
noteEditText.setLines(lineCount);
noteEditText.setMaxLines(lineCount);
scrollView.scrollTo(0, 0);
keyboardActuallyOpen = false;
//scrollView.requestFocus();
//setAdjustResize(1);
LinearLayout mainLayout = (LinearLayout) findViewById(R.id.add_record);
if (mainLayout != null) {
mainLayout.requestFocus();
}
}
}
}
});
noteEditText.setOnFocusChangeListener(new View.OnFocusChangeListener() {
@Override
public void onFocusChange(View v, boolean hasFocus) {
Log.d(TAG, "onFocusChange");
if (firstStart) {
scrollView.scrollTo(0, 0);
noteEditText.setLines(12);
noteEditText.setCursorVisible(true);
firstStart = false;
}
if (hasFocus) {
Log.d(TAG, "Has Focus");
keyboardActuallyOpen = true;
} else {
Log.d(TAG, "Lost focus");
keyboardActuallyOpen = false;
setAdjustResize(2);
}
}
});
noteEditText.setOnKeyListener(new View.OnKeyListener() {
@Override
public boolean onKey(View v, int keyCode, KeyEvent event) {
if (event.getKeyCode() == KeyEvent.KEYCODE_BACK) {
Log.d(TAG, "closeKeyboard");
keyboardActuallyOpen = false;
noteEditText.setMinLines(lineCount);
noteEditText.setLines(lineCount);
noteEditText.setMaxLines(lineCount);
noteEditText.setVerticalScrollBarEnabled(false);
noteEditText.setCursorVisible(false);
noteEditText.requestLayout();
scrollView.scrollTo(0, 0);
scrollView.setVerticalScrollBarEnabled(true);
scrollView.requestLayout();
noteEditText.clearFocus();
}
return false;
}
});
}
Java(按下键盘关闭按钮的子类EditText):
public class ExtendedEditText extends EditText {
public ExtendedEditText(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
public ExtendedEditText(Context context, AttributeSet attrs) {
super(context, attrs);
}
public ExtendedEditText(Context context) {
super(context);
}
@Override
public boolean onKeyPreIme(int keyCode, KeyEvent event) {
if (event.getKeyCode() == KeyEvent.KEYCODE_BACK) {
dispatchKeyEvent(event);
return false;
}
return super.onKeyPreIme(keyCode, event);
}
}
XML格式
<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/add_scrollView"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:descendantFocusability="beforeDescendants"
android:focusableInTouchMode="true"
android:fillViewport="true">
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/add_record"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:padding="20dp"
android:focusable="true"
android:focusableInTouchMode="true">
<EditText
android:id="@+id/title_edittext"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:ems="10"
android:hint="@string/enter_title"
android:inputType="textCapSentences"
android:textColor="@color/fontPrimary"
android:theme="@style/EditTextCustomCursor">
</EditText>
<com.securenotes.utils.ExtendedEditText
android:id="@+id/note_edittext"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@null"
android:ellipsize="end"
android:gravity="top|left"
android:hint="@string/enter_note"
android:inputType="textCapSentences|textMultiLine"
android:lines="50"
android:maxLines="20"
android:minLines="5"
android:paddingLeft="5dp"
android:scrollHorizontally="false"
android:scrollbars="vertical"
android:textColor="@color/fontPrimary"
android:theme="@style/EditTextCustomCursor" />
</LinearLayout>
</ScrollView>