|
|
|
@ -37,6 +37,7 @@ import net.micode.notes.R;
|
|
|
|
|
import java.util.HashMap;
|
|
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
|
|
//笔记编辑ui
|
|
|
|
|
public class NoteEditText extends EditText {
|
|
|
|
|
private static final String TAG = "NoteEditText";
|
|
|
|
|
private int mIndex;
|
|
|
|
@ -99,71 +100,118 @@ public class NoteEditText extends EditText {
|
|
|
|
|
// TODO Auto-generated constructor stub
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 处理触摸事件,实现文本选择功能
|
|
|
|
|
*
|
|
|
|
|
* @param event 触摸事件对象,包含触摸位置等信息
|
|
|
|
|
* @return 返回true表示已处理该事件,false表示未处理
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onTouchEvent(MotionEvent event) {
|
|
|
|
|
switch (event.getAction()) {
|
|
|
|
|
case MotionEvent.ACTION_DOWN:
|
|
|
|
|
|
|
|
|
|
case MotionEvent.ACTION_DOWN: // 手指按下屏幕事件
|
|
|
|
|
// 1. 获取触摸点的原始坐标
|
|
|
|
|
int x = (int) event.getX();
|
|
|
|
|
int y = (int) event.getY();
|
|
|
|
|
x -= getTotalPaddingLeft();
|
|
|
|
|
y -= getTotalPaddingTop();
|
|
|
|
|
x += getScrollX();
|
|
|
|
|
y += getScrollY();
|
|
|
|
|
|
|
|
|
|
// 2. 调整坐标,去除内边距的影响
|
|
|
|
|
x -= getTotalPaddingLeft(); // 减去左侧内边距
|
|
|
|
|
y -= getTotalPaddingTop(); // 减去顶部内边距
|
|
|
|
|
|
|
|
|
|
// 3. 考虑滚动偏移量
|
|
|
|
|
x += getScrollX(); // 加上水平滚动距离
|
|
|
|
|
y += getScrollY(); // 加上垂直滚动距离
|
|
|
|
|
|
|
|
|
|
// 4. 获取文本布局对象
|
|
|
|
|
Layout layout = getLayout();
|
|
|
|
|
|
|
|
|
|
// 5. 根据y坐标获取行号
|
|
|
|
|
int line = layout.getLineForVertical(y);
|
|
|
|
|
|
|
|
|
|
// 6. 根据x坐标获取该行内的字符偏移量
|
|
|
|
|
int off = layout.getOffsetForHorizontal(line, x);
|
|
|
|
|
|
|
|
|
|
// 7. 设置文本选择位置
|
|
|
|
|
Selection.setSelection(getText(), off);
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 调用父类方法处理其他触摸事件(如移动、抬起等)
|
|
|
|
|
return super.onTouchEvent(event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onKeyDown(int keyCode, KeyEvent event) {
|
|
|
|
|
// 处理键盘按下事件
|
|
|
|
|
switch (keyCode) {
|
|
|
|
|
case KeyEvent.KEYCODE_ENTER:
|
|
|
|
|
case KeyEvent.KEYCODE_ENTER: // 回车键处理
|
|
|
|
|
// 当存在文本变化监听器时,不消费回车键事件
|
|
|
|
|
// 以便监听器可以自定义回车键行为(如换行处理)
|
|
|
|
|
if (mOnTextViewChangeListener != null) {
|
|
|
|
|
return false;
|
|
|
|
|
return false; // false表示允许事件继续传递
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case KeyEvent.KEYCODE_DEL:
|
|
|
|
|
|
|
|
|
|
case KeyEvent.KEYCODE_DEL: // 删除键处理
|
|
|
|
|
// 记录删除前的光标起始位置
|
|
|
|
|
// 用于支持撤销操作或特殊删除逻辑
|
|
|
|
|
mSelectionStartBeforeDelete = getSelectionStart();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
// 其他按键不做特殊处理
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 未被处理的按键事件交由父类默认处理
|
|
|
|
|
return super.onKeyDown(keyCode, event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 处理键盘按键抬起事件
|
|
|
|
|
* @param keyCode 按键代码,如KeyEvent.KEYCODE_DEL
|
|
|
|
|
* @param event 按键事件对象
|
|
|
|
|
* @return 返回true表示事件已处理,false表示未处理
|
|
|
|
|
*/
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onKeyUp(int keyCode, KeyEvent event) {
|
|
|
|
|
switch(keyCode) {
|
|
|
|
|
case KeyEvent.KEYCODE_DEL:
|
|
|
|
|
case KeyEvent.KEYCODE_DEL: // 删除键抬起事件
|
|
|
|
|
// 当设置了文本变化监听器时
|
|
|
|
|
if (mOnTextViewChangeListener != null) {
|
|
|
|
|
// 如果光标在文本开头且不是第一个编辑框
|
|
|
|
|
if (0 == mSelectionStartBeforeDelete && mIndex != 0) {
|
|
|
|
|
// 回调删除事件,传递当前索引和文本内容
|
|
|
|
|
mOnTextViewChangeListener.onEditTextDelete(mIndex, getText().toString());
|
|
|
|
|
return true;
|
|
|
|
|
return true; // 事件已处理
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
Log.d(TAG, "OnTextViewChangeListener was not seted");
|
|
|
|
|
Log.d(TAG, "OnTextViewChangeListener was not set");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case KeyEvent.KEYCODE_ENTER:
|
|
|
|
|
|
|
|
|
|
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");
|
|
|
|
|
Log.d(TAG, "OnTextViewChangeListener was not set");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
// 其他按键不做特殊处理
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
// 未被处理的按键事件交由父类处理
|
|
|
|
|
return super.onKeyUp(keyCode, event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -179,39 +227,57 @@ public class NoteEditText extends EditText {
|
|
|
|
|
super.onFocusChanged(focused, direction, previouslyFocusedRect);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 创建上下文菜单(长按文本时弹出的菜单)
|
|
|
|
|
* 主要处理文本中的URL链接,为链接添加对应的菜单项
|
|
|
|
|
*/
|
|
|
|
|
@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对象(超链接)
|
|
|
|
|
final URLSpan[] urls = ((Spanned) getText()).getSpans(min, max, URLSpan.class);
|
|
|
|
|
|
|
|
|
|
// 如果只选中了一个URL链接
|
|
|
|
|
if (urls.length == 1) {
|
|
|
|
|
int defaultResId = 0;
|
|
|
|
|
for(String schema: sSchemaActionResMap.keySet()) {
|
|
|
|
|
int defaultResId = 0; // 默认菜单项资源ID
|
|
|
|
|
|
|
|
|
|
// 遍历预定义的URL协议映射表
|
|
|
|
|
for(String schema : sSchemaActionResMap.keySet()) {
|
|
|
|
|
// 检查当前URL是否包含特定协议(如http, mailto等)
|
|
|
|
|
if(urls[0].getURL().indexOf(schema) >= 0) {
|
|
|
|
|
// 获取对应的菜单项文本资源ID
|
|
|
|
|
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
|
|
|
|
|
// 当菜单项被点击时,触发URL链接的默认行为
|
|
|
|
|
// 比如打开浏览器或邮件客户端
|
|
|
|
|
urls[0].onClick(NoteEditText.this);
|
|
|
|
|
return true;
|
|
|
|
|
return true; // 表示事件已处理
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 调用父类方法继续处理其他上下文菜单逻辑
|
|
|
|
|
super.onCreateContextMenu(menu);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|