|
|
|
@ -14,204 +14,269 @@
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
package net.micode.notes.ui;
|
|
|
|
|
package net.micode.notes.ui;
|
|
|
|
|
|
|
|
|
|
import android.content.Context;
|
|
|
|
|
import android.graphics.Rect;
|
|
|
|
|
import android.text.Layout;
|
|
|
|
|
import android.text.Selection;
|
|
|
|
|
import android.text.Spanned;
|
|
|
|
|
import android.text.TextUtils;
|
|
|
|
|
import android.text.style.URLSpan;
|
|
|
|
|
import android.util.AttributeSet;
|
|
|
|
|
import android.util.Log;
|
|
|
|
|
import android.view.ContextMenu;
|
|
|
|
|
import android.view.KeyEvent;
|
|
|
|
|
import android.view.MenuItem;
|
|
|
|
|
import android.view.MenuItem.OnMenuItemClickListener;
|
|
|
|
|
import android.view.MotionEvent;
|
|
|
|
|
import android.widget.EditText;
|
|
|
|
|
|
|
|
|
|
import net.micode.notes.R;
|
|
|
|
|
|
|
|
|
|
import java.util.HashMap;
|
|
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
|
|
public class NoteEditText extends EditText {
|
|
|
|
|
private static final String TAG = "NoteEditText";
|
|
|
|
|
private int mIndex;
|
|
|
|
|
private int mSelectionStartBeforeDelete;
|
|
|
|
|
|
|
|
|
|
private static final String SCHEME_TEL = "tel:" ;
|
|
|
|
|
private static final String SCHEME_HTTP = "http:" ;
|
|
|
|
|
private static final String SCHEME_EMAIL = "mailto:" ;
|
|
|
|
|
|
|
|
|
|
private static final Map<String, Integer> sSchemaActionResMap = new HashMap<String, Integer>();
|
|
|
|
|
static {
|
|
|
|
|
sSchemaActionResMap.put(SCHEME_TEL, R.string.note_link_tel);
|
|
|
|
|
sSchemaActionResMap.put(SCHEME_HTTP, R.string.note_link_web);
|
|
|
|
|
sSchemaActionResMap.put(SCHEME_EMAIL, R.string.note_link_email);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Call by the {@link NoteEditActivity} to delete or add edit text
|
|
|
|
|
*/
|
|
|
|
|
public interface OnTextViewChangeListener {
|
|
|
|
|
/**
|
|
|
|
|
* Delete current edit text when {@link KeyEvent#KEYCODE_DEL} happens
|
|
|
|
|
* and the text is null
|
|
|
|
|
*/
|
|
|
|
|
void onEditTextDelete(int index, String text);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Add edit text after current edit text when {@link KeyEvent#KEYCODE_ENTER}
|
|
|
|
|
* happen
|
|
|
|
|
*/
|
|
|
|
|
void onEditTextEnter(int index, String text);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Hide or show item option when text change
|
|
|
|
|
*/
|
|
|
|
|
void onTextChange(int index, boolean hasText);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private OnTextViewChangeListener mOnTextViewChangeListener;
|
|
|
|
|
|
|
|
|
|
public NoteEditText(Context context) {
|
|
|
|
|
super(context, null);
|
|
|
|
|
mIndex = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void setIndex(int index) {
|
|
|
|
|
mIndex = index;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void setOnTextViewChangeListener(OnTextViewChangeListener listener) {
|
|
|
|
|
mOnTextViewChangeListener = listener;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public NoteEditText(Context context, AttributeSet attrs) {
|
|
|
|
|
super(context, attrs, android.R.attr.editTextStyle);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public NoteEditText(Context context, AttributeSet attrs, int defStyle) {
|
|
|
|
|
super(context, attrs, defStyle);
|
|
|
|
|
// TODO Auto-generated constructor stub
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onTouchEvent(MotionEvent event) {
|
|
|
|
|
switch (event.getAction()) {
|
|
|
|
|
case MotionEvent.ACTION_DOWN:
|
|
|
|
|
|
|
|
|
|
int x = (int) event.getX();
|
|
|
|
|
int y = (int) event.getY();
|
|
|
|
|
x -= getTotalPaddingLeft();
|
|
|
|
|
y -= getTotalPaddingTop();
|
|
|
|
|
x += getScrollX();
|
|
|
|
|
y += getScrollY();
|
|
|
|
|
|
|
|
|
|
Layout layout = getLayout();
|
|
|
|
|
int line = layout.getLineForVertical(y);
|
|
|
|
|
int off = layout.getOffsetForHorizontal(line, x);
|
|
|
|
|
Selection.setSelection(getText(), off);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return super.onTouchEvent(event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onKeyDown(int keyCode, KeyEvent event) {
|
|
|
|
|
switch (keyCode) {
|
|
|
|
|
case KeyEvent.KEYCODE_ENTER:
|
|
|
|
|
if (mOnTextViewChangeListener != null) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case KeyEvent.KEYCODE_DEL:
|
|
|
|
|
mSelectionStartBeforeDelete = getSelectionStart();
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return super.onKeyDown(keyCode, event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onKeyUp(int keyCode, KeyEvent event) {
|
|
|
|
|
switch(keyCode) {
|
|
|
|
|
case KeyEvent.KEYCODE_DEL:
|
|
|
|
|
if (mOnTextViewChangeListener != null) {
|
|
|
|
|
if (0 == mSelectionStartBeforeDelete && mIndex != 0) {
|
|
|
|
|
mOnTextViewChangeListener.onEditTextDelete(mIndex, getText().toString());
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
Log.d(TAG, "OnTextViewChangeListener was not seted");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case KeyEvent.KEYCODE_ENTER:
|
|
|
|
|
if (mOnTextViewChangeListener != null) {
|
|
|
|
|
int selectionStart = getSelectionStart();
|
|
|
|
|
String text = getText().subSequence(selectionStart, length()).toString();
|
|
|
|
|
setText(getText().subSequence(0, selectionStart));
|
|
|
|
|
mOnTextViewChangeListener.onEditTextEnter(mIndex + 1, text);
|
|
|
|
|
} else {
|
|
|
|
|
Log.d(TAG, "OnTextViewChangeListener was not seted");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return super.onKeyUp(keyCode, event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
|
|
|
|
|
if (mOnTextViewChangeListener != null) {
|
|
|
|
|
if (!focused && TextUtils.isEmpty(getText())) {
|
|
|
|
|
mOnTextViewChangeListener.onTextChange(mIndex, false);
|
|
|
|
|
} else {
|
|
|
|
|
mOnTextViewChangeListener.onTextChange(mIndex, true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
super.onFocusChanged(focused, direction, previouslyFocusedRect);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onCreateContextMenu(ContextMenu menu) {
|
|
|
|
|
if (getText() instanceof Spanned) {
|
|
|
|
|
int selStart = getSelectionStart();
|
|
|
|
|
int selEnd = getSelectionEnd();
|
|
|
|
|
|
|
|
|
|
int min = Math.min(selStart, selEnd);
|
|
|
|
|
int max = Math.max(selStart, selEnd);
|
|
|
|
|
|
|
|
|
|
final URLSpan[] urls = ((Spanned) getText()).getSpans(min, max, URLSpan.class);
|
|
|
|
|
if (urls.length == 1) {
|
|
|
|
|
int defaultResId = 0;
|
|
|
|
|
for(String schema: sSchemaActionResMap.keySet()) {
|
|
|
|
|
if(urls[0].getURL().indexOf(schema) >= 0) {
|
|
|
|
|
defaultResId = sSchemaActionResMap.get(schema);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (defaultResId == 0) {
|
|
|
|
|
defaultResId = R.string.note_link_other;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
menu.add(0, 0, 0, defaultResId).setOnMenuItemClickListener(
|
|
|
|
|
new OnMenuItemClickListener() {
|
|
|
|
|
public boolean onMenuItemClick(MenuItem item) {
|
|
|
|
|
// goto a new intent
|
|
|
|
|
urls[0].onClick(NoteEditText.this);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
super.onCreateContextMenu(menu);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
import android.content.Context;
|
|
|
|
|
import android.graphics.Rect;
|
|
|
|
|
import android.text.Layout;
|
|
|
|
|
import android.text.Selection;
|
|
|
|
|
import android.text.Spanned;
|
|
|
|
|
import android.text.TextUtils;
|
|
|
|
|
import android.text.style.URLSpan;
|
|
|
|
|
import android.util.AttributeSet;
|
|
|
|
|
import android.util.Log;
|
|
|
|
|
import android.view.ContextMenu;
|
|
|
|
|
import android.view.KeyEvent;
|
|
|
|
|
import android.view.MenuItem;
|
|
|
|
|
import android.view.MenuItem.OnMenuItemClickListener;
|
|
|
|
|
import android.view.MotionEvent;
|
|
|
|
|
import android.widget.EditText;
|
|
|
|
|
|
|
|
|
|
import net.micode.notes.R;
|
|
|
|
|
|
|
|
|
|
import java.util.HashMap;
|
|
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
|
|
// NoteEditText类继承自EditText,是一个自定义的文本编辑框控件,在原生EditText的基础上添加了一些特定的功能和交互逻辑,
|
|
|
|
|
// 例如处理特定按键事件、触摸事件以及上下文菜单相关操作等,用于笔记编辑等相关场景。
|
|
|
|
|
public class NoteEditText extends EditText {
|
|
|
|
|
private static final String TAG = "NoteEditText";
|
|
|
|
|
// 用于记录当前文本编辑框的索引位置,可能在多个编辑框组成的列表等场景下用于区分不同的编辑框个体。
|
|
|
|
|
private int mIndex;
|
|
|
|
|
// 用于记录在删除操作前文本选择的起始位置,方便后续判断删除相关逻辑,例如是否删除整个编辑框内容等情况。
|
|
|
|
|
private int mSelectionStartBeforeDelete;
|
|
|
|
|
|
|
|
|
|
// 定义表示电话号码链接的协议头(scheme)字符串,用于识别文本中是否包含电话号码链接。
|
|
|
|
|
private static final String SCHEME_TEL = "tel:" ;
|
|
|
|
|
// 定义表示网页链接(HTTP协议)的协议头字符串,用于识别文本中的网页链接。
|
|
|
|
|
private static final String SCHEME_HTTP = "http:" ;
|
|
|
|
|
// 定义表示电子邮件链接的协议头字符串,用于识别文本中的邮件链接。
|
|
|
|
|
private static final String SCHEME_EMAIL = "mailto:" ;
|
|
|
|
|
|
|
|
|
|
// 创建一个HashMap,用于存储不同协议头(scheme)与对应的字符串资源ID的映射关系,
|
|
|
|
|
// 这些字符串资源ID通常用于在界面上显示对应链接类型的操作提示文本等内容。
|
|
|
|
|
private static final Map<String, Integer> sSchemaActionResMap = new HashMap<String, Integer>();
|
|
|
|
|
static {
|
|
|
|
|
// 将不同协议头与对应的字符串资源ID添加到映射表中,方便后续根据链接类型查找对应的显示文本资源。
|
|
|
|
|
sSchemaActionResMap.put(SCHEME_TEL, R.string.note_link_tel);
|
|
|
|
|
sSchemaActionResMap.put(SCHEME_HTTP, R.string.note_link_web);
|
|
|
|
|
sSchemaActionResMap.put(SCHEME_EMAIL, R.string.note_link_email);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 定义一个接口,用于与外部类进行交互,外部类实现此接口可以监听该文本编辑框内文本相关的变化事件,
|
|
|
|
|
// 例如文本删除、回车键按下添加新编辑框以及文本有无变化等情况,并做出相应处理。
|
|
|
|
|
public interface OnTextViewChangeListener {
|
|
|
|
|
/**
|
|
|
|
|
* 当按下删除键({@link KeyEvent#KEYCODE_DEL})且文本内容为空时,删除当前编辑文本框,
|
|
|
|
|
* 外部类实现此方法可处理对应删除逻辑,比如从列表中移除该编辑框等操作。
|
|
|
|
|
*/
|
|
|
|
|
void onEditTextDelete(int index, String text);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 当按下回车键({@link KeyEvent#KEYCODE_ENTER})时,在当前编辑文本框之后添加新的编辑文本框,
|
|
|
|
|
* 外部类实现此方法可进行创建新编辑框并添加到合适位置等相关操作。
|
|
|
|
|
*/
|
|
|
|
|
void onEditTextEnter(int index, String text);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 当文本内容发生变化时,隐藏或显示相关的菜单项选项等,外部类可根据文本有无来控制对应UI元素的显示隐藏状态。
|
|
|
|
|
*/
|
|
|
|
|
void onTextChange(int index, boolean hasText);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 用于存储实现了OnTextViewChangeListener接口的实例,以便在文本编辑框相关事件发生时通知外部类进行相应处理。
|
|
|
|
|
private OnTextViewChangeListener mOnTextViewChangeListener;
|
|
|
|
|
|
|
|
|
|
// 构造函数,只传入上下文(Context)参数,调用父类构造函数创建实例,并初始化索引位置为0,
|
|
|
|
|
// 这种构造方式常用于通过代码动态创建该编辑框实例的场景。
|
|
|
|
|
public NoteEditText(Context context) {
|
|
|
|
|
super(context, null);
|
|
|
|
|
mIndex = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 设置当前文本编辑框的索引位置的方法,外部可调用此方法来更新索引值,方便区分不同编辑框个体。
|
|
|
|
|
public void setIndex(int index) {
|
|
|
|
|
mIndex = index;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 设置文本变化监听器的方法,外部类通过传入实现了OnTextViewChangeListener接口的实例,
|
|
|
|
|
// 来注册监听该编辑框相关文本变化事件,以便做出相应处理。
|
|
|
|
|
public void setOnTextViewChangeListener(OnTextViewChangeListener listener) {
|
|
|
|
|
mOnTextViewChangeListener = listener;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 构造函数,传入上下文(Context)和属性集(AttributeSet)参数,按照安卓系统默认的文本编辑框样式(通过android.R.attr.editTextStyle指定)创建实例,
|
|
|
|
|
// 通常用于在XML布局文件中定义该编辑框并解析相关属性进行初始化的场景。
|
|
|
|
|
public NoteEditText(Context context, AttributeSet attrs) {
|
|
|
|
|
super(context, attrs, android.R.attr.editTextStyle);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 构造函数,传入上下文(Context)、属性集(AttributeSet)和默认样式(defStyle)参数,创建编辑框实例,
|
|
|
|
|
// 可用于更灵活地自定义编辑框的样式和属性,当前构造函数中没有添加额外的初始化逻辑(TODO处可按需添加)。
|
|
|
|
|
public NoteEditText(Context context, AttributeSet attrs, int defStyle) {
|
|
|
|
|
super(context, attrs, defStyle);
|
|
|
|
|
// TODO Auto-generated constructor stub
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 重写父类的onTouchEvent方法,用于处理触摸事件,在这里主要是处理按下(ACTION_DOWN)动作,
|
|
|
|
|
// 目的是根据触摸的坐标位置来设置文本的选择位置,方便用户后续进行复制、粘贴等操作与选中位置相关的操作。
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onTouchEvent(MotionEvent event) {
|
|
|
|
|
switch (event.getAction()) {
|
|
|
|
|
case MotionEvent.ACTION_DOWN:
|
|
|
|
|
// 获取触摸点的原始X坐标
|
|
|
|
|
int x = (int) event.getX();
|
|
|
|
|
// 获取触摸点的原始Y坐标
|
|
|
|
|
int y = (int) event.getY();
|
|
|
|
|
// 减去左边的总内边距,将坐标转换为相对于文本内容区域的坐标
|
|
|
|
|
x -= getTotalPaddingLeft();
|
|
|
|
|
// 减去上边的总内边距,同样将坐标转换为相对于文本内容区域的坐标
|
|
|
|
|
y -= getTotalPaddingTop();
|
|
|
|
|
// 加上滚动的X偏移量,考虑文本滚动情况,获取准确的相对于文本内容的坐标
|
|
|
|
|
x += getScrollX();
|
|
|
|
|
// 加上滚动的Y偏移量,获取准确的相对于文本内容的坐标
|
|
|
|
|
y += getScrollY();
|
|
|
|
|
|
|
|
|
|
// 获取文本的布局信息对象,用于后续根据坐标获取对应文本位置等操作
|
|
|
|
|
Layout layout = getLayout();
|
|
|
|
|
// 根据触摸点的垂直坐标(Y坐标)获取对应的文本行数
|
|
|
|
|
int line = layout.getLineForVertical(y);
|
|
|
|
|
// 根据触摸点的水平坐标(X坐标)以及所在行数,获取对应的文本字符偏移量(即文本中的位置索引)
|
|
|
|
|
int off = layout.getOffsetForHorizontal(line, x);
|
|
|
|
|
// 根据获取到的字符偏移量设置文本的选择范围,这里只设置了一个位置,相当于将光标定位到该位置
|
|
|
|
|
Selection.setSelection(getText(), off);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 调用父类的onTouchEvent方法继续处理其他触摸事件相关逻辑,保证原生的触摸行为也能正常执行
|
|
|
|
|
return super.onTouchEvent(event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 重写父类的onKeyDown方法,用于处理按键按下事件,在这里主要是针对回车键(KEYCODE_ENTER)和删除键(KEYCODE_DEL)按下时做一些前期记录等准备工作,
|
|
|
|
|
// 具体的业务逻辑处理在按键抬起(onKeyUp)方法中进行。
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onKeyDown(int keyCode, KeyEvent event) {
|
|
|
|
|
switch (keyCode) {
|
|
|
|
|
case KeyEvent.KEYCODE_ENTER:
|
|
|
|
|
// 如果设置了文本变化监听器(mOnTextViewChangeListener不为空),表示外部类关注回车键按下事件,
|
|
|
|
|
// 这里返回false,让后续的按键抬起(onKeyUp)事件能继续处理相关逻辑,例如添加新编辑框等操作。
|
|
|
|
|
if (mOnTextViewChangeListener!= null) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case KeyEvent.KEYCODE_DEL:
|
|
|
|
|
// 当按下删除键时,记录当前文本选择的起始位置,方便后续判断是否删除整个编辑框内容等情况。
|
|
|
|
|
mSelectionStartBeforeDelete = getSelectionStart();
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
// 调用父类的onKeyDown方法继续处理其他按键按下相关逻辑,保证原生的按键按下行为也能正常执行
|
|
|
|
|
return super.onKeyDown(keyCode, event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 重写父类的onKeyUp方法,用于处理按键抬起事件,在这里针对回车键(KEYCODE_ENTER)和删除键(KEYCODE_DEL)抬起时,
|
|
|
|
|
// 根据不同情况执行相应的业务逻辑,例如删除编辑框、添加新编辑框等操作,前提是设置了文本变化监听器(mOnTextViewChangeListener不为空)。
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onKeyUp(int keyCode, KeyEvent event) {
|
|
|
|
|
switch(keyCode) {
|
|
|
|
|
case KeyEvent.KEYCODE_DEL:
|
|
|
|
|
if (mOnTextViewChangeListener!= null) {
|
|
|
|
|
// 判断如果文本选择的起始位置为0(通常表示光标在文本开头)且当前编辑框索引不为0(说明不是第一个编辑框),
|
|
|
|
|
// 则调用文本变化监听器的onEditTextDelete方法通知外部类删除当前编辑框,然后返回true,表示已经处理了该按键事件。
|
|
|
|
|
if (0 == mSelectionStartBeforeDelete && mIndex!= 0) {
|
|
|
|
|
mOnTextViewChangeListener.onEditTextDelete(mIndex, getText().toString());
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// 如果没有设置文本变化监听器,则打印日志提示信息,表示监听器未设置,无法处理删除相关逻辑。
|
|
|
|
|
Log.d(TAG, "OnTextViewChangeListener was not seted");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case KeyEvent.KEYCODE_ENTER:
|
|
|
|
|
if (mOnTextViewChangeListener!= null) {
|
|
|
|
|
// 获取当前文本选择的起始位置,用于后续截取文本操作。
|
|
|
|
|
int selectionStart = getSelectionStart();
|
|
|
|
|
// 获取从文本选择起始位置到文本末尾的子字符串,可能是要添加到新编辑框的内容(如果有需要的话)。
|
|
|
|
|
String text = getText().subSequence(selectionStart, length()).toString();
|
|
|
|
|
// 将当前编辑框的文本内容截取为从开头到选择起始位置的部分,相当于清空了选择位置之后的文本(可能是要添加到新编辑框的部分)。
|
|
|
|
|
setText(getText().subSequence(0, selectionStart));
|
|
|
|
|
// 调用文本变化监听器的onEditTextEnter方法通知外部类在当前编辑框之后添加新编辑框,并传入新编辑框索引和可能要添加的文本内容。
|
|
|
|
|
mOnTextViewChangeListener.onEditTextEnter(mIndex + 1, text);
|
|
|
|
|
} else {
|
|
|
|
|
// 如果没有设置文本变化监听器,则打印日志提示信息,表示监听器未设置,无法处理回车键按下添加新编辑框相关逻辑。
|
|
|
|
|
Log.d(TAG, "OnTextViewChangeListener was not seted");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
// 调用父类的onKeyUp方法继续处理其他按键抬起相关逻辑,保证原生的按键抬起行为也能正常执行
|
|
|
|
|
return super.onKeyUp(keyCode, event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 重写父类的onFocusChanged方法,用于处理焦点变化事件,在这里根据焦点状态(是否获得焦点)以及文本内容是否为空,
|
|
|
|
|
// 通过文本变化监听器(mOnTextViewChangeListener)通知外部类文本有无情况,以便外部类进行相应的UI元素显示隐藏等操作。
|
|
|
|
|
@Override
|
|
|
|
|
protected void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
|
|
|
|
|
if (mOnTextViewChangeListener!= null) {
|
|
|
|
|
if (!focused && TextUtils.isEmpty(getText())) {
|
|
|
|
|
// 如果失去焦点且文本内容为空,调用文本变化监听器的onTextChange方法通知外部类文本为空的情况,参数hasText为false。
|
|
|
|
|
mOnTextViewChangeListener.onTextChange(mIndex, false);
|
|
|
|
|
} else {
|
|
|
|
|
// 如果获得焦点或者文本内容不为空,调用文本变化监听器的onTextChange方法通知外部类文本有内容的情况,参数hasText为true。
|
|
|
|
|
mOnTextViewChangeListener.onTextChange(mIndex, true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
super.onFocusChanged(focused, direction, previouslyFocusedRect);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 重写父类的onCreateContextMenu方法,用于创建上下文菜单(长按文本等操作弹出的菜单),
|
|
|
|
|
// 在这里主要是检查当前选中的文本中是否包含URL链接(通过URLSpan判断),如果包含且只有一个链接,
|
|
|
|
|
// 则根据链接的协议头(scheme)查找对应的字符串资源ID,添加一个菜单项用于执行链接相关操作(如打开网页、拨打电话等)。
|
|
|
|
|
@Override
|
|
|
|
|
protected void onCreateContextMenu(ContextMenu menu) {
|
|
|
|
|
if (getText() instanceof Spanned) {
|
|
|
|
|
// 获取当前文本选择的起始位置
|
|
|
|
|
int selStart = getSelectionStart();
|
|
|
|
|
// 获取当前文本选择的结束位置
|
|
|
|
|
int selEnd = getSelectionEnd();
|
|
|
|
|
|
|
|
|
|
// 获取选择起始位置和结束位置中的较小值,用于后续范围判断等操作
|
|
|
|
|
int min = Math.min(selStart, selEnd);
|
|
|
|
|
// 获取选择起始位置和结束位置中的较大值,用于后续范围判断等操作
|
|
|
|
|
int max = Math.max(selStart, selEnd);
|
|
|
|
|
|
|
|
|
|
// 获取在选择文本范围内的所有URLSpan对象(即包含的链接信息),返回一个数组,可能包含多个链接(但这里后续只处理长度为1的情况)。
|
|
|
|
|
final URLSpan[] urls = ((Spanned) getText()).getSpans(min, max, URLSpan.class);
|
|
|
|
|
if (urls.length == 1) {
|
|
|
|
|
int defaultResId = 0;
|
|
|
|
|
for(String schema: sSchemaActionResMap.keySet()) {
|
|
|
|
|
// 遍历存储协议头与字符串资源ID映射关系的集合,检查链接的协议头是否匹配已定义的协议头(如tel、http、mailto等)
|
|
|
|
|
if(urls[0].getURL().indexOf(schema) >= 0) {
|
|
|
|
|
// 如果匹配,获取对应的字符串资源ID,用于设置菜单项显示的文本内容。
|
|
|
|
|
defaultResId = sSchemaActionResMap.get(schema);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (defaultResId == 0) {
|
|
|
|
|
// 如果没有匹配到已知的协议头,设置一个默认的字符串资源ID,表示其他类型链接。
|
|
|
|
|
defaultResId = R.string.note_link_other;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 添加一个菜单项到上下文菜单中,菜单项的ID、顺序、分组等参数暂设为0(可根据实际需求调整),显示的文本内容根据获取到的字符串资源ID获取,
|
|
|
|
|
// 并设置菜单项的点击监听器,当点击该菜单项时,通过URLSpan的onClick方法执行链接相关操作(如打开网页、拨打电话等)。
|
|
|
|
|
menu.add(0, 0, 0, defaultResId).setOnMenuItemClickListener(
|
|
|
|
|
new OnMenuItemClickListener() {
|
|
|
|
|
public boolean onMenuItemClick(MenuItem item) {
|
|
|
|
|
// goto a new intent
|
|
|
|
|
urls[0].onClick(NoteEditText.this);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
super.onCreateContextMenu(menu);
|
|
|
|
|
}
|
|
|
|
|
}
|