/*
* Copyright (C) 2010 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.ex.editstyledtext;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.HashMap;
import com.android.ex.editstyledtext.EditStyledText.EditModeActions.EditModeActionBase;
import com.android.ex.editstyledtext.EditStyledText.EditStyledTextSpans.HorizontalLineSpan;
import com.android.ex.editstyledtext.EditStyledText.EditStyledTextSpans.MarqueeSpan;
import com.android.ex.editstyledtext.EditStyledText.EditStyledTextSpans.RescalableImageSpan;
import android.R;
import android.app.AlertDialog;
import android.app.AlertDialog.Builder;
import android.content.Context;
import android.content.DialogInterface;
import android.content.DialogInterface.OnCancelListener;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Rect;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.Drawable;
import android.graphics.drawable.ShapeDrawable;
import android.graphics.drawable.shapes.RectShape;
import android.net.Uri;
import android.os.Bundle;
import android.os.Parcel;
import android.os.Parcelable;
import android.os.ResultReceiver;
import android.text.ClipboardManager;
import android.text.Editable;
import android.text.Html;
import android.text.Layout;
import android.text.NoCopySpan;
import android.text.NoCopySpan.Concrete;
import android.text.Selection;
import android.text.Spannable;
import android.text.SpannableStringBuilder;
import android.text.Spanned;
import android.text.TextPaint;
import android.text.Html.ImageGetter;
import android.text.Html.TagHandler;
import android.text.method.ArrowKeyMovementMethod;
import android.text.style.AbsoluteSizeSpan;
import android.text.style.AlignmentSpan;
import android.text.style.BackgroundColorSpan;
import android.text.style.CharacterStyle;
import android.text.style.DynamicDrawableSpan;
import android.text.style.ForegroundColorSpan;
import android.text.style.ImageSpan;
import android.text.style.ParagraphStyle;
import android.text.style.QuoteSpan;
import android.text.style.UnderlineSpan;
import android.util.AttributeSet;
import android.util.Log;
import android.view.ContextMenu;
import android.view.Gravity;
import android.view.KeyEvent;
import android.view.MenuItem;
import android.view.MotionEvent;
import android.view.View;
import android.view.inputmethod.EditorInfo;
import android.view.inputmethod.InputConnection;
import android.view.inputmethod.InputConnectionWrapper;
import android.view.inputmethod.InputMethodManager;
import android.widget.Button;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.TextView;
/**
* EditStyledText extends EditText for managing the flow and status to edit the styled text. This
* manages the states and flows of editing, supports inserting image, import/export HTML.
*/
public class EditStyledText extends EditText {
private static final String TAG = "EditStyledText";
/**
* DBG should be false at checking in.
*/
private static final boolean DBG = true;
/**
* Modes of editing actions.
*/
/** The mode that no editing action is done. */
public static final int MODE_NOTHING = 0;
/** The mode of copy. */
public static final int MODE_COPY = 1;
/** The mode of paste. */
public static final int MODE_PASTE = 2;
/** The mode of changing size. */
public static final int MODE_SIZE = 3;
/** The mode of changing color. */
public static final int MODE_COLOR = 4;
/** The mode of selection. */
public static final int MODE_SELECT = 5;
/** The mode of changing alignment. */
public static final int MODE_ALIGN = 6;
/** The mode of changing cut. */
public static final int MODE_CUT = 7;
public static final int MODE_TELOP = 8;
public static final int MODE_SWING = 9;
public static final int MODE_MARQUEE = 10;
public static final int MODE_SELECTALL = 11;
public static final int MODE_HORIZONTALLINE = 12;
public static final int MODE_STOP_SELECT = 13;
public static final int MODE_CLEARSTYLES = 14;
public static final int MODE_IMAGE = 15;
public static final int MODE_BGCOLOR = 16;
public static final int MODE_PREVIEW = 17;
public static final int MODE_CANCEL = 18;
public static final int MODE_TEXTVIEWFUNCTION = 19;
public static final int MODE_START_EDIT = 20;
public static final int MODE_END_EDIT = 21;
public static final int MODE_RESET = 22;
public static final int MODE_SHOW_MENU = 23;
/**
* States of selection.
*/
/** The state that selection isn't started. */
public static final int STATE_SELECT_OFF = 0;
/** The state that selection is started. */
public static final int STATE_SELECT_ON = 1;
/** The state that selection is done, but not fixed. */
public static final int STATE_SELECTED = 2;
/** The state that selection is done and not fixed. */
public static final int STATE_SELECT_FIX = 3;
/**
* Help message strings.
*/
public static final int HINT_MSG_NULL = 0;
public static final int HINT_MSG_COPY_BUF_BLANK = 1;
public static final int HINT_MSG_SELECT_START = 2;
public static final int HINT_MSG_SELECT_END = 3;
public static final int HINT_MSG_PUSH_COMPETE = 4;
public static final int HINT_MSG_BIG_SIZE_ERROR = 5;
public static final int HINT_MSG_END_PREVIEW = 6;
public static final int HINT_MSG_END_COMPOSE = 7;
/**
* Fixed Values.
*/
public static final int DEFAULT_TRANSPARENT_COLOR = 0x00FFFFFF;
public static final int DEFAULT_FOREGROUND_COLOR = 0xFF000000;
public static final char ZEROWIDTHCHAR = '\u2060';
public static final char IMAGECHAR = '\uFFFC';
private static final int ID_SELECT_ALL = android.R.id.selectAll;
private static final int ID_START_SELECTING_TEXT = android.R.id.startSelectingText;
private static final int ID_STOP_SELECTING_TEXT = android.R.id.stopSelectingText;
private static final int ID_PASTE = android.R.id.paste;
private static final int ID_COPY = android.R.id.copy;
private static final int ID_CUT = android.R.id.cut;
private static final int ID_HORIZONTALLINE = 0x00FFFF01;
private static final int ID_CLEARSTYLES = 0x00FFFF02;
private static final int ID_SHOWEDIT = 0x00FFFF03;
private static final int ID_HIDEEDIT = 0x00FFFF04;
private static final int MAXIMAGEWIDTHDIP = 300;
/**
* Strings for context menu. TODO: Extract the strings to strings.xml.
*/
private static CharSequence STR_HORIZONTALLINE;
private static CharSequence STR_CLEARSTYLES;
private static CharSequence STR_PASTE;
private float mPaddingScale = 0;
private ArrayList<EditStyledTextNotifier> mESTNotifiers;
private Drawable mDefaultBackground;
// EditStyledTextEditorManager manages the flow and status of each function of StyledText.
private EditorManager mManager;
private InputConnection mInputConnection;
private StyledTextConverter mConverter;
private StyledTextDialog mDialog;
private static final Concrete SELECTING = new NoCopySpan.Concrete();
private static final int PRESSED = Spannable.SPAN_MARK_MARK | (1 << Spannable.SPAN_USER_SHIFT);
/**
* EditStyledText extends EditText for managing flow of each editing action.
*/
public EditStyledText(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
init();
}
public EditStyledText(Context context, AttributeSet attrs) {
super(context, attrs);
init();
}
public EditStyledText(Context context) {
super(context);
init();
}
@Override
public boolean onTouchEvent(MotionEvent event) {
boolean superResult;
if (event.getAction() == MotionEvent.ACTION_UP) {
cancelLongPress();
boolean editting = isEditting();
// If View is touched but not in Edit Mode, starts Edit Mode.
if (!editting) {
onStartEdit();
}
int oldSelStart = Selection.getSelectionStart(getText());
int oldSelEnd = Selection.getSelectionEnd(getText());
superResult = super.onTouchEvent(event);
if (isFocused()) {
// If selection is started, don't open soft key by
// touching.
if (getSelectState() == STATE_SELECT_OFF) {
if (editting) {
mManager.showSoftKey(Selection.getSelectionStart(getText()),
Selection.getSelectionEnd(getText()));
} else {
mManager.showSoftKey(oldSelStart, oldSelEnd);
}
}
}
mManager.onCursorMoved();
mManager.unsetTextComposingMask();
} else {
superResult = super.onTouchEvent(event);
}
sendOnTouchEvent(event);
return superResult;
}
@Override
public Parcelable onSaveInstanceState() {
Parcelable superState = super.onSaveInstanceState();
SavedStyledTextState ss = new SavedStyledTextState(superState);
ss.mBackgroundColor = mManager.getBackgroundColor();
return ss;
}
@Override
public void onRestoreInstanceState(Parcelable state) {
if (!(state instanceof SavedStyledTextState)) {
super.onRestoreInstanceState(state);
return;
}
SavedStyledTextState ss = (SavedStyledTextState) state;
super.onRestoreInstanceState(ss.getSuperState());
setBackgroundColor(ss.mBackgroundColor);
}
@Override
protected void drawableStateChanged() {
super.drawableStateChanged();
if (mManager != null) {
mManager.onRefreshStyles();
}
}
@Override
public boolean onTextContextMenuItem(int id) {
boolean selection = getSelectionStart() != getSelectionEnd();
switch (id) {
case ID_SELECT_ALL:
onStartSelectAll();
return true;
case ID_START_SELECTING_TEXT:
onStartSelect();
mManager.blockSoftKey();
break;
case ID_STOP_SELECTING_TEXT:
onFixSelectedItem();
break;
case ID_PASTE:
onStartPaste();
return true;
case ID_COPY:
if (selection) {
onStartCopy();
} else {
mManager.onStartSelectAll(false);
onStartCopy();
}
return true;
case ID_CUT:
if (selection) {
onStartCut();
} else {
mManager.onStartSelectAll(false);
onStartCut();
}
return true;
case ID_HORIZONTALLINE:
onInsertHorizontalLine();
return true;
case ID_CLEARSTYLES:
onClearStyles();
return true;
case ID_SHOWEDIT:
onStartEdit();
return true;
case ID_HIDEEDIT:
onEndEdit();
return true;
}
return super.onTextContextMenuItem(id);
}
@Override
protected void onCreateContextMenu(ContextMenu menu) {
super.onCreateContextMenu(menu);
MenuHandler handler = new MenuHandler();
if (STR_HORIZONTALLINE != null) {
menu.add(0, ID_HORIZONTALLINE, 0, STR_HORIZONTALLINE).setOnMenuItemClickListener(
handler);
}
if (isStyledText() && STR_CLEARSTYLES != null) {
menu.add(0, ID_CLEARSTYLES, 0, STR_CLEARSTYLES)
.setOnMenuItemClickListener(handler);
}
if (mManager.canPaste()) {
menu.add(0, ID_PASTE, 0, STR_PASTE)
.setOnMenuItemClickListener(handler).setAlphabeticShortcut('v');
}
}
@Override
protected void onTextChanged(CharSequence text, int start, int before, int after) {
// onTextChanged will be called super's constructor.
if (mManager != null) {
mManager.updateSpanNextToCursor(getText(), start, before, after);
mManager.updateSpanPreviousFromCursor(getText(), start, before, after);
if (after > before) {
mManager.setTextComposingMask(start, start + after);
} else if (before < after) {
mManager.unsetTextComposingMask();
}
if (mManager.isWaitInput()) {
if (after > before) {
mManager.onCursorMoved();
onFixSelectedItem();
} else if (after < before) {
mManager.onAction(MODE_RESET);
}
}
}
super.onTextChanged(text, start, before, after);
}
@Override
public InputConnection onCreateInputConnection(EditorInfo outAttrs) {
mInputConnection =
new StyledTextInputConnection(super.onCreateInputConnection(outAttrs), this);
return mInputConnection;
}
@Override
protected void onFocusChanged(boolean focused, int direction, Rect previouslyFocusedRect) {
super.onFocusChanged(focused, direction, previouslyFocusedRect);
if (focused) {
onStartEdit();
} else if (!isButtonsFocused()) {
onEndEdit();
}
}
/**
* Initialize members.
*/
private void init() {
mConverter = new StyledTextConverter(this, new StyledTextHtmlStandard());
mDialog = new StyledTextDialog(this);
mManager = new EditorManager(this, mDialog);
setMovementMethod(new StyledTextArrowKeyMethod(mManager));
mDefaultBackground = getBackground();
requestFocus();
}
public interface StyledTextHtmlConverter {
public String toHtml(Spanned text);
public String toHtml(Spanned text, boolean escapeNonAsciiChar);
public String toHtml(Spanned text, boolean escapeNonAsciiChar, int width, float scale);
public Spanned fromHtml(String string);
public Spanned fromHtml(String source, ImageGetter imageGetter, TagHandler tagHandler);
}
public void setStyledTextHtmlConverter(StyledTextHtmlConverter html) {
mConverter.setStyledTextHtmlConverter(html);
}
/**
* EditStyledTextInterface provides functions for notifying messages to calling class.
*/
public interface EditStyledTextNotifier {
public void sendHintMsg(int msgId);
public void onStateChanged(int mode, int state);
public boolean sendOnTouchEvent(MotionEvent event);
public boolean isButtonsFocused();
public boolean showPreview();
public void cancelViewManager();
public boolean showInsertImageSelectAlertDialog();
public boolean showMenuAlertDialog();
}
/**
* Add Notifier.
*/
public void addEditStyledTextListener(EditStyledTextNotifier estInterface) {
if (mESTNotifiers == null) {
mESTNotifiers = new ArrayList<EditStyledTextNotifier>();
}
mESTNotifiers.add(estInterface);
}
/**
* Remove Notifier.
*/
public void removeEditStyledTextListener(EditStyledTextNotifier estInterface) {
if (mESTNotifiers != null) {
int i = mESTNotifiers.indexOf(estInterface);
if (i > 0) {
mESTNotifiers.remove(i);
}
}
}
private void sendOnTouchEvent(MotionEvent event) {
if (mESTNotifiers != null) {
for (EditStyledTextNotifier notifier : mESTNotifiers) {
notifier.sendOnTouchEvent(event);
}
}
}
public boolean isButtonsFocused() {
boolean retval = false;
if (mESTNotifiers != null) {
for (EditStyledTextNotifier notifier : mESTNotifiers) {
retval |= notifier.isButtonsFocused();
}
}
return retval;
}
private void showPreview() {
if (mESTNotifiers != null) {
for (EditStyledTextNotifier notifier : mESTNotifiers) {
if (notifier.showPreview()) {
break;
}
}
}
}
private void cancelViewManagers() {
if (mESTNotifiers != null) {
for (EditStyledTextNotifier notifier : mESTNotifiers) {
notifier.cancelViewManager();
}
}
}
private void showInsertImageSelectAlertDialog() {
if (mESTNotifiers != null) {
for (EditStyledTextNotifier notifier : mESTNotifiers) {
if (notifier.showInsertImageSelectAlertDialog()) {
break;
}
}
}
}
private void showMenuAlertDialog() {
if (mESTNotifiers != null) {
for (EditStyledTextNotifier notifier : mESTNotifiers) {
if (notifier.showMenuAlertDialog()) {
break;
}
}
}
}
/**
* Notify hint messages what action is expected to calling class.
*
* @param msgId Id of the hint message.
*/
private void sendHintMessage(int msgId) {
if (mESTNotifiers != null) {
for (EditStyledTextNotifier notifier : mESTNotifiers) {
notifier.sendHintMsg(msgId);
}
}
}
/**
* Notify the event that the mode and state are changed.
*
* @param mode Mode of the editing action.
* @param state Mode of the selection state.
*/
private void notifyStateChanged(int mode, int state) {
if (mESTNotifiers != null) {
for (EditStyledTextNotifier notifier : mESTNotifiers) {
notifier.onStateChanged(mode, state);
}
}
}
/** Start to edit styled text */
public void onStartEdit() {
mManager.onAction(MODE_START_EDIT);
}
/** End of editing styled text */
public void onEndEdit() {
mManager.onAction(MODE_END_EDIT);
}
public void onResetEdit() {
mManager.onAction(MODE_RESET);
}
/** Start to copy styled text */
public void onStartCopy() {
mManager.onAction(MODE_COPY);
}
/** Start to cut styled text */
public void onStartCut() {
mManager.onAction(MODE_CUT);
}
/** Start to paste styled text */
public void onStartPaste() {
mManager.onAction(MODE_PASTE);
}
/** Start to change size */
public void onStartSize() {
mManager.onAction(MODE_SIZE);
}
/** Start to change color */
public void onStartColor() {
mManager.onAction(MODE_COLOR);
}
/** Start to change background color */
public void onStartBackgroundColor() {
mManager.onAction(MODE_BGCOLOR);
}
/** Start to change Alignment */
public void onStartAlign() {
mManager.onAction(MODE_ALIGN);
}
public void onStartTelop() {
mManager.onAction(MODE_TELOP);
}
public void onStartSwing() {
mManager.onAction(MODE_SWING);
}
public void onStartMarquee() {
mManager.onAction(MODE_MARQUEE);
}
/** Start to select a text */
public void onStartSelect() {
mManager.onStartSelect(true);
}
/** Start to select all characters */
public void onStartSelectAll() {
mManager.onStartSelectAll(true);
}
public void onStartShowPreview() {
mManager.onAction(MODE_PREVIEW);
}
public void onStartShowMenuAlertDialog() {
mManager.onStartShowMenuAlertDialog();
}
public void onStartAction(int mode, boolean notifyStateChanged) {
mManager.onAction(mode, notifyStateChanged);
}
/** Fix selection */
public void onFixSelectedItem() {
mManager.onFixSelectedItem();
}
public void onInsertImage() {
mManager.onAction(MODE_IMAGE);
}
/**
* InsertImage to TextView by using URI
*
* @param uri URI of the iamge inserted to TextView.
*/
public void onInsertImage(Uri uri) {
mManager.onInsertImage(uri);
}
/**
* InsertImage to TextView by using resource ID
*
* @param resId Resource ID of the iamge inserted to TextView.
*/
public void onInsertImage(int resId) {
mManager.onInsertImage(resId);
}
public void onInsertHorizontalLine() {
mManager.onAction(MODE_HORIZONTALLINE);
}
public void onClearStyles() {
mManager.onClearStyles();
}
public void onBlockSoftKey() {
mManager.blockSoftKey();
}
public void onUnblockSoftKey() {
mManager.unblockSoftKey();
}
public void onCancelViewManagers() {
mManager.onCancelViewManagers();
}
private void onRefreshStyles() {
mManager.onRefreshStyles();
}
private void onRefreshZeoWidthChar() {
mManager.onRefreshZeoWidthChar();
}
/**
* Set Size of the Item.
*
* @param size The size of the Item.
*/
public void setItemSize(int size) {
mManager.setItemSize(size, true);
}
/**
* Set Color of the Item.
*
* @param color The color of the Item.
*/
public void setItemColor(int color) {
mManager.setItemColor(color, true);
}
/**
* Set Alignment of the Item.
*
* @param align The color of the Item.
*/
public void setAlignment(Layout.Alignment align) {
mManager.setAlignment(align);
}
/**
* Set Background color of View.
*
* @param color The background color of view.
*/
@Override
public void setBackgroundColor(int color) {
if (color != DEFAULT_TRANSPARENT_COLOR) {
super.setBackgroundColor(color);
} else {
setBackgroundDrawable(mDefaultBackground);
}
mManager.setBackgroundColor(color);
onRefreshStyles();
}
public void setMarquee(int marquee) {
mManager.setMarquee(marquee);
}
/**
* Set html to EditStyledText.
*
* @param html The html to be set.
*/
public void setHtml(String html) {
mConverter.SetHtml(html);
}
/**
* Set Builder for AlertDialog.
*
* @param builder Builder for opening Alert Dialog.
*/
public void setBuilder(Builder builder) {
mDialog.setBuilder(builder);
}
/**
* Set Parameters for ColorAlertDialog.
*
* @param colortitle Title for Alert Dialog.
* @param colornames List of name of selecting color.
* @param colorints List of int of color.
*/
public void setColorAlertParams(CharSequence colortitle, CharSequence[] colornames,
CharSequence[] colorints, CharSequence transparent) {
mDialog.setColorAlertParams(colortitle, colornames, colorints, transparent);
}
/**
* Set Parameters for SizeAlertDialog.
*
* @param sizetitle Title for Alert Dialog.
* @param sizenames List of name of selecting size.
* @param sizedisplayints List of int of size displayed in TextView.
* @param sizesendints List of int of size exported to HTML.
*/
public void setSizeAlertParams(CharSequence sizetitle, CharSequence[] sizenames,
CharSequence[] sizedisplayints, CharSequence[] sizesendints) {
mDialog.setSizeAlertParams(sizetitle, sizenames, sizedisplayints, sizesendints);
}
public void setAlignAlertParams(CharSequence aligntitle, CharSequence[] alignnames) {
mDialog.setAlignAlertParams(aligntitle, alignnames);
}
public void setMarqueeAlertParams(CharSequence marqueetitle, CharSequence[] marqueenames) {
mDialog.setMarqueeAlertParams(marqueetitle, marqueenames);
}
public void setContextMenuStrings(CharSequence horizontalline, CharSequence clearstyles,
CharSequence paste) {
STR_HORIZONTALLINE = horizontalline;
STR_CLEARSTYLES = clearstyles;
STR_PASTE = paste;
}
/**
* Check whether editing is started or not.
*
* @return Whether editing is started or not.
*/
public boolean isEditting() {
return mManager.isEditting();
}
/**
* Check whether styled text or not.
*
* @return Whether styled text or not.
*/
public boolean isStyledText() {
return mManager.isStyledText();
}
/**
* Check whether SoftKey is Blocked or not.
*
* @return whether SoftKey is Blocked or not.
*/
public boolean isSoftKeyBlocked() {
return mManager.isSoftKeyBlocked();
}
/**
* Get the mode of the action.
*
* @return The mode of the action.
*/
public int getEditMode() {
return mManager.getEditMode();
}
/**
* Get the state of the selection.
*
* @return The state of the selection.
*/
public int getSelectState() {
return mManager.getSelectState();
}
/**
* Get the state of the selection.
*
* @return The state of the selection.
*/
public String getHtml() {
return mConverter.getHtml(true);
}
public String getHtml(boolean escapeFlag) {
return mConverter.getHtml(escapeFlag);
}
/**
* Get the state of the selection.
*
* @param uris The array of used uris.
* @return The state of the selection.
*/
public String getHtml(ArrayList<Uri> uris, boolean escapeFlag) {
mConverter.getUriArray(uris, getText());
return mConverter.getHtml(escapeFlag);
}
public String getPreviewHtml() {
return mConverter.getPreviewHtml();
}
/**
* Get Background color of View.
*
* @return The background color of View.
*/
public int getBackgroundColor() {
return mManager.getBackgroundColor();
}
public EditorManager getEditStyledTextManager() {
return mManager;
}
/**
* Get Foreground color of View.
*
* @return The background color of View.
*/
public int getForegroundColor(int pos) {
if (pos < 0 || pos > getText().length()) {
return DEFAULT_FOREGROUND_COLOR;
} else {
ForegroundColorSpan[] spans =
getText().getSpans(pos, pos, ForegroundColorSpan.class);
if (spans.length > 0) {
return spans[0].getForegroundColor();
} else {
return DEFAULT_FOREGROUND_COLOR;
}
}
}
private void finishComposingText() {
if (mInputConnection != null && !mManager.mTextIsFinishedFlag) {
mInputConnection.finishComposingText();
mManager.mTextIsFinishedFlag = true;
}
}
private float getPaddingScale() {
if (mPaddingScale <= 0) {
mPaddingScale = getContext().getResources().getDisplayMetrics().density;
}
return mPaddingScale;
}
/** Convert pixcel to DIP */
private int dipToPx(int dip) {
if (mPaddingScale <= 0) {
mPaddingScale = getContext().getResources().getDisplayMetrics().density;
}
return (int) ((float) dip * getPaddingScale() + 0.5);
}
private int getMaxImageWidthDip() {
return MAXIMAGEWIDTHDIP;
}
private int getMaxImageWidthPx() {
return dipToPx(MAXIMAGEWIDTHDIP);
}
public void addAction(int mode, EditModeActionBase action) {
mManager.addAction(mode, action);
}
public void addInputExtra(boolean create, String extra) {
Bundle bundle = super.getInputExtras(create);
if (bundle != null) {
bundle.putBoolean(extra, true);
}
}
private static void startSelecting(View view, Spannable content) {
content.setSpan(SELECTING, 0, 0, PRESSED);
}
private static void stopSelecting(View view, Spannable content) {
content.removeSpan(SELECTING);
}
/**
* EditorManager manages the flow and status of editing actions.
*/
private class EditorManager {
static final private String LOG_TAG = "EditStyledText.EditorManager";
private boolean mEditFlag = false;
private boolean mSoftKeyBlockFlag = false;
private boolean mKeepNonLineSpan = false;
private boolean mWaitInputFlag = false;
private boolean mTextIsFinishedFlag = false;
private int mMode = MODE_NOTHING;
private int mState = STATE_SELECT_OFF;
private int mCurStart = 0;
private int mCurEnd = 0;
private int mColorWaitInput = DEFAULT_TRANSPARENT_COLOR;
private int mSizeWaitInput = 0;
private int mBackgroundColor = DEFAULT_TRANSPARENT_COLOR;
private BackgroundColorSpan mComposingTextMask;
private EditStyledText mEST;
private EditModeActions mActions;
private SoftKeyReceiver mSkr;
private SpannableStringBuilder mCopyBuffer;
EditorManager(EditStyledText est, StyledTextDialog dialog) {
mEST = est;
mActions = new EditModeActions(mEST, this, dialog);
mSkr = new SoftKeyReceiver(mEST);
}
public void addAction(int mode, EditModeActionBase action) {
mActions.addAction(mode, action);
}
public void onAction(int mode) {
onAction(mode, true);
}
public void onAction(int mode, boolean notifyStateChanged) {
mActions.onAction(mode);
if (notifyStateChanged) {
mEST.notifyStateChanged(mMode, mState);
}
}
private void startEdit() {
resetEdit();
showSoftKey();
}
public void onStartSelect(boolean notifyStateChanged) {
if (DBG) {
Log.d(LOG_TAG, "--- onClickSelect");
}
mMode = MODE_SELECT;
if (mState == STATE_SELECT_OFF) {
mActions.onSelectAction();
} else {
unsetSelect();
mActions.onSelectAction();
}
if (notifyStateChanged) {
mEST.notifyStateChanged(mMode, mState);
}
}
public void onCursorMoved() {
if (DBG) {
Log.d(LOG_TAG, "--- onClickView");
}
if (mState == STATE_SELECT_ON || mState == STATE_SELECTED) {
mActions.onSelectAction();
mEST.notifyStateChanged(mMode, mState);
}
}
public void onStartSelectAll(boolean notifyStateChanged) {
if (DBG) {
Log.d(LOG_TAG, "--- onClickSelectAll");
}
handleSelectAll();
if (notifyStateChanged) {
mEST.notifyStateChanged(mMode, mState);
}
}
public void onStartShowMenuAlertDialog() {
mActions.onAction(MODE_SHOW_MENU);
// don't call notify state changed because it have to continue
// to the next action.
// mEST.notifyStateChanged(mMode, mState);
}
public void onFixSelectedItem() {
if (DBG) {
Log.d(LOG_TAG, "--- onFixSelectedItem");
}
fixSelectionAndDoNextAction();
mEST.notifyStateChanged(mMode, mState);
}
public void onInsertImage(Uri uri) {
mActions.onAction(MODE_IMAGE, uri);
mEST.notifyStateChanged(mMode, mState);
}
public void onInsertImage(int resId) {
mActions.onAction(MODE_IMAGE, resId);
mEST.notifyStateChanged(mMode, mState);
}
private void insertImageFromUri(Uri uri) {
insertImageSpan(new EditStyledTextSpans.RescalableImageSpan(mEST.getContext(),
uri, mEST.getMaxImageWidthPx()), mEST.getSelectionStart());
}
private void insertImageFromResId(int resId) {
insertImageSpan(new EditStyledTextSpans.RescalableImageSpan(mEST.getContext(),
resId, mEST.getMaxImageWidthDip()), mEST.getSelectionStart());
}
private void insertHorizontalLine() {
if (DBG) {
Log.d(LOG_TAG, "--- onInsertHorizontalLine:");
}
int curpos = mEST.getSelectionStart();
if (curpos > 0 && mEST.getText().charAt(curpos - 1) != '\n') {
mEST.getText().insert(curpos++, "\n");
}
insertImageSpan(
new HorizontalLineSpan(0xFF000000, mEST.getWidth(), mEST.getText()),
curpos++);
mEST.getText().insert(curpos++, "\n");
mEST.setSelection(curpos);
mEST.notifyStateChanged(mMode, mState);
}
private void clearStyles(CharSequence txt) {
if (DBG) {
Log.d("EditStyledText", "--- onClearStyles");
}
int len = txt.length();
if (txt instanceof Editable) {
Editable editable = (Editable) txt;
Object[] styles = editable.getSpans(0, len, Object.class);
for (Object style : styles) {
if (style instanceof ParagraphStyle || style instanceof QuoteSpan
|| style instanceof CharacterStyle
&& !(style instanceof UnderlineSpan)) {
if (style instanceof ImageSpan || style instanceof HorizontalLineSpan) {
int start = editable.getSpanStart(style);
int end = editable.getSpanEnd(style);
editable.replace(start, end, "");
}
editable.removeSpan(style);
}
}
}
}
public void onClearStyles() {
mActions.onAction(MODE_CLEARSTYLES);
}
public void onCancelViewManagers() {
mActions.onAction(MODE_CANCEL);
}
private void clearStyles() {
if (DBG) {
Log.d(LOG_TAG, "--- onClearStyles");
}
clearStyles(mEST.getText());
mEST.setBackgroundDrawable(mEST.mDefaultBackground);
mBackgroundColor = DEFAULT_TRANSPARENT_COLOR;
onRefreshZeoWidthChar();
}
public void onRefreshZeoWidthChar() {
Editable txt = mEST.getText();
for (int i = 0; i < txt.length(); i++) {
if (txt.charAt(i) == ZEROWIDTHCHAR) {
txt.replace(i, i + 1, "");
i--;
}
}
}
public void onRefreshStyles() {
if (DBG) {
Log.d(LOG_TAG, "--- onRefreshStyles");
}
Editable txt = mEST.getText();
int len = txt.length();
int width = mEST.getWidth();
HorizontalLineSpan[] lines = txt.getSpans(0, len, HorizontalLineSpan.class);
for (HorizontalLineSpan line : lines) {
line.resetWidth(width);
}
MarqueeSpan[] marquees = txt.getSpans(0, len, MarqueeSpan.class);
for (MarqueeSpan marquee : marquees) {
marquee.resetColor(mEST.getBackgroundColor());
}
if (lines.length > 0) {
// This is hack, bad needed for renewing View
// by inserting new line.
txt.replace(0, 1, "" + txt.charAt(0));
}
}
public void setBackgroundColor(int color) {
mBackgroundColor = color;
}
public void setItemSize(int size, boolean reset) {
if (DBG) {
Log.d(LOG_TAG, "--- setItemSize");
}
if (isWaitingNextAction()) {
mSizeWaitInput = size;
} else if (mState == STATE_SELECTED || mState == STATE_SELECT_FIX) {
if (size > 0) {
changeSizeSelectedText(size);
}
if (reset) {
resetEdit();
}
}
}
public void setItemColor(int color, boolean reset) {
if (DBG) {
Log.d(LOG_TAG, "--- setItemColor");
}
if (isWaitingNextAction()) {
mColorWaitInput = color;
} else if (mState == STATE_SELECTED || mState == STATE_SELECT_FIX) {
if (color != DEFAULT_TRANSPARENT_COLOR) {
changeColorSelectedText(color);
}
if (reset) {
resetEdit();
}
}
}
public void setAlignment(Layout.Alignment align) {
if (mState == STATE_SELECTED || mState == STATE_SELECT_FIX) {
changeAlign(align);
resetEdit();
}
}
public void setTelop() {
if (mState == STATE_SELECTED || mState == STATE_SELECT_FIX) {
addTelop();
resetEdit();
}
}
public void setSwing() {
if (mState == STATE_SELECTED || mState == STATE_SELECT_FIX) {
addSwing();
resetEdit();
}
}
public void setMarquee(int marquee) {
if (mState == STATE_SELECTED || mState == STATE_SELECT_FIX) {
addMarquee(marquee);
resetEdit();
}
}
public void setTextComposingMask(int start, int end) {
if (DBG) {
Log.d(TAG, "--- setTextComposingMask:" + start + "," + end);
}
int min = Math.min(start, end);
int max = Math.max(start, end);
int foregroundColor;
if (isWaitInput() && mColorWaitInput != DEFAULT_TRANSPARENT_COLOR) {
foregroundColor = mColorWaitInput;
} else {
foregroundColor = mEST.getForegroundColor(min);
}
int backgroundColor = mEST.getBackgroundColor();
if (DBG) {
Log.d(TAG,
"--- fg:" + Integer.toHexString(foregroundColor) + ",bg:"
+ Integer.toHexString(backgroundColor) + "," + isWaitInput()
+ "," + "," + mMode);
}
if (foregroundColor == backgroundColor) {
int maskColor = 0x80000000 | ~(backgroundColor | 0xFF000000);
if (mComposingTextMask == null
|| mComposingTextMask.getBackgroundColor() != maskColor) {
mComposingTextMask = new BackgroundColorSpan(maskColor);
}
mEST.getText().setSpan(mComposingTextMask, min, max,
Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
}
}
private void setEditMode(int mode) {
mMode = mode;
}
private void setSelectState(int state) {
mState = state;
}
public void unsetTextComposingMask() {
if (DBG) {
Log.d(TAG, "--- unsetTextComposingMask");
}
if (mComposingTextMask != null) {
mEST.getText().removeSpan(mComposingTextMask);
mComposingTextMask = null;
}
}
public boolean isEditting() {
return mEditFlag;
}
/* If the style of the span is added, add check case for that style */
public boolean isStyledText() {
Editable txt = mEST.getText();
int len = txt.length();
if (txt.getSpans(0, len, ParagraphStyle.class).length > 0
|| txt.getSpans(0, len, QuoteSpan.class).length > 0
|| txt.getSpans(0, len, CharacterStyle.class).length > 0
|| mBackgroundColor != DEFAULT_TRANSPARENT_COLOR) {
return true;
}
return false;
}
public boolean isSoftKeyBlocked() {
return mSoftKeyBlockFlag;
}
public boolean isWaitInput() {
return mWaitInputFlag;
}
public int getBackgroundColor() {
return mBackgroundColor;
}
public int getEditMode() {
return mMode;
}
public int getSelectState() {
return mState;
}
public int getSelectionStart() {
return mCurStart;
}
public int getSelectionEnd() {
return mCurEnd;
}
public int getSizeWaitInput() {
return mSizeWaitInput;
}
public int getColorWaitInput() {
return mColorWaitInput;
}
private void setInternalSelection(int curStart, int curEnd) {
mCurStart = curStart;
mCurEnd = curEnd;
}
public void
updateSpanPreviousFromCursor(Editable txt, int start, int before, int after) {
if (DBG) {
Log.d(LOG_TAG, "updateSpanPrevious:" + start + "," + before + "," + after);
}
int end = start + after;
int min = Math.min(start, end);
int max = Math.max(start, end);
Object[] spansBefore = txt.getSpans(min, min, Object.class);
for (Object span : spansBefore) {
if (span instanceof ForegroundColorSpan || span instanceof AbsoluteSizeSpan
|| span instanceof MarqueeSpan || span instanceof AlignmentSpan) {
int spanstart = txt.getSpanStart(span);
int spanend = txt.getSpanEnd(span);
if (DBG) {
Log.d(LOG_TAG, "spantype:" + span.getClass() + "," + spanstart);
}
int tempmax = max;
if (span instanceof MarqueeSpan || span instanceof AlignmentSpan) {
// Line Span
tempmax = findLineEnd(mEST.getText(), max);
} else {
if (mKeepNonLineSpan) {
tempmax = spanend;
}
}
if (spanend < tempmax) {
if (DBG) {
Log.d(LOG_TAG, "updateSpanPrevious: extend span");
}
txt.setSpan(span, spanstart, tempmax,
Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
}
} else if (span instanceof HorizontalLineSpan) {
int spanstart = txt.getSpanStart(span);
int spanend = txt.getSpanEnd(span);
if (before > after) {
// When text is deleted just after horizontalLineSpan, horizontalLineSpan
// has to be deleted, because the charactor just after horizontalLineSpan
// is '\n'.
txt.replace(spanstart, spanend, "");
txt.removeSpan(span);
} else {
// When text is added just after horizontalLineSpan add '\n' just after
// horizontalLineSpan.
if (spanend == end && end < txt.length()
&& mEST.getText().charAt(end) != '\n') {
mEST.getText().insert(end, "\n");
}
}
}
}
}
public void updateSpanNextToCursor(Editable txt, int start, int before, int after) {
if (DBG) {
Log.d(LOG_TAG, "updateSpanNext:" + start + "," + before + "," + after);
}
int end = start + after;
int min = Math.min(start, end);
int max = Math.max(start, end);
Object[] spansAfter = txt.getSpans(max, max, Object.class);
for (Object span : spansAfter) {
if (span instanceof MarqueeSpan || span instanceof AlignmentSpan) {
int spanstart = txt.getSpanStart(span);
int spanend = txt.getSpanEnd(span);
if (DBG) {
Log.d(LOG_TAG, "spantype:" + span.getClass() + "," + spanend);
}
int tempmin = min;
if (span instanceof MarqueeSpan || span instanceof AlignmentSpan) {
tempmin = findLineStart(mEST.getText(), min);
}
if (tempmin < spanstart && before > after) {
txt.removeSpan(span);
} else if (spanstart > min) {
txt.setSpan(span, min, spanend, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
}
} else if (span instanceof HorizontalLineSpan) {
int spanstart = txt.getSpanStart(span);
// Whene text is changed just before horizontalLineSpan and there is no '\n'
// just before horizontalLineSpan add '\n'
if (spanstart == end && end > 0 && mEST.getText().charAt(end - 1) != '\n') {
mEST.getText().insert(end, "\n");
mEST.setSelection(end);
}
}
}
}
/** canPaste returns true only if ClipboardManager doen't contain text. */
public boolean canPaste() {
return (mCopyBuffer != null && mCopyBuffer.length() > 0 && removeImageChar(
mCopyBuffer).length() == 0);
}
private void endEdit() {
if (DBG) {
Log.d(LOG_TAG, "--- handleCancel");
}
mMode = MODE_NOTHING;
mState = STATE_SELECT_OFF;
mEditFlag = false;
mColorWaitInput = DEFAULT_TRANSPARENT_COLOR;
mSizeWaitInput = 0;
mWaitInputFlag = false;
mSoftKeyBlockFlag = false;
mKeepNonLineSpan = false;
mTextIsFinishedFlag = false;
unsetSelect();
mEST.setOnClickListener(null);
unblockSoftKey();
}
private void fixSelectionAndDoNextAction() {
if (DBG) {
Log.d(LOG_TAG, "--- handleComplete:" + mCurStart + "," + mCurEnd);
}
if (!mEditFlag) {
return;
}
if (mCurStart == mCurEnd) {
if (DBG) {
Log.d(LOG_TAG, "--- cancel handle complete:" + mCurStart);
}
resetEdit();
return;
}
if (mState == STATE_SELECTED) {
mState = STATE_SELECT_FIX;
}
// When the complete button is clicked, this should do action.
mActions.doNext(mMode);
//MetaKeyKeyListener.stopSelecting(mEST, mEST.getText());
stopSelecting(mEST, mEST.getText());
}
// Remove obj character for DynamicDrawableSpan from clipboard.
private SpannableStringBuilder removeImageChar(SpannableStringBuilder text) {
SpannableStringBuilder buf = new SpannableStringBuilder(text);
DynamicDrawableSpan[] styles =
buf.getSpans(0, buf.length(), DynamicDrawableSpan.class);
for (DynamicDrawableSpan style : styles) {
if (style instanceof HorizontalLineSpan
|| style instanceof RescalableImageSpan) {
int start = buf.getSpanStart(style);
int end = buf.getSpanEnd(style);
buf.replace(start, end, "");
}
}
return buf;
}
private void copyToClipBoard() {
int min = Math.min(getSelectionStart(), getSelectionEnd());
int max = Math.max(getSelectionStart(), getSelectionEnd());
mCopyBuffer = (SpannableStringBuilder) mEST.getText().subSequence(min, max);
SpannableStringBuilder clipboardtxt = removeImageChar(mCopyBuffer);
ClipboardManager clip =
(ClipboardManager) getContext()
.getSystemService(Context.CLIPBOARD_SERVICE);
clip.setText(clipboardtxt);
if (DBG) {
dumpSpannableString(clipboardtxt);
dumpSpannableString(mCopyBuffer);
}
}
private void cutToClipBoard() {
copyToClipBoard();
int min = Math.min(getSelectionStart(), getSelectionEnd());
int max = Math.max(getSelectionStart(), getSelectionEnd());
mEST.getText().delete(min, max);
}
private boolean isClipBoardChanged(CharSequence clipboardText) {
if (DBG) {
Log.d(TAG, "--- isClipBoardChanged:" + clipboardText);
}
if (mCopyBuffer == null) {
return true;
}
int len = clipboardText.length();
CharSequence removedClipBoard = removeImageChar(mCopyBuffer);
if (DBG) {
Log.d(TAG, "--- clipBoard:" + len + "," + removedClipBoard + clipboardText);
}
if (len != removedClipBoard.length()) {
return true;
}
for (int i = 0; i < len; ++i) {
if (clipboardText.charAt(i) != removedClipBoard.charAt(i)) {
return true;
}
}
return false;
}
private void pasteFromClipboard() {
int min = Math.min(mEST.getSelectionStart(), mEST.getSelectionEnd());
int max = Math.max(mEST.getSelectionStart(), mEST.getSelectionEnd());
// TODO: Find more smart way to set Span to Clipboard.
Selection.setSelection(mEST.getText(), max);
ClipboardManager clip =
(ClipboardManager) getContext()
.getSystemService(Context.CLIPBOARD_SERVICE);
mKeepNonLineSpan = true;
mEST.getText().replace(min, max, clip.getText());
if (!isClipBoardChanged(clip.getText())) {
if (DBG) {
Log.d(TAG, "--- handlePaste: startPasteImage");
}
DynamicDrawableSpan[] styles =
mCopyBuffer.getSpans(0, mCopyBuffer.length(),
DynamicDrawableSpan.class);
for (DynamicDrawableSpan style : styles) {
int start = mCopyBuffer.getSpanStart(style);
if (style instanceof HorizontalLineSpan) {
insertImageSpan(new HorizontalLineSpan(0xFF000000, mEST.getWidth(),
mEST.getText()), min + start);
} else if (style instanceof RescalableImageSpan) {
insertImageSpan(
new RescalableImageSpan(mEST.getContext(),
((RescalableImageSpan) style).getContentUri(),
mEST.getMaxImageWidthPx()), min + start);
}
}
}
}
private void handleSelectAll() {
if (!mEditFlag) {
return;
}
mActions.onAction(MODE_SELECTALL);
}
private void selectAll() {
Selection.selectAll(mEST.getText());
mCurStart = mEST.getSelectionStart();
mCurEnd = mEST.getSelectionEnd();
mMode = MODE_SELECT;
mState = STATE_SELECT_FIX;
}
private void resetEdit() {
endEdit();
mEditFlag = true;
mEST.notifyStateChanged(mMode, mState);
}
private void setSelection() {
if (DBG) {
Log.d(LOG_TAG, "--- onSelect:" + mCurStart + "," + mCurEnd);
}
if (mCurStart >= 0 && mCurStart <= mEST.getText().length() && mCurEnd >= 0
&& mCurEnd <= mEST.getText().length()) {
if (mCurStart < mCurEnd) {
mEST.setSelection(mCurStart, mCurEnd);
mState = STATE_SELECTED;
} else if (mCurStart > mCurEnd) {
mEST.setSelection(mCurEnd, mCurStart);
mState = STATE_SELECTED;
} else {
mState = STATE_SELECT_ON;
}
} else {
Log.e(LOG_TAG, "Select is on, but cursor positions are illigal.:"
+ mEST.getText().length() + "," + mCurStart + "," + mCurEnd);
}
}
private void unsetSelect() {
if (DBG) {
Log.d(LOG_TAG, "--- offSelect");
}
//MetaKeyKeyListener.stopSelecting(mEST, mEST.getText());
stopSelecting(mEST, mEST.getText());
int currpos = mEST.getSelectionStart();
mEST.setSelection(currpos, currpos);
mState = STATE_SELECT_OFF;
}
private void setSelectStartPos() {
if (DBG) {
Log.d(LOG_TAG, "--- setSelectStartPos");
}
mCurStart = mEST.getSelectionStart();
mState = STATE_SELECT_ON;
}
private void setSelectEndPos() {
if (mEST.getSelectionEnd() == mCurStart) {
setEndPos(mEST.getSelectionStart());
} else {
setEndPos(mEST.getSelectionEnd());
}
}
public void setEndPos(int pos) {
if (DBG) {
Log.d(LOG_TAG, "--- setSelectedEndPos:" + pos);
}
mCurEnd = pos;
setSelection();
}
private boolean isWaitingNextAction() {
if (DBG) {
Log.d(LOG_TAG, "--- waitingNext:" + mCurStart + "," + mCurEnd + "," + mState);
}
if (mCurStart == mCurEnd && mState == STATE_SELECT_FIX) {
waitSelection();
return true;
} else {
resumeSelection();
return false;
}
}
private void waitSelection() {
if (DBG) {
Log.d(LOG_TAG, "--- waitSelection");
}
mWaitInputFlag = true;
if (mCurStart == mCurEnd) {
mState = STATE_SELECT_ON;
} else {
mState = STATE_SELECTED;
}
//MetaKeyKeyListener.startSelecting(mEST, mEST.getText());
startSelecting(mEST, mEST.getText());
}
private void resumeSelection() {
if (DBG) {
Log.d(LOG_TAG, "--- resumeSelection");
}
mWaitInputFlag = false;
mState = STATE_SELECT_FIX;
//MetaKeyKeyListener.stopSelecting(mEST, mEST.getText());
stopSelecting(mEST, mEST.getText());
}
private boolean isTextSelected() {
return (mState == STATE_SELECTED || mState == STATE_SELECT_FIX);
}
private void setStyledTextSpan(Object span, int start, int end) {
if (DBG) {
Log.d(LOG_TAG, "--- setStyledTextSpan:" + mMode + "," + start + "," + end);
}
int min = Math.min(start, end);
int max = Math.max(start, end);
mEST.getText().setSpan(span, min, max, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
Selection.setSelection(mEST.getText(), max);
}
private void setLineStyledTextSpan(Object span) {
int min = Math.min(mCurStart, mCurEnd);
int max = Math.max(mCurStart, mCurEnd);
int current = mEST.getSelectionStart();
int start = findLineStart(mEST.getText(), min);
int end = findLineEnd(mEST.getText(), max);
if (start == end) {
mEST.getText().insert(end, "\n");
setStyledTextSpan(span, start, end + 1);
} else {
setStyledTextSpan(span, start, end);
}
Selection.setSelection(mEST.getText(), current);
}
private void changeSizeSelectedText(int size) {
if (mCurStart != mCurEnd) {
setStyledTextSpan(new AbsoluteSizeSpan(size), mCurStart, mCurEnd);
} else {
Log.e(LOG_TAG, "---changeSize: Size of the span is zero");
}
}
private void changeColorSelectedText(int color) {
if (mCurStart != mCurEnd) {
setStyledTextSpan(new ForegroundColorSpan(color), mCurStart, mCurEnd);
} else {
Log.e(LOG_TAG, "---changeColor: Size of the span is zero");
}
}
private void changeAlign(Layout.Alignment align) {
setLineStyledTextSpan(new AlignmentSpan.Standard(align));
}
private void addTelop() {
addMarquee(MarqueeSpan.ALTERNATE);
}
private void addSwing() {
addMarquee(MarqueeSpan.SCROLL);
}
private void addMarquee(int marquee) {
if (DBG) {
Log.d(LOG_TAG, "--- addMarquee:" + marquee);
}
setLineStyledTextSpan(new MarqueeSpan(marquee, mEST.getBackgroundColor()));
}
private void insertImageSpan(DynamicDrawableSpan span, int curpos) {
if (DBG) {
Log.d(LOG_TAG, "--- insertImageSpan:");
}
if (span != null && span.getDrawable() != null) {
mEST.getText().insert(curpos, "" + IMAGECHAR);
mEST.getText().setSpan(span, curpos, curpos + 1,
Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
mEST.notifyStateChanged(mMode, mState);
} else {
Log.e(LOG_TAG, "--- insertImageSpan: null span was inserted");
mEST.sendHintMessage(HINT_MSG_BIG_SIZE_ERROR);
}
}
private int findLineStart(Editable text, int current) {
int pos = current;
for (; pos > 0; pos--) {
if (text.charAt(pos - 1) == '\n') {
break;
}
}
if (DBG) {
Log.d(LOG_TAG, "--- findLineStart:" + current + "," + text.length() + ","
+ pos);
}
return pos;
}
private int findLineEnd(Editable text, int current) {
int pos = current;
for (; pos < text.length(); pos++) {
if (text.charAt(pos) == '\n') {
pos++;
break;
}
}
if (DBG) {
Log.d(LOG_TAG, "--- findLineEnd:" + current + "," + text.length() + "," + pos);
}
return pos;
}
// Only for debug.
private void dumpSpannableString(CharSequence txt) {
if (txt instanceof Spannable) {
Spannable spannable = (Spannable) txt;
int len = spannable.length();
if (DBG) {
Log.d(TAG, "--- dumpSpannableString, txt:" + spannable + ", len:" + len);
}
Object[] styles = spannable.getSpans(0, len, Object.class);
for (Object style : styles) {
if (DBG) {
Log.d(TAG,
"--- dumpSpannableString, class:" + style + ","
+ spannable.getSpanStart(style) + ","
+ spannable.getSpanEnd(style) + ","
+ spannable.getSpanFlags(style));
}
}
}
}
public void showSoftKey() {
showSoftKey(mEST.getSelectionStart(), mEST.getSelectionEnd());
}
public void showSoftKey(int oldSelStart, int oldSelEnd) {
if (DBG) {
Log.d(LOG_TAG, "--- showsoftkey");
}
if (!mEST.isFocused() || isSoftKeyBlocked()) {
return;
}
mSkr.mNewStart = Selection.getSelectionStart(mEST.getText());
mSkr.mNewEnd = Selection.getSelectionEnd(mEST.getText());
InputMethodManager imm =
(InputMethodManager) getContext().getSystemService(
Context.INPUT_METHOD_SERVICE);
if (imm.showSoftInput(mEST, 0, mSkr) && mSkr != null) {
Selection.setSelection(getText(), oldSelStart, oldSelEnd);
}
}
public void hideSoftKey() {
if (DBG) {
Log.d(LOG_TAG, "--- hidesoftkey");
}
if (!mEST.isFocused()) {
return;
}
mSkr.mNewStart = Selection.getSelectionStart(mEST.getText());
mSkr.mNewEnd = Selection.getSelectionEnd(mEST.getText());
InputMethodManager imm =
(InputMethodManager) mEST.getContext().getSystemService(
Context.INPUT_METHOD_SERVICE);
imm.hideSoftInputFromWindow(mEST.getWindowToken(), 0, mSkr);
}
public void blockSoftKey() {
if (DBG) {
Log.d(LOG_TAG, "--- blockSoftKey:");
}
hideSoftKey();
mSoftKeyBlockFlag = true;
}
public void unblockSoftKey() {
if (DBG) {
Log.d(LOG_TAG, "--- unblockSoftKey:");
}
mSoftKeyBlockFlag = false;
}
}
private class StyledTextHtmlStandard implements StyledTextHtmlConverter {
public String toHtml(Spanned text) {
return Html.toHtml(text);
}
public String toHtml(Spanned text, boolean escapeNonAsciiChar) {
return Html.toHtml(text);
}
public String toHtml(Spanned text, boolean escapeNonAsciiChar, int width, float scale) {
return Html.toHtml(text);
}
public Spanned fromHtml(String source) {
return Html.fromHtml(source);
}
public Spanned fromHtml(String source, ImageGetter imageGetter, TagHandler tagHandler) {
return Html.fromHtml(source, imageGetter, tagHandler);
}
}
private class StyledTextConverter {
private EditStyledText mEST;
private StyledTextHtmlConverter mHtml;
public StyledTextConverter(EditStyledText est, StyledTextHtmlConverter html) {
mEST = est;
mHtml = html;
}
public void setStyledTextHtmlConverter(StyledTextHtmlConverter html) {
mHtml = html;
}
public String getHtml(boolean escapeFlag) {
mEST.clearComposingText();
mEST.onRefreshZeoWidthChar();
String htmlBody = mHtml.toHtml(mEST.getText(), escapeFlag);
if (DBG) {
Log.d(TAG, "--- getHtml:" + htmlBody);
}
return htmlBody;
}
public String getPreviewHtml() {
mEST.clearComposingText();
mEST.onRefreshZeoWidthChar();
String html =
mHtml.toHtml(mEST.getText(), true, getMaxImageWidthDip(),
getPaddingScale());
int bgColor = mEST.getBackgroundColor();
html =
String.format("<body bgcolor=\"#%02X%02X%02X\">%s</body>",
Color.red(bgColor), Color.green(bgColor), Color.blue(bgColor),
html);
if (DBG) {
Log.d(TAG, "--- getPreviewHtml:" + html + "," + mEST.getWidth());
}
return html;
}
public void getUriArray(ArrayList<Uri> uris, Editable text) {
uris.clear();
if (DBG) {
Log.d(TAG, "--- getUriArray:");
}
int len = text.length();
int next;
for (int i = 0; i < text.length(); i = next) {
next = text.nextSpanTransition(i, len, ImageSpan.class);
ImageSpan[] images = text.getSpans(i, next, ImageSpan.class);
for (int j = 0; j < images.length; j++) {
if (DBG) {
Log.d(TAG, "--- getUriArray: foundArray" + images[j].getSource());
}
uris.add(Uri.parse(images[j].getSource()));
}
}
}
public void SetHtml(String html) {
final Spanned spanned = mHtml.fromHtml(html, new Html.ImageGetter() {
public Drawable getDrawable(String src) {
Log.d(TAG, "--- sethtml: src=" + src);
if (src.startsWith("content://")) {
Uri uri = Uri.parse(src);
try {
Drawable drawable = null;
Bitmap bitmap = null;
System.gc();
InputStream is =
mEST.getContext().getContentResolver().openInputStream(uri);
BitmapFactory.Options opt = new BitmapFactory.Options();
opt.inJustDecodeBounds = true;
BitmapFactory.decodeStream(is, null, opt);
is.close();
is = mEST.getContext().getContentResolver().openInputStream(uri);
int width, height;
width = opt.outWidth;
height = opt.outHeight;
if (opt.outWidth > getMaxImageWidthPx()) {
width = getMaxImageWidthPx();
height = height * getMaxImageWidthPx() / opt.outWidth;
Rect padding = new Rect(0, 0, width, height);
bitmap = BitmapFactory.decodeStream(is, padding, null);
} else {
bitmap = BitmapFactory.decodeStream(is);
}
drawable = new BitmapDrawable(
mEST.getContext().getResources(), bitmap);
drawable.setBounds(0, 0, width, height);
is.close();
return drawable;
} catch (Exception e) {
Log.e(TAG, "--- set html: Failed to loaded content " + uri, e);
return null;
} catch (OutOfMemoryError e) {
Log.e(TAG, "OutOfMemoryError");
mEST.setHint(HINT_MSG_BIG_SIZE_ERROR);
return null;
}
}
return null;
}
}, null);
mEST.setText(spanned);
}
}
private static class SoftKeyReceiver extends ResultReceiver {
int mNewStart;
int mNewEnd;
EditStyledText mEST;
SoftKeyReceiver(EditStyledText est) {
super(est.getHandler());
mEST = est;
}
@Override
protected void onReceiveResult(int resultCode, Bundle resultData) {
if (resultCode != InputMethodManager.RESULT_SHOWN) {
Selection.setSelection(mEST.getText(), mNewStart, mNewEnd);
}
}
}
public static class SavedStyledTextState extends BaseSavedState {
public int mBackgroundColor;
SavedStyledTextState(Parcelable superState) {
super(superState);
}
@Override
public void writeToParcel(Parcel out, int flags) {
super.writeToParcel(out, flags);
out.writeInt(mBackgroundColor);
}
@Override
public String toString() {
return "EditStyledText.SavedState{"
+ Integer.toHexString(System.identityHashCode(this)) + " bgcolor="
+ mBackgroundColor + "}";
}
}
private static class StyledTextDialog {
private static final int TYPE_FOREGROUND = 0;
private static final int TYPE_BACKGROUND = 1;
private Builder mBuilder;
private AlertDialog mAlertDialog;
private CharSequence mColorTitle;
private CharSequence mSizeTitle;
private CharSequence mAlignTitle;
private CharSequence mMarqueeTitle;
private CharSequence[] mColorNames;
private CharSequence[] mColorInts;
private CharSequence[] mSizeNames;
private CharSequence[] mSizeDisplayInts;
private CharSequence[] mSizeSendInts;
private CharSequence[] mAlignNames;
private CharSequence[] mMarqueeNames;
private CharSequence mColorDefaultMessage;
private EditStyledText mEST;
public StyledTextDialog(EditStyledText est) {
mEST = est;
}
public void setBuilder(Builder builder) {
mBuilder = builder;
}
public void setColorAlertParams(CharSequence colortitle, CharSequence[] colornames,
CharSequence[] colorInts, CharSequence defaultColorMessage) {
mColorTitle = colortitle;
mColorNames = colornames;
mColorInts = colorInts;
mColorDefaultMessage = defaultColorMessage;
}
public void setSizeAlertParams(CharSequence sizetitle, CharSequence[] sizenames,
CharSequence[] sizedisplayints, CharSequence[] sizesendints) {
mSizeTitle = sizetitle;
mSizeNames = sizenames;
mSizeDisplayInts = sizedisplayints;
mSizeSendInts = sizesendints;
}
public void setAlignAlertParams(CharSequence aligntitle, CharSequence[] alignnames) {
mAlignTitle = aligntitle;
mAlignNames = alignnames;
}
public void setMarqueeAlertParams(CharSequence marqueetitle,
CharSequence[] marqueenames) {
mMarqueeTitle = marqueetitle;
mMarqueeNames = marqueenames;
}
private boolean checkColorAlertParams() {
if (DBG) {
Log.d(TAG, "--- checkParams");
}
if (mBuilder == null) {
Log.e(TAG, "--- builder is null.");
return false;
} else if (mColorTitle == null || mColorNames == null || mColorInts == null) {
Log.e(TAG, "--- color alert params are null.");
return false;
} else if (mColorNames.length != mColorInts.length) {
Log.e(TAG, "--- the length of color alert params are " + "different.");
return false;
}
return true;
}
private boolean checkSizeAlertParams() {
if (DBG) {
Log.d(TAG, "--- checkParams");
}
if (mBuilder == null) {
Log.e(TAG, "--- builder is null.");
return false;
} else if (mSizeTitle == null || mSizeNames == null || mSizeDisplayInts == null
|| mSizeSendInts == null) {
Log.e(TAG, "--- size alert params are null.");
return false;
} else if (mSizeNames.length != mSizeDisplayInts.length
&& mSizeSendInts.length != mSizeDisplayInts.length) {
Log.e(TAG, "--- the length of size alert params are " + "different.");
return false;
}
return true;
}
private boolean checkAlignAlertParams() {
if (DBG) {
Log.d(TAG, "--- checkAlignAlertParams");
}
if (mBuilder == null) {
Log.e(TAG, "--- builder is null.");
return false;
} else if (mAlignTitle == null) {
Log.e(TAG, "--- align alert params are null.");
return false;
}
return true;
}
private boolean checkMarqueeAlertParams() {
if (DBG) {
Log.d(TAG, "--- checkMarqueeAlertParams");
}
if (mBuilder == null) {
Log.e(TAG, "--- builder is null.");
return false;
} else if (mMarqueeTitle == null) {
Log.e(TAG, "--- Marquee alert params are null.");
return false;
}
return true;
}
private void buildDialogue(CharSequence title, CharSequence[] names,
DialogInterface.OnClickListener l) {
mBuilder.setTitle(title);
mBuilder.setIcon(0);
mBuilder.setPositiveButton(null, null);
mBuilder.setNegativeButton(R.string.cancel, new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
mEST.onStartEdit();
}
});
mBuilder.setItems(names, l);
mBuilder.setView(null);
mBuilder.setCancelable(true);
mBuilder.setOnCancelListener(new OnCancelListener() {
public void onCancel(DialogInterface arg0) {
if (DBG) {
Log.d(TAG, "--- oncancel");
}
mEST.onStartEdit();
}
});
mBuilder.show();
}
private void buildAndShowColorDialogue(int type, CharSequence title, int[] colors) {
final int HORIZONTAL_ELEMENT_NUM = 5;
final int BUTTON_SIZE = mEST.dipToPx(50);
final int BUTTON_MERGIN = mEST.dipToPx(2);
final int BUTTON_PADDING = mEST.dipToPx(15);
mBuilder.setTitle(title);
mBuilder.setIcon(0);
mBuilder.setPositiveButton(null, null);
mBuilder.setNegativeButton(R.string.cancel, new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
mEST.onStartEdit();
}
});
mBuilder.setItems(null, null);
LinearLayout verticalLayout = new LinearLayout(mEST.getContext());
verticalLayout.setOrientation(LinearLayout.VERTICAL);
verticalLayout.setGravity(Gravity.CENTER_HORIZONTAL);
verticalLayout.setPadding(BUTTON_PADDING, BUTTON_PADDING, BUTTON_PADDING,
BUTTON_PADDING);
LinearLayout horizontalLayout = null;
for (int i = 0; i < colors.length; i++) {
if (i % HORIZONTAL_ELEMENT_NUM == 0) {
horizontalLayout = new LinearLayout(mEST.getContext());
verticalLayout.addView(horizontalLayout);
}
Button button = new Button(mEST.getContext());
button.setHeight(BUTTON_SIZE);
button.setWidth(BUTTON_SIZE);
ColorPaletteDrawable cp =
new ColorPaletteDrawable(colors[i], BUTTON_SIZE, BUTTON_SIZE,
BUTTON_MERGIN);
button.setBackgroundDrawable(cp);
button.setDrawingCacheBackgroundColor(colors[i]);
if (type == TYPE_FOREGROUND) {
button.setOnClickListener(new View.OnClickListener() {
public void onClick(View view) {
mEST.setItemColor(view.getDrawingCacheBackgroundColor());
if (mAlertDialog != null) {
mAlertDialog.setView(null);
mAlertDialog.dismiss();
mAlertDialog = null;
} else {
Log.e(TAG,
"--- buildAndShowColorDialogue: can't find alertDialog");
}
}
});
} else if (type == TYPE_BACKGROUND) {
button.setOnClickListener(new View.OnClickListener() {
public void onClick(View view) {
mEST.setBackgroundColor(view.getDrawingCacheBackgroundColor());
if (mAlertDialog != null) {
mAlertDialog.setView(null);
mAlertDialog.dismiss();
mAlertDialog = null;
} else {
Log.e(TAG,
"--- buildAndShowColorDialogue: can't find alertDialog");
}
}
});
}
horizontalLayout.addView(button);
}
if (type == TYPE_BACKGROUND) {
mBuilder.setPositiveButton(mColorDefaultMessage,
new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
mEST.setBackgroundColor(DEFAULT_TRANSPARENT_COLOR);
}
});
} else if (type == TYPE_FOREGROUND) {
mBuilder.setPositiveButton(mColorDefaultMessage,
new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
mEST.setItemColor(DEFAULT_FOREGROUND_COLOR);
}
});
}
mBuilder.setView(verticalLayout);
mBuilder.setCancelable(true);
mBuilder.setOnCancelListener(new OnCancelListener() {
public void onCancel(DialogInterface arg0) {
mEST.onStartEdit();
}
});
mAlertDialog = mBuilder.show();
}
private void onShowForegroundColorAlertDialog() {
if (DBG) {
Log.d(TAG, "--- onShowForegroundColorAlertDialog");
}
if (!checkColorAlertParams()) {
return;
}
int[] colorints = new int[mColorInts.length];
for (int i = 0; i < colorints.length; i++) {
colorints[i] = Integer.parseInt((String) mColorInts[i], 16) - 0x01000000;
}
buildAndShowColorDialogue(TYPE_FOREGROUND, mColorTitle, colorints);
}
private void onShowBackgroundColorAlertDialog() {
if (DBG) {
Log.d(TAG, "--- onShowBackgroundColorAlertDialog");
}
if (!checkColorAlertParams()) {
return;
}
int[] colorInts = new int[mColorInts.length];
for (int i = 0; i < colorInts.length; i++) {
colorInts[i] = Integer.parseInt((String) mColorInts[i], 16) - 0x01000000;
}
buildAndShowColorDialogue(TYPE_BACKGROUND, mColorTitle, colorInts);
}
private void onShowSizeAlertDialog() {
if (DBG) {
Log.d(TAG, "--- onShowSizeAlertDialog");
}
if (!checkSizeAlertParams()) {
return;
}
buildDialogue(mSizeTitle, mSizeNames, new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
Log.d(TAG, "mBuilder.onclick:" + which);
int size =
mEST.dipToPx(Integer.parseInt((String) mSizeDisplayInts[which]));
mEST.setItemSize(size);
}
});
}
private void onShowAlignAlertDialog() {
if (DBG) {
Log.d(TAG, "--- onShowAlignAlertDialog");
}
if (!checkAlignAlertParams()) {
return;
}
buildDialogue(mAlignTitle, mAlignNames, new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
Layout.Alignment align = Layout.Alignment.ALIGN_NORMAL;
switch (which) {
case 0:
align = Layout.Alignment.ALIGN_NORMAL;
break;
case 1:
align = Layout.Alignment.ALIGN_CENTER;
break;
case 2:
align = Layout.Alignment.ALIGN_OPPOSITE;
break;
default:
Log.e(TAG, "--- onShowAlignAlertDialog: got illigal align.");
break;
}
mEST.setAlignment(align);
}
});
}
private void onShowMarqueeAlertDialog() {
if (DBG) {
Log.d(TAG, "--- onShowMarqueeAlertDialog");
}
if (!checkMarqueeAlertParams()) {
return;
}
buildDialogue(mMarqueeTitle, mMarqueeNames, new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int which) {
if (DBG) {
Log.d(TAG, "mBuilder.onclick:" + which);
}
mEST.setMarquee(which);
}
});
}
}
private class MenuHandler implements MenuItem.OnMenuItemClickListener {
public boolean onMenuItemClick(MenuItem item) {
return onTextContextMenuItem(item.getItemId());
}
}
private static class StyledTextArrowKeyMethod extends ArrowKeyMovementMethod {
EditorManager mManager;
String LOG_TAG = "StyledTextArrowKeyMethod";
StyledTextArrowKeyMethod(EditorManager manager) {
super();
mManager = manager;
}
@Override
public boolean
onKeyDown(TextView widget, Spannable buffer, int keyCode, KeyEvent event) {
if (DBG) {
Log.d(LOG_TAG, "---onkeydown:" + keyCode);
}
mManager.unsetTextComposingMask();
if (mManager.getSelectState() == STATE_SELECT_ON
|| mManager.getSelectState() == STATE_SELECTED) {
return executeDown(widget, buffer, keyCode);
} else {
return super.onKeyDown(widget, buffer, keyCode, event);
}
}
private int getEndPos(TextView widget) {
int end;
if (widget.getSelectionStart() == mManager.getSelectionStart()) {
end = widget.getSelectionEnd();
} else {
end = widget.getSelectionStart();
}
return end;
}
protected boolean up(TextView widget, Spannable buffer) {
if (DBG) {
Log.d(LOG_TAG, "--- up:");
}
Layout layout = widget.getLayout();
int end = getEndPos(widget);
int line = layout.getLineForOffset(end);
if (line > 0) {
int to;
if (layout.getParagraphDirection(line) == layout
.getParagraphDirection(line - 1)) {
float h = layout.getPrimaryHorizontal(end);
to = layout.getOffsetForHorizontal(line - 1, h);
} else {
to = layout.getLineStart(line - 1);
}
mManager.setEndPos(to);
mManager.onCursorMoved();
}
return true;
}
protected boolean down(TextView widget, Spannable buffer) {
if (DBG) {
Log.d(LOG_TAG, "--- down:");
}
Layout layout = widget.getLayout();
int end = getEndPos(widget);
int line = layout.getLineForOffset(end);
if (line < layout.getLineCount() - 1) {
int to;
if (layout.getParagraphDirection(line) == layout
.getParagraphDirection(line + 1)) {
float h = layout.getPrimaryHorizontal(end);
to = layout.getOffsetForHorizontal(line + 1, h);
} else {
to = layout.getLineStart(line + 1);
}
mManager.setEndPos(to);
mManager.onCursorMoved();
}
return true;
}
protected boolean left(TextView widget, Spannable buffer) {
if (DBG) {
Log.d(LOG_TAG, "--- left:");
}
Layout layout = widget.getLayout();
int to = layout.getOffsetToLeftOf(getEndPos(widget));
mManager.setEndPos(to);
mManager.onCursorMoved();
return true;
}
protected boolean right(TextView widget, Spannable buffer) {
if (DBG) {
Log.d(LOG_TAG, "--- right:");
}
Layout layout = widget.getLayout();
int to = layout.getOffsetToRightOf(getEndPos(widget));
mManager.setEndPos(to);
mManager.onCursorMoved();
return true;
}
private boolean executeDown(TextView widget, Spannable buffer, int keyCode) {
if (DBG) {
Log.d(LOG_TAG, "--- executeDown: " + keyCode);
}
boolean handled = false;
switch (keyCode) {
case KeyEvent.KEYCODE_DPAD_UP:
handled |= up(widget, buffer);
break;
case KeyEvent.KEYCODE_DPAD_DOWN:
handled |= down(widget, buffer);
break;
case KeyEvent.KEYCODE_DPAD_LEFT:
handled |= left(widget, buffer);
break;
case KeyEvent.KEYCODE_DPAD_RIGHT:
handled |= right(widget, buffer);
break;
case KeyEvent.KEYCODE_DPAD_CENTER:
mManager.onFixSelectedItem();
handled = true;
break;
}
return handled;
}
}
public static class StyledTextInputConnection extends InputConnectionWrapper {
EditStyledText mEST;
public StyledTextInputConnection(InputConnection target, EditStyledText est) {
super(target, true);
mEST = est;
}
@Override
public boolean commitText(CharSequence text, int newCursorPosition) {
if (DBG) {
Log.d(TAG, "--- commitText:");
}
mEST.mManager.unsetTextComposingMask();
return super.commitText(text, newCursorPosition);
}
@Override
public boolean finishComposingText() {
if (DBG) {
Log.d(TAG, "--- finishcomposing:");
}
if (!mEST.isSoftKeyBlocked() && !mEST.isButtonsFocused() && !mEST.isEditting()) {
// TODO onEndEdit isn't called temporally .
mEST.onEndEdit();
}
return super.finishComposingText();
}
}
public static class EditStyledTextSpans {
private static final String LOG_TAG = "EditStyledTextSpan";
public static class HorizontalLineSpan extends DynamicDrawableSpan {
HorizontalLineDrawable mDrawable;
public HorizontalLineSpan(int color, int width, Spannable spannable) {
super(ALIGN_BOTTOM);
mDrawable = new HorizontalLineDrawable(color, width, spannable);
}
@Override
public Drawable getDrawable() {
return mDrawable;
}
public void resetWidth(int width) {
mDrawable.renewBounds(width);
}
public int getColor() {
return mDrawable.getPaint().getColor();
}
}
public static class MarqueeSpan extends CharacterStyle {
public static final int SCROLL = 0;
public static final int ALTERNATE = 1;
public static final int NOTHING = 2;
private int mType;
private int mMarqueeColor;
public MarqueeSpan(int type, int bgc) {
mType = type;
checkType(type);
mMarqueeColor = getMarqueeColor(type, bgc);
}
public MarqueeSpan(int type) {
this(type, EditStyledText.DEFAULT_TRANSPARENT_COLOR);
}
public int getType() {
return mType;
}
public void resetColor(int bgc) {
mMarqueeColor = getMarqueeColor(mType, bgc);
}
private int getMarqueeColor(int type, int bgc) {
int THRESHOLD = 128;
int a = Color.alpha(bgc);
int r = Color.red(bgc);
int g = Color.green(bgc);
int b = Color.blue(bgc);
if (a == 0) {
a = 0x80;
}
switch (type) {
case SCROLL:
if (r > THRESHOLD) {
r = r / 2;
} else {
r = (0XFF - r) / 2;
}
break;
case ALTERNATE:
if (g > THRESHOLD) {
g = g / 2;
} else {
g = (0XFF - g) / 2;
}
break;
case NOTHING:
return DEFAULT_TRANSPARENT_COLOR;
default:
Log.e(TAG, "--- getMarqueeColor: got illigal marquee ID.");
return DEFAULT_TRANSPARENT_COLOR;
}
return Color.argb(a, r, g, b);
}
private boolean checkType(int type) {
if (type == SCROLL || type == ALTERNATE) {
return true;
} else {
Log.e(LOG_TAG, "--- Invalid type of MarqueeSpan");
return false;
}
}
@Override
public void updateDrawState(TextPaint tp) {
tp.bgColor = mMarqueeColor;
}
}
public static class RescalableImageSpan extends ImageSpan {
Uri mContentUri;
private Drawable mDrawable;
private Context mContext;
public int mIntrinsicWidth = -1;
public int mIntrinsicHeight = -1;
private final int MAXWIDTH;
public RescalableImageSpan(Context context, Uri uri, int maxwidth) {
super(context, uri);
mContext = context;
mContentUri = uri;
MAXWIDTH = maxwidth;
}
public RescalableImageSpan(Context context, int resourceId, int maxwidth) {
super(context, resourceId);
mContext = context;
MAXWIDTH = maxwidth;
}
@Override
public Drawable getDrawable() {
if (mDrawable != null) {
return mDrawable;
} else if (mContentUri != null) {
Bitmap bitmap = null;
System.gc();
try {
InputStream is =
mContext.getContentResolver().openInputStream(mContentUri);
BitmapFactory.Options opt = new BitmapFactory.Options();
opt.inJustDecodeBounds = true;
BitmapFactory.decodeStream(is, null, opt);
is.close();
is = mContext.getContentResolver().openInputStream(mContentUri);
int width, height;
width = opt.outWidth;
height = opt.outHeight;
mIntrinsicWidth = width;
mIntrinsicHeight = height;
if (opt.outWidth > MAXWIDTH) {
width = MAXWIDTH;
height = height * MAXWIDTH / opt.outWidth;
Rect padding = new Rect(0, 0, width, height);
bitmap = BitmapFactory.decodeStream(is, padding, null);
} else {
bitmap = BitmapFactory.decodeStream(is);
}
mDrawable = new BitmapDrawable(mContext.getResources(), bitmap);
mDrawable.setBounds(0, 0, width, height);
is.close();
} catch (Exception e) {
Log.e(LOG_TAG, "Failed to loaded content " + mContentUri, e);
return null;
} catch (OutOfMemoryError e) {
Log.e(LOG_TAG, "OutOfMemoryError");
return null;
}
} else {
mDrawable = super.getDrawable();
rescaleBigImage(mDrawable);
mIntrinsicWidth = mDrawable.getIntrinsicWidth();
mIntrinsicHeight = mDrawable.getIntrinsicHeight();
}
return mDrawable;
}
public boolean isOverSize() {
return (getDrawable().getIntrinsicWidth() > MAXWIDTH);
}
public Uri getContentUri() {
return mContentUri;
}
private void rescaleBigImage(Drawable image) {
if (DBG) {
Log.d(LOG_TAG, "--- rescaleBigImage:");
}
if (MAXWIDTH < 0) {
return;
}
int image_width = image.getIntrinsicWidth();
int image_height = image.getIntrinsicHeight();
if (DBG) {
Log.d(LOG_TAG, "--- rescaleBigImage:" + image_width + "," + image_height
+ "," + MAXWIDTH);
}
if (image_width > MAXWIDTH) {
image_width = MAXWIDTH;
image_height = image_height * MAXWIDTH / image_width;
}
image.setBounds(0, 0, image_width, image_height);
}
}
public static class HorizontalLineDrawable extends ShapeDrawable {
private Spannable mSpannable;
private int mWidth;
private static boolean DBG_HL = false;
public HorizontalLineDrawable(int color, int width, Spannable spannable) {
super(new RectShape());
mSpannable = spannable;
mWidth = width;
renewColor(color);
renewBounds(width);
}
@Override
public void draw(Canvas canvas) {
renewColor();
Rect rect = new Rect(0, 9, mWidth, 11);
canvas.drawRect(rect, getPaint());
}
public void renewBounds(int width) {
int MARGIN = 20;
int HEIGHT = 20;
if (DBG_HL) {
Log.d(LOG_TAG, "--- renewBounds:" + width);
}
if (width > MARGIN) {
width -= MARGIN;
}
mWidth = width;
setBounds(0, 0, width, HEIGHT);
}
private void renewColor(int color) {
if (DBG_HL) {
Log.d(LOG_TAG, "--- renewColor:" + color);
}
getPaint().setColor(color);
}
private void renewColor() {
HorizontalLineSpan parent = getParentSpan();
Spannable text = mSpannable;
int start = text.getSpanStart(parent);
int end = text.getSpanEnd(parent);
ForegroundColorSpan[] spans =
text.getSpans(start, end, ForegroundColorSpan.class);
if (DBG_HL) {
Log.d(LOG_TAG, "--- renewColor:" + spans.length);
}
if (spans.length > 0) {
renewColor(spans[spans.length - 1].getForegroundColor());
}
}
private HorizontalLineSpan getParentSpan() {
Spannable text = mSpannable;
HorizontalLineSpan[] images =
text.getSpans(0, text.length(), HorizontalLineSpan.class);
if (images.length > 0) {
for (HorizontalLineSpan image : images) {
if (image.getDrawable() == this) {
return image;
}
}
}
Log.e(LOG_TAG, "---renewBounds: Couldn't find");
return null;
}
}
}
public static class ColorPaletteDrawable extends ShapeDrawable {
private Rect mRect;
public ColorPaletteDrawable(int color, int width, int height, int mergin) {
super(new RectShape());
mRect = new Rect(mergin, mergin, width - mergin, height - mergin);
getPaint().setColor(color);
}
@Override
public void draw(Canvas canvas) {
canvas.drawRect(mRect, getPaint());
}
}
public class EditModeActions {
private static final String TAG = "EditModeActions";
private static final boolean DBG = true;
private EditStyledText mEST;
private EditorManager mManager;
private StyledTextDialog mDialog;
private int mMode = EditStyledText.MODE_NOTHING;
private HashMap<Integer, EditModeActionBase> mActionMap =
new HashMap<Integer, EditModeActionBase>();
private NothingAction mNothingAction = new NothingAction();
private CopyAction mCopyAction = new CopyAction();
private PasteAction mPasteAction = new PasteAction();
private SelectAction mSelectAction = new SelectAction();
private CutAction mCutAction = new CutAction();
private SelectAllAction mSelectAllAction = new SelectAllAction();
private HorizontalLineAction mHorizontalLineAction = new HorizontalLineAction();
private StopSelectionAction mStopSelectionAction = new StopSelectionAction();
private ClearStylesAction mClearStylesAction = new ClearStylesAction();
private ImageAction mImageAction = new ImageAction();
private BackgroundColorAction mBackgroundColorAction = new BackgroundColorAction();
private PreviewAction mPreviewAction = new PreviewAction();
private CancelAction mCancelEditAction = new CancelAction();
private TextViewAction mTextViewAction = new TextViewAction();
private StartEditAction mStartEditAction = new StartEditAction();
private EndEditAction mEndEditAction = new EndEditAction();
private ResetAction mResetAction = new ResetAction();
private ShowMenuAction mShowMenuAction = new ShowMenuAction();
private AlignAction mAlignAction = new AlignAction();
private TelopAction mTelopAction = new TelopAction();
private SwingAction mSwingAction = new SwingAction();
private MarqueeDialogAction mMarqueeDialogAction = new MarqueeDialogAction();
private ColorAction mColorAction = new ColorAction();
private SizeAction mSizeAction = new SizeAction();
EditModeActions(EditStyledText est, EditorManager manager, StyledTextDialog dialog) {
mEST = est;
mManager = manager;
mDialog = dialog;
mActionMap.put(EditStyledText.MODE_NOTHING, mNothingAction);
mActionMap.put(EditStyledText.MODE_COPY, mCopyAction);
mActionMap.put(EditStyledText.MODE_PASTE, mPasteAction);
mActionMap.put(EditStyledText.MODE_SELECT, mSelectAction);
mActionMap.put(EditStyledText.MODE_CUT, mCutAction);
mActionMap.put(EditStyledText.MODE_SELECTALL, mSelectAllAction);
mActionMap.put(EditStyledText.MODE_HORIZONTALLINE, mHorizontalLineAction);
mActionMap.put(EditStyledText.MODE_STOP_SELECT, mStopSelectionAction);
mActionMap.put(EditStyledText.MODE_CLEARSTYLES, mClearStylesAction);
mActionMap.put(EditStyledText.MODE_IMAGE, mImageAction);
mActionMap.put(EditStyledText.MODE_BGCOLOR, mBackgroundColorAction);
mActionMap.put(EditStyledText.MODE_PREVIEW, mPreviewAction);
mActionMap.put(EditStyledText.MODE_CANCEL, mCancelEditAction);
mActionMap.put(EditStyledText.MODE_TEXTVIEWFUNCTION, mTextViewAction);
mActionMap.put(EditStyledText.MODE_START_EDIT, mStartEditAction);
mActionMap.put(EditStyledText.MODE_END_EDIT, mEndEditAction);
mActionMap.put(EditStyledText.MODE_RESET, mResetAction);
mActionMap.put(EditStyledText.MODE_SHOW_MENU, mShowMenuAction);
mActionMap.put(EditStyledText.MODE_ALIGN, mAlignAction);
mActionMap.put(EditStyledText.MODE_TELOP, mTelopAction);
mActionMap.put(EditStyledText.MODE_SWING, mSwingAction);
mActionMap.put(EditStyledText.MODE_MARQUEE, mMarqueeDialogAction);
mActionMap.put(EditStyledText.MODE_COLOR, mColorAction);
mActionMap.put(EditStyledText.MODE_SIZE, mSizeAction);
}
public void addAction(int modeId, EditModeActionBase action) {
mActionMap.put(modeId, action);
}
public void onAction(int newMode, Object[] params) {
getAction(newMode).addParams(params);
mMode = newMode;
doNext(newMode);
}
public void onAction(int newMode, Object param) {
onAction(newMode, new Object[] { param });
}
public void onAction(int newMode) {
onAction(newMode, null);
}
public void onSelectAction() {
doNext(EditStyledText.MODE_SELECT);
}
private EditModeActionBase getAction(int mode) {
if (mActionMap.containsKey(mode)) {
return mActionMap.get(mode);
}
return null;
}
public boolean doNext() {
return doNext(mMode);
}
public boolean doNext(int mode) {
if (DBG) {
Log.d(TAG, "--- do the next action: " + mode + "," + mManager.getSelectState());
}
EditModeActionBase action = getAction(mode);
if (action == null) {
Log.e(TAG, "--- invalid action error.");
return false;
}
switch (mManager.getSelectState()) {
case EditStyledText.STATE_SELECT_OFF:
return action.doNotSelected();
case EditStyledText.STATE_SELECT_ON:
return action.doStartPosIsSelected();
case EditStyledText.STATE_SELECTED:
return action.doEndPosIsSelected();
case EditStyledText.STATE_SELECT_FIX:
if (mManager.isWaitInput()) {
return action.doSelectionIsFixedAndWaitingInput();
} else {
return action.doSelectionIsFixed();
}
default:
return false;
}
}
public class EditModeActionBase {
private Object[] mParams;
protected boolean canOverWrap() {
return false;
}
protected boolean canSelect() {
return false;
}
protected boolean canWaitInput() {
return false;
}
protected boolean needSelection() {
return false;
}
protected boolean isLine() {
return false;
}
protected boolean doNotSelected() {
return false;
}
protected boolean doStartPosIsSelected() {
return doNotSelected();
}
protected boolean doEndPosIsSelected() {
return doStartPosIsSelected();
}
protected boolean doSelectionIsFixed() {
return doEndPosIsSelected();
}
protected boolean doSelectionIsFixedAndWaitingInput() {
return doEndPosIsSelected();
}
protected boolean fixSelection() {
mEST.finishComposingText();
mManager.setSelectState(EditStyledText.STATE_SELECT_FIX);
return true;
}
protected void addParams(Object[] o) {
mParams = o;
}
protected Object getParam(int num) {
if (mParams == null || num > mParams.length) {
if (DBG) {
Log.d(TAG, "--- Number of the parameter is out of bound.");
}
return null;
} else {
return mParams[num];
}
}
}
public class NothingAction extends EditModeActionBase {
}
public class TextViewActionBase extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
if (mManager.getEditMode() == EditStyledText.MODE_NOTHING
|| mManager.getEditMode() == EditStyledText.MODE_SELECT) {
mManager.setEditMode(mMode);
onSelectAction();
return true;
}
return false;
}
@Override
protected boolean doEndPosIsSelected() {
if (mManager.getEditMode() == EditStyledText.MODE_NOTHING
|| mManager.getEditMode() == EditStyledText.MODE_SELECT) {
mManager.setEditMode(mMode);
fixSelection();
doNext();
return true;
} else if (mManager.getEditMode() != mMode) {
mManager.resetEdit();
mManager.setEditMode(mMode);
doNext();
return true;
}
return false;
}
}
public class TextViewAction extends TextViewActionBase {
@Override
protected boolean doEndPosIsSelected() {
if (super.doEndPosIsSelected()) {
return true;
}
Object param = getParam(0);
if (param != null && param instanceof Integer) {
mEST.onTextContextMenuItem((Integer) param);
}
mManager.resetEdit();
return true;
}
}
public class CopyAction extends TextViewActionBase {
@Override
protected boolean doEndPosIsSelected() {
if (super.doEndPosIsSelected()) {
return true;
}
mManager.copyToClipBoard();
mManager.resetEdit();
return true;
}
}
public class CutAction extends TextViewActionBase {
@Override
protected boolean doEndPosIsSelected() {
if (super.doEndPosIsSelected()) {
return true;
}
mManager.cutToClipBoard();
mManager.resetEdit();
return true;
}
}
public class SelectAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
if (mManager.isTextSelected()) {
Log.e(TAG, "Selection is off, but selected");
}
mManager.setSelectStartPos();
mEST.sendHintMessage(EditStyledText.HINT_MSG_SELECT_END);
return true;
}
@Override
protected boolean doStartPosIsSelected() {
if (mManager.isTextSelected()) {
Log.e(TAG, "Selection now start, but selected");
}
mManager.setSelectEndPos();
mEST.sendHintMessage(EditStyledText.HINT_MSG_PUSH_COMPETE);
if (mManager.getEditMode() != EditStyledText.MODE_SELECT) {
doNext(mManager.getEditMode()); // doNextHandle needs edit mode in editor.
}
return true;
}
@Override
protected boolean doSelectionIsFixed() {
return false;
}
}
public class PasteAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mManager.pasteFromClipboard();
mManager.resetEdit();
return true;
}
}
public class SelectAllAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mManager.selectAll();
return true;
}
}
public class HorizontalLineAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mManager.insertHorizontalLine();
return true;
}
}
public class ClearStylesAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mManager.clearStyles();
return true;
}
}
public class StopSelectionAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mManager.fixSelectionAndDoNextAction();
return true;
}
}
public class CancelAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mEST.cancelViewManagers();
return true;
}
}
public class ImageAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
Object param = getParam(0);
if (param != null) {
if (param instanceof Uri) {
mManager.insertImageFromUri((Uri) param);
} else if (param instanceof Integer) {
mManager.insertImageFromResId((Integer) param);
}
} else {
mEST.showInsertImageSelectAlertDialog();
}
return true;
}
}
public class BackgroundColorAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mDialog.onShowBackgroundColorAlertDialog();
return true;
}
}
public class PreviewAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mEST.showPreview();
return true;
}
}
public class StartEditAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mManager.startEdit();
return true;
}
}
public class EndEditAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mManager.endEdit();
return true;
}
}
public class ResetAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mManager.resetEdit();
return true;
}
}
public class ShowMenuAction extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
mEST.showMenuAlertDialog();
return true;
}
}
public class SetSpanActionBase extends EditModeActionBase {
@Override
protected boolean doNotSelected() {
if (mManager.getEditMode() == EditStyledText.MODE_NOTHING
|| mManager.getEditMode() == EditStyledText.MODE_SELECT) {
mManager.setEditMode(mMode);
mManager.setInternalSelection(mEST.getSelectionStart(),
mEST.getSelectionEnd());
fixSelection();
doNext();
return true;
} else if (mManager.getEditMode() != mMode) {
Log.d(TAG, "--- setspanactionbase" + mManager.getEditMode() + "," + mMode);
if (!mManager.isWaitInput()) {
mManager.resetEdit();
mManager.setEditMode(mMode);
} else {
mManager.setEditMode(EditStyledText.MODE_NOTHING);
mManager.setSelectState(EditStyledText.STATE_SELECT_OFF);
}
doNext();
return true;
}
return false;
}
@Override
protected boolean doStartPosIsSelected() {
if (mManager.getEditMode() == EditStyledText.MODE_NOTHING
|| mManager.getEditMode() == EditStyledText.MODE_SELECT) {
mManager.setEditMode(mMode);
onSelectAction();
return true;
}
return doNotSelected();
}
@Override
protected boolean doEndPosIsSelected() {
if (mManager.getEditMode() == EditStyledText.MODE_NOTHING
|| mManager.getEditMode() == EditStyledText.MODE_SELECT) {
mManager.setEditMode(mMode);
fixSelection();
doNext();
return true;
}
return doStartPosIsSelected();
}
@Override
protected boolean doSelectionIsFixed() {
if (doEndPosIsSelected()) {
return true;
}
mEST.sendHintMessage(EditStyledText.HINT_MSG_NULL);
return false;
}
}
public class AlignAction extends SetSpanActionBase {
@Override
protected boolean doSelectionIsFixed() {
if (super.doSelectionIsFixed()) {
return true;
}
mDialog.onShowAlignAlertDialog();
return true;
}
}
public class TelopAction extends SetSpanActionBase {
@Override
protected boolean doSelectionIsFixed() {
if (super.doSelectionIsFixed()) {
return true;
}
mManager.setTelop();
return true;
}
}
public class SwingAction extends SetSpanActionBase {
@Override
protected boolean doSelectionIsFixed() {
if (super.doSelectionIsFixed()) {
return true;
}
mManager.setSwing();
return true;
}
}
public class MarqueeDialogAction extends SetSpanActionBase {
@Override
protected boolean doSelectionIsFixed() {
if (super.doSelectionIsFixed()) {
return true;
}
mDialog.onShowMarqueeAlertDialog();
return true;
}
}
public class ColorAction extends SetSpanActionBase {
@Override
protected boolean doSelectionIsFixed() {
if (super.doSelectionIsFixed()) {
return true;
}
mDialog.onShowForegroundColorAlertDialog();
return true;
}
@Override
protected boolean doSelectionIsFixedAndWaitingInput() {
if (super.doSelectionIsFixedAndWaitingInput()) {
return true;
}
int size = mManager.getSizeWaitInput();
mManager.setItemColor(mManager.getColorWaitInput(), false);
// selection was resumed
if (!mManager.isWaitInput()) {
mManager.setItemSize(size, false);
mManager.resetEdit();
} else {
fixSelection();
mDialog.onShowForegroundColorAlertDialog();
}
return true;
}
}
public class SizeAction extends SetSpanActionBase {
@Override
protected boolean doSelectionIsFixed() {
if (super.doSelectionIsFixed()) {
return true;
}
mDialog.onShowSizeAlertDialog();
return true;
}
@Override
protected boolean doSelectionIsFixedAndWaitingInput() {
if (super.doSelectionIsFixedAndWaitingInput()) {
return true;
}
int color = mManager.getColorWaitInput();
mManager.setItemSize(mManager.getSizeWaitInput(), false);
if (!mManager.isWaitInput()) {
mManager.setItemColor(color, false);
mManager.resetEdit();
} else {
fixSelection();
mDialog.onShowSizeAlertDialog();
}
return true;
}
}
}
}