diff --git a/src/Notes-master/mainfests/AndroidManifest.xml b/src/Notes-master/mainfests/AndroidManifest.xml
index a85867b..e099d24 100644
--- a/src/Notes-master/mainfests/AndroidManifest.xml
+++ b/src/Notes-master/mainfests/AndroidManifest.xml
@@ -17,6 +17,7 @@
+
diff --git a/src/Notes-master/res/layout/note_edit.xml b/src/Notes-master/res/layout/note_edit.xml
index abec1dc..9d1b4e2 100644
--- a/src/Notes-master/res/layout/note_edit.xml
+++ b/src/Notes-master/res/layout/note_edit.xml
@@ -74,6 +74,41 @@
android:background="@drawable/bg_btn_set_color" />
+
+
+
+
+
+
+
+
+
-
-
-
-
-
-
-
-
-
不能为空便签设置闹钟提醒
不能将空便签发送到桌面
清单模式下不支持格式化
+ 拒绝许可,无法访问图像
+ 插入失败,请再尝试一次
请先选择文本
导出成功
导出失败
diff --git a/src/Notes-master/res/values/strings.xml b/src/Notes-master/res/values/strings.xml
index 9889716..6d6e9c8 100644
--- a/src/Notes-master/res/values/strings.xml
+++ b/src/Notes-master/res/values/strings.xml
@@ -90,6 +90,8 @@
Sorry, can not send and empty note to home
Formatting is not supported in list mode
Please select text first
+ Permission denied, cannot access images
+ Failed to insert image, please try again
Export successful
Export fail
Export text file (%1$s) to SD (%2$s) directory
diff --git a/src/Notes-master/src/net/micode/notes/tool/DataUtils.java b/src/Notes-master/src/net/micode/notes/tool/DataUtils.java
index 8754e57..3822dc7 100644
--- a/src/Notes-master/src/net/micode/notes/tool/DataUtils.java
+++ b/src/Notes-master/src/net/micode/notes/tool/DataUtils.java
@@ -58,11 +58,11 @@ public class DataUtils {
// Query the note from note table
cursor = resolver.query(ContentUris.withAppendedId(Notes.CONTENT_NOTE_URI, id),
null, null, null, null);
-
+
if (cursor != null && cursor.moveToFirst()) {
// Read all columns from the note
ContentValues trashValues = new ContentValues();
-
+
// Copy all columns from note to trash
for (String column : cursor.getColumnNames()) {
int columnIndex = cursor.getColumnIndex(column);
@@ -86,17 +86,17 @@ public class DataUtils {
}
}
}
-
+
// Set deleted_date to current time
trashValues.put("deleted_date", System.currentTimeMillis());
-
+
// Insert into trash table
resolver.insert(Notes.CONTENT_TRASH_URI, trashValues);
-
+
// Delete from note table
resolver.delete(ContentUris.withAppendedId(Notes.CONTENT_NOTE_URI, id), null, null);
}
-
+
if (cursor != null) {
cursor.close();
cursor = null;
@@ -127,7 +127,7 @@ public class DataUtils {
}
public static boolean batchMoveToFolder(ContentResolver resolver, HashSet ids,
- long folderId) {
+ long folderId) {
if (ids == null) {
Log.d(TAG, "the ids is null");
return true;
@@ -230,8 +230,8 @@ public class DataUtils {
public static boolean checkVisibleFolderName(ContentResolver resolver, String name) {
Cursor cursor = resolver.query(Notes.CONTENT_NOTE_URI, null,
NoteColumns.TYPE + "=" + Notes.TYPE_FOLDER +
- " AND " + NoteColumns.PARENT_ID + "<>" + Notes.ID_TRASH_FOLER +
- " AND " + NoteColumns.SNIPPET + "=?",
+ " AND " + NoteColumns.PARENT_ID + "<>" + Notes.ID_TRASH_FOLER +
+ " AND " + NoteColumns.SNIPPET + "=?",
new String[] { name }, null);
boolean exist = false;
if(cursor != null) {
@@ -298,11 +298,11 @@ public class DataUtils {
// Query the note from trash table
cursor = resolver.query(ContentUris.withAppendedId(Notes.CONTENT_TRASH_URI, trashId),
null, null, null, null);
-
+
if (cursor != null && cursor.moveToFirst()) {
// Read all columns from the trash note
ContentValues noteValues = new ContentValues();
-
+
// Copy all columns from trash to note (except deleted_date)
for (String column : cursor.getColumnNames()) {
if ("deleted_date".equals(column)) {
@@ -329,13 +329,13 @@ public class DataUtils {
}
}
}
-
+
// Insert back to note table
resolver.insert(Notes.CONTENT_NOTE_URI, noteValues);
-
+
// Delete from trash table
resolver.delete(ContentUris.withAppendedId(Notes.CONTENT_TRASH_URI, trashId), null, null);
-
+
return true;
}
return false;
@@ -381,7 +381,7 @@ public class DataUtils {
Cursor cursor = resolver.query(Notes.CONTENT_DATA_URI,
new String [] { CallNote.NOTE_ID },
CallNote.CALL_DATE + "=? AND " + CallNote.MIME_TYPE + "=? AND PHONE_NUMBERS_EQUAL("
- + CallNote.PHONE_NUMBER + ",?)",
+ + CallNote.PHONE_NUMBER + ",?)",
new String [] { String.valueOf(callDate), CallNote.CONTENT_ITEM_TYPE, phoneNumber },
null);
@@ -419,6 +419,8 @@ public class DataUtils {
public static String getFormattedSnippet(String snippet) {
if (snippet != null) {
snippet = snippet.trim();
+ // 替换[IMAGE]标签为友好的占位符
+ snippet = snippet.replaceAll("\\[IMAGE\\].*?\\[/IMAGE\\]", "[图片]");
int index = snippet.indexOf('\n');
if (index != -1) {
snippet = snippet.substring(0, index);
diff --git a/src/Notes-master/src/net/micode/notes/ui/NoteEditActivity.java b/src/Notes-master/src/net/micode/notes/ui/NoteEditActivity.java
index 944d043..ef5d3d5 100644
--- a/src/Notes-master/src/net/micode/notes/ui/NoteEditActivity.java
+++ b/src/Notes-master/src/net/micode/notes/ui/NoteEditActivity.java
@@ -27,18 +27,26 @@ import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.SharedPreferences;
+import android.content.pm.PackageManager;
import android.graphics.Paint;
+import android.net.Uri;
+import android.os.Build;
import android.os.Bundle;
import android.preference.PreferenceManager;
+import android.provider.MediaStore;
import android.text.Spannable;
import android.text.SpannableString;
+import android.text.SpannableStringBuilder;
import android.text.TextUtils;
+import android.text.Editable;
import android.text.Html;
import android.text.format.DateUtils;
import android.text.style.BackgroundColorSpan;
import android.text.style.StyleSpan;
import android.text.style.UnderlineSpan;
-import android.graphics.Typeface;
+import android.text.style.ImageSpan;
+import java.util.regex.Pattern;
+import java.util.regex.Matcher;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.Menu;
@@ -51,23 +59,19 @@ import android.widget.CheckBox;
import android.widget.CompoundButton;
import android.widget.CompoundButton.OnCheckedChangeListener;
import android.widget.EditText;
+import android.widget.ImageButton;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;
import android.widget.Toast;
-import android.widget.ImageButton;
-import android.provider.MediaStore;
-import android.net.Uri;
-import java.io.File;
-import android.text.SpannableString;
-import android.text.SpannableStringBuilder;
-import android.text.Editable;
-import android.text.style.ImageSpan;
+import android.graphics.Typeface;
+import android.graphics.Bitmap;
+import android.graphics.BitmapFactory;
import android.graphics.drawable.Drawable;
import android.content.ContentResolver;
import android.content.res.Resources;
-import android.graphics.Bitmap;
-import android.graphics.BitmapFactory;
+// import android.support.annotation.NonNull;
+import android.Manifest;
import java.io.InputStream;
import net.micode.notes.R;
@@ -172,7 +176,7 @@ public class NoteEditActivity extends Activity implements OnClickListener,
private SharedPreferences mSharedPrefs; // 共享偏好设置
private int mFontSizeId; // 字体大小ID
private ImageButton mBtnInsertImage; // 插入图片按钮 OMO
-
+
// 富文本工具栏按钮
private ImageButton mBtnBold; // 加粗按钮
private ImageButton mBtnItalic; // 斜体按钮
@@ -364,79 +368,22 @@ public class NoteEditActivity extends Activity implements OnClickListener,
mNoteEditor.setText("");
mNoteEditor.setSelection(0);
} else {
- Spannable spannable;
-
- // 检查是否是HTML格式(包含HTML标签)
- if (content.contains("<") && content.contains(">")) {
- try {
- // 从HTML格式恢复SpannableString,保留样式(加粗、斜体、下划线)
- // Html.fromHtml会自动将HTML标签转换为对应的Span
- spannable = (Spannable) Html.fromHtml(content);
- } catch (Exception e) {
- // 如果HTML解析失败,使用普通文本
- Log.e(TAG, "Error parsing HTML content: " + e.getMessage());
- spannable = new SpannableString(content);
- }
- } else {
- // 普通文本格式,直接创建SpannableString
- spannable = new SpannableString(content);
- }
-
- // 处理图片:将包含[IMAGE]标签的文本转换为包含ImageSpan的SpannableString
- // 注意:convertTextToSpannableWithImages会处理[IMAGE]标记,但会保留文本内容
- String textContent = spannable.toString();
- SpannableString finalSpannable = convertTextToSpannableWithImages(textContent);
-
- // 将样式从原始Spannable复制到处理后的SpannableString
- // 由于convertTextToSpannableWithImages可能改变了文本长度(用空格替换了[IMAGE]标签),
- // 我们需要基于文本内容来映射样式位置
- if (textContent.length() == finalSpannable.length()) {
- // 文本长度相同,直接复制样式
- StyleSpan[] styleSpans = spannable.getSpans(0, spannable.length(), StyleSpan.class);
- UnderlineSpan[] underlineSpans = spannable.getSpans(0, spannable.length(), UnderlineSpan.class);
-
- for (StyleSpan span : styleSpans) {
- int start = spannable.getSpanStart(span);
- int end = spannable.getSpanEnd(span);
- if (start >= 0 && end <= finalSpannable.length() && start < end) {
- finalSpannable.setSpan(new StyleSpan(span.getStyle()), start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
- }
- }
-
- for (UnderlineSpan span : underlineSpans) {
- int start = spannable.getSpanStart(span);
- int end = spannable.getSpanEnd(span);
- if (start >= 0 && end <= finalSpannable.length() && start < end) {
- finalSpannable.setSpan(new UnderlineSpan(), start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
- }
- }
- } else {
- // 文本长度不同,说明有图片被处理了
- // 这种情况下,样式位置需要重新计算
- // 为了简化,我们基于原始文本位置映射样式(可能会有偏差,但应该能处理大部分情况)
- StyleSpan[] styleSpans = spannable.getSpans(0, spannable.length(), StyleSpan.class);
- UnderlineSpan[] underlineSpans = spannable.getSpans(0, spannable.length(), UnderlineSpan.class);
-
- for (StyleSpan span : styleSpans) {
- int start = Math.max(0, Math.min(spannable.getSpanStart(span), finalSpannable.length()));
- int end = Math.max(start, Math.min(spannable.getSpanEnd(span), finalSpannable.length()));
- if (start < end) {
- finalSpannable.setSpan(new StyleSpan(span.getStyle()), start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
- }
- }
-
- for (UnderlineSpan span : underlineSpans) {
- int start = Math.max(0, Math.min(spannable.getSpanStart(span), finalSpannable.length()));
- int end = Math.max(start, Math.min(spannable.getSpanEnd(span), finalSpannable.length()));
- if (start < end) {
- finalSpannable.setSpan(new UnderlineSpan(), start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
- }
- }
+ // 第一步:处理HTML编码字符,将xx;转换为实际字符
+ String decodedContent = content;
+ try {
+ // 使用Html.fromHtml解码HTML编码字符
+ decodedContent = Html.fromHtml(content).toString();
+ } catch (Exception e) {
+ Log.e(TAG, "Error decoding HTML content: " + e.getMessage());
}
-
- mNoteEditor.setText(finalSpannable);
+
+ // 第二步:处理[IMAGE]标签,将其转换为ImageSpan
+ SpannableString spannable = convertTextToSpannableWithImages(decodedContent);
+
+ // 第三步:设置文本,确保文字和图片都能正常显示
+ mNoteEditor.setText(spannable);
// 将光标定位到文本末尾
- mNoteEditor.setSelection(finalSpannable.length());
+ mNoteEditor.setSelection(spannable.length());
}
}
@@ -472,6 +419,8 @@ public class NoteEditActivity extends Activity implements OnClickListener,
return new SpannableString("");
}
+ Log.d(TAG, "Converting text to Spannable with images: " + text);
+
// 创建一个新的SpannableStringBuilder
SpannableStringBuilder builder = new SpannableStringBuilder();
@@ -480,6 +429,8 @@ public class NoteEditActivity extends Activity implements OnClickListener,
int imageStart = text.indexOf("[IMAGE]", startIndex);
while (imageStart != -1) {
+ Log.d(TAG, "Found [IMAGE] tag at position: " + imageStart);
+
// 添加[IMAGE]标签之前的文本
builder.append(text.substring(startIndex, imageStart));
@@ -491,43 +442,97 @@ public class NoteEditActivity extends Activity implements OnClickListener,
break;
}
+ Log.d(TAG, "Found [/IMAGE] tag at position: " + imageEnd);
+
// 提取图片URI
String imageUri = text.substring(imageStart + "[IMAGE]".length(), imageEnd);
- Log.d(TAG, "Found image URI: " + imageUri);
+ Log.d(TAG, "Extracted image URI: " + imageUri);
- // 创建一个占位符字符串,用于插入ImageSpan
- String placeholder = " ";
- int placeholderStart = builder.length();
- builder.append(placeholder);
+ // 保留原始的[IMAGE]uri[/IMAGE]标签,直接在上面添加ImageSpan
+ String fullImageTag = "[IMAGE]" + imageUri + "[/IMAGE]";
+ int tagStart = builder.length();
+ builder.append(fullImageTag);
// 尝试将图片URI转换为Drawable并创建ImageSpan
try {
+ // 确保URI是有效的
+ Uri uri = Uri.parse(imageUri);
+ if (uri == null) {
+ Log.e(TAG, "Invalid URI: " + imageUri);
+ // 更新索引,继续处理下一个标签
+ startIndex = imageEnd + "[/IMAGE]".length();
+ imageStart = text.indexOf("[IMAGE]", startIndex);
+ continue;
+ }
+
+ Log.d(TAG, "Attempting to load image from URI: " + uri.toString());
+
+ // 使用BitmapFactory直接加载图片,这是最可靠的方式
ContentResolver resolver = getContentResolver();
- InputStream inputStream = resolver.openInputStream(Uri.parse(imageUri));
- Bitmap bitmap = BitmapFactory.decodeStream(inputStream);
- inputStream.close();
-
- // 计算图片的合适大小,确保它能适应EditText的宽度
- int screenWidth = getResources().getDisplayMetrics().widthPixels;
- int imageWidth = bitmap.getWidth();
- int imageHeight = bitmap.getHeight();
- float scale = (float) screenWidth / (float) imageWidth;
- int scaledHeight = (int) (imageHeight * scale);
-
- // 缩放图片
- Bitmap scaledBitmap = Bitmap.createScaledBitmap(bitmap, screenWidth, scaledHeight, true);
- Drawable drawable = new android.graphics.drawable.BitmapDrawable(getResources(), scaledBitmap);
- drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight());
-
- // 创建ImageSpan并添加到SpannableStringBuilder中
- ImageSpan imageSpan = new ImageSpan(drawable, ImageSpan.ALIGN_BASELINE);
- builder.setSpan(imageSpan, placeholderStart, placeholderStart + placeholder.length(), SpannableString.SPAN_INCLUSIVE_EXCLUSIVE);
+ InputStream inputStream = null;
+ try {
+ inputStream = resolver.openInputStream(uri);
+ if (inputStream == null) {
+ Log.e(TAG, "Failed to open input stream for URI: " + uri.toString());
+ // 更新索引,继续处理下一个标签
+ startIndex = imageEnd + "[/IMAGE]".length();
+ imageStart = text.indexOf("[IMAGE]", startIndex);
+ continue;
+ }
+
+ // 解码图片
+ Bitmap bitmap = BitmapFactory.decodeStream(inputStream);
+ if (bitmap == null) {
+ Log.e(TAG, "Failed to decode bitmap from URI: " + uri.toString());
+ // 更新索引,继续处理下一个标签
+ startIndex = imageEnd + "[/IMAGE]".length();
+ imageStart = text.indexOf("[IMAGE]", startIndex);
+ continue;
+ }
+
+ Log.d(TAG, "Successfully decoded bitmap with dimensions: " + bitmap.getWidth() + "x" + bitmap.getHeight());
+
+ // 计算图片的合适大小,确保它能适应EditText的宽度
+ int screenWidth = getResources().getDisplayMetrics().widthPixels;
+ // 减去左右边距,确保图片能完整显示
+ int maxWidth = screenWidth - 40;
+ int imageWidth = bitmap.getWidth();
+ int imageHeight = bitmap.getHeight();
+
+ // 计算缩放比例
+ float scale = 1.0f;
+ if (imageWidth > maxWidth) {
+ scale = (float) maxWidth / (float) imageWidth;
+ }
+ Log.d(TAG, "Calculated scale: " + scale);
+
+ // 缩放图片(如果需要)
+ if (scale < 1.0f) {
+ bitmap = Bitmap.createScaledBitmap(bitmap, (int)(imageWidth * scale), (int)(imageHeight * scale), true);
+ Log.d(TAG, "Scaled bitmap to: " + bitmap.getWidth() + "x" + bitmap.getHeight());
+ }
+
+ // 创建Drawable并设置边界
+ Drawable drawable = new android.graphics.drawable.BitmapDrawable(getResources(), bitmap);
+ drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight());
+
+ // 创建ImageSpan并添加到SpannableStringBuilder中
+ ImageSpan imageSpan = new ImageSpan(drawable, ImageSpan.ALIGN_BASELINE);
+ builder.setSpan(imageSpan, tagStart, tagStart + fullImageTag.length(), SpannableString.SPAN_INCLUSIVE_EXCLUSIVE);
+ Log.d(TAG, "Successfully added ImageSpan for URI: " + imageUri);
+ } finally {
+ // 确保InputStream被正确关闭
+ if (inputStream != null) {
+ try {
+ inputStream.close();
+ } catch (Exception e) {
+ Log.e(TAG, "Error closing input stream: " + e.toString());
+ }
+ }
+ }
} catch (Exception e) {
Log.e(TAG, "Error loading image: " + e.toString());
- // 如果加载图片失败,直接显示图片URI
- builder.append("[IMAGE]");
- builder.append(imageUri);
- builder.append("[/IMAGE]");
+ e.printStackTrace();
}
// 更新索引,继续查找下一个[IMAGE]标签
@@ -538,6 +543,7 @@ public class NoteEditActivity extends Activity implements OnClickListener,
// 添加剩余的文本
builder.append(text.substring(startIndex));
+ Log.d(TAG, "Final SpannableStringBuilder content: " + builder.toString());
return new SpannableString(builder);
}
@@ -679,12 +685,12 @@ public class NoteEditActivity extends Activity implements OnClickListener,
if (mBtnInsertImage != null) {
mBtnInsertImage.setOnClickListener(this);
}
-
+
// 初始化富文本工具栏按钮
mBtnBold = (ImageButton) findViewById(R.id.btn_bold);
mBtnItalic = (ImageButton) findViewById(R.id.btn_italic);
mBtnUnderline = (ImageButton) findViewById(R.id.btn_underline);
-
+
// 为富文本工具栏按钮设置点击监听器
if (mBtnBold != null) {
mBtnBold.setOnClickListener(this);
@@ -1266,16 +1272,10 @@ public class NoteEditActivity extends Activity implements OnClickListener,
}
mWorkingNote.setWorkingText(sb.toString());
} else {
- // 保存时,将SpannableString转换为HTML格式以保留样式信息
- Spannable spannable = mNoteEditor.getText();
- if (spannable != null && spannable.length() > 0) {
- // 将Spannable转换为HTML格式,保留加粗、斜体、下划线等样式
- // Html.toHtml会自动处理StyleSpan和UnderlineSpan
- String htmlContent = Html.toHtml(spannable);
- mWorkingNote.setWorkingText(htmlContent);
- } else {
- mWorkingNote.setWorkingText("");
- }
+ // 对于普通模式,直接保存EditText中的原始文本
+ // 这样可以保留[IMAGE]标签,确保图片信息不会丢失
+ String content = mNoteEditor.getText().toString();
+ mWorkingNote.setWorkingText(content);
}
return hasChecked;
}
@@ -1372,9 +1372,39 @@ public class NoteEditActivity extends Activity implements OnClickListener,
/**
* 从相册选择图片 OMO
*/
+ private static final int REQUEST_CODE_PERMISSION_STORAGE = 101;
+
private void pickImageFromGallery() {
- Intent intent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
- startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE);
+ // 检查并申请存储权限
+ if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
+ if (checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
+ // 申请权限
+ requestPermissions(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, REQUEST_CODE_PERMISSION_STORAGE);
+ } else {
+ // 权限已授予,直接选择图片
+ Intent intent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
+ startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE);
+ }
+ } else {
+ // Android 6.0以下版本,直接选择图片
+ Intent intent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
+ startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE);
+ }
+ }
+
+ // 处理权限请求结果
+ @Override
+ public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
+ super.onRequestPermissionsResult(requestCode, permissions, grantResults);
+ if (requestCode == REQUEST_CODE_PERMISSION_STORAGE) {
+ if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
+ // 权限被授予,执行图片选择
+ pickImageFromGallery();
+ } else {
+ // 权限被拒绝,显示提示
+ showToast(R.string.error_permission_denied);
+ }
+ }
}
/**
@@ -1388,27 +1418,50 @@ public class NoteEditActivity extends Activity implements OnClickListener,
Uri imageUri = data.getData();
if (imageUri != null) {
Log.d(TAG, "Image URI: " + imageUri.toString());
+
+ // 获取当前编辑器内容
+ Editable editable = mNoteEditor.getText();
+ int currentPosition = mNoteEditor.getSelectionStart();
+
try {
- // 将图片路径添加到笔记内容中
- String currentContent = mNoteEditor.getText().toString();
+ // 1. 保存当前编辑器的原始内容,用于后续恢复
+ String originalContent = editable.toString();
+ int originalSelection = mNoteEditor.getSelectionStart();
+
+ // 2. 插入完整的[IMAGE]uri[/IMAGE]标签到原始内容中
String imagePath = "[IMAGE]" + imageUri.toString() + "[/IMAGE]";
- String newContent = currentContent + (currentContent.isEmpty() ? "" : "\n") + imagePath;
- Log.d(TAG, "New content: " + newContent);
+ StringBuilder sb = new StringBuilder(originalContent);
+ sb.insert(originalSelection, imagePath);
+ String newContentWithTags = sb.toString();
+
+ // 3. 首先将包含ImageSpan的SpannableString显示到编辑器中
+ SpannableString spannable = convertTextToSpannableWithImages(newContentWithTags);
+ mNoteEditor.setText(spannable);
+
+ // 4. 然后将原始的[IMAGE]标签内容设置到WorkingNote中
+ mWorkingNote.setWorkingText(newContentWithTags);
+
+ // 5. 保存笔记(注意:这里不再调用getWorkingText(),而是直接保存WorkingNote中的内容)
+ // 我们需要修改saveNote()方法,或者直接调用mWorkingNote.saveNote()
+ if (mWorkingNote.saveNote()) {
+ showToast(R.string.info_image_inserted);
+ } else {
+ showToast(R.string.error_image_insert_failed);
+ }
- // 直接设置文本,不使用ImageSpan,这样可以确保内容被保存
- mNoteEditor.setText(newContent);
- mNoteEditor.setSelection(newContent.length());
+ // 6. 将光标移动到图片后面
+ int newPosition = originalSelection + imagePath.length();
+ mNoteEditor.setSelection(newPosition);
- // 保存笔记内容,确保图片路径被正确存储
- saveNote();
- showToast(R.string.info_image_inserted);
} catch (Exception e) {
Log.e(TAG, "Error inserting image: " + e.toString());
- showToast(R.string.error_note_not_exist);
+ e.printStackTrace();
+ // 如果插入图片失败,显示错误信息
+ showToast(R.string.error_image_insert_failed);
}
} else {
Log.e(TAG, "Image URI is null");
- showToast(R.string.error_note_not_exist);
+ showToast(R.string.error_image_insert_failed);
}
} else {
Log.d(TAG, "Image selection canceled or failed, resultCode: " + resultCode);
@@ -1498,38 +1551,38 @@ public class NoteEditActivity extends Activity implements OnClickListener,
} else {
// 加粗或斜体样式
StyleSpan[] styleSpans = editable.getSpans(selectionStart, selectionEnd, StyleSpan.class);
-
+
// 检查选中范围内是否已经有该样式
boolean hasStyle = false;
int existingStyle = 0;
-
+
// 收集选中范围内的所有样式
for (StyleSpan span : styleSpans) {
int spanStart = editable.getSpanStart(span);
int spanEnd = editable.getSpanEnd(span);
-
+
// 如果span与选中范围重叠
if (spanStart < selectionEnd && spanEnd > selectionStart) {
int spanStyle = span.getStyle();
existingStyle |= spanStyle;
-
+
// 检查是否包含目标样式
if ((spanStyle & styleType) == styleType) {
hasStyle = true;
}
}
}
-
+
// 移除选中范围内的所有StyleSpan,以便重新应用
for (int i = styleSpans.length - 1; i >= 0; i--) {
StyleSpan span = styleSpans[i];
int spanStart = editable.getSpanStart(span);
int spanEnd = editable.getSpanEnd(span);
-
+
// 如果span与选中范围重叠
if (spanStart < selectionEnd && spanEnd > selectionStart) {
editable.removeSpan(span);
-
+
// 处理span范围大于选中范围的情况,保留未选中部分的样式
int spanStyle = span.getStyle();
if (spanStart < selectionStart) {
@@ -1542,7 +1595,7 @@ public class NoteEditActivity extends Activity implements OnClickListener,
}
}
}
-
+
// 切换样式:如果有则移除,如果没有则添加
int finalStyle = existingStyle;
if (hasStyle) {
@@ -1552,7 +1605,7 @@ public class NoteEditActivity extends Activity implements OnClickListener,
// 添加目标样式
finalStyle |= styleType;
}
-
+
// 应用最终样式
if (finalStyle != 0) {
editable.setSpan(new StyleSpan(finalStyle), selectionStart, selectionEnd, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);