web-dev-qa-db-ja.com

テキスト選択ハンドルを表示するEditTextのカスタムカット/コピーアクションバー

ユーザーがテキストを選択し、ボタンを押してそのテキストを処理できるようにするTextView(またはEditText)を表示できるようにするアプリがあります。これをAndroid Honeycombより前のバージョンで実装しても問題ありませんが、Honeycombでデフォルトの長押しアクションを超えると、コピー/カット/貼り付けオプションでアクションバーが表示されます。を押して独自のアクションバーを表示しますが、テキスト選択ハンドルが表示されません。

独自のActionModeを開始した後、テキスト選択ハンドルを表示するにはどうすればよいですか?

次に、ActionModeを開始するために使用しているコードを示します。これは、テキスト選択ハンドルが表示されていないことを除いて機能します。

public boolean onLongClick(View v) {
    if(actionMode == null)
        actionMode = startActionMode(new QuoteCallback());
    return true;
}

class QuoteCallback implements ActionMode.Callback {

    public boolean onCreateActionMode(ActionMode mode, Menu menu) {
        MenuInflater inflater = mode.getMenuInflater();
        inflater.inflate(R.menu.quote, menu);
        return true;
    }

    public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
        return false;
    }

    public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
        switch(item.getItemId()) {

        case R.id.quote:
            Log.d(TAG, "Selected menu");
            mode.finish();
            // here is where I would grab the selected text
            return true;
        }
        return false;
    }

    public void onDestroyActionMode(ActionMode mode) {
        actionMode = null;
    }
}
34
Clyde

私は自分の質問に対する答えを見つけました。 TextView(したがってEditText)にはsetCustomSelectionActionModeCallback()の代わりに使用するメソッドstartActionMode()があります。これを使用すると、TextViewがテキスト選択に使用するメニューをカスタマイズできます。サンプルコード:

bodyView.setCustomSelectionActionModeCallback(new StyleCallback());

styleCallbackは、[すべて選択]を削除していくつかのスタイルアクションを追加することにより、テキスト選択メニューをカスタマイズします。

class StyleCallback implements ActionMode.Callback {

    public boolean onCreateActionMode(ActionMode mode, Menu menu) {
        Log.d(TAG, "onCreateActionMode");
        MenuInflater inflater = mode.getMenuInflater();
        inflater.inflate(R.menu.style, menu);
        menu.removeItem(Android.R.id.selectAll);
        return true;
    }

    public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
        return false;
    }

    public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
        Log.d(TAG, String.format("onActionItemClicked item=%s/%d", item.toString(), item.getItemId()));
        CharacterStyle cs;
        int start = bodyView.getSelectionStart();
        int end = bodyView.getSelectionEnd();
        SpannableStringBuilder ssb = new SpannableStringBuilder(bodyView.getText());

        switch(item.getItemId()) {

        case R.id.bold:
            cs = new StyleSpan(Typeface.BOLD);
            ssb.setSpan(cs, start, end, 1);
            bodyView.setText(ssb);
            return true;

        case R.id.italic:
            cs = new StyleSpan(Typeface.ITALIC);
            ssb.setSpan(cs, start, end, 1);
            bodyView.setText(ssb);
            return true;

        case R.id.underline:
            cs = new UnderlineSpan();
            ssb.setSpan(cs, start, end, 1);
            bodyView.setText(ssb);
            return true;
        }
        return false;
    }

    public void onDestroyActionMode(ActionMode mode) {
    }
}

メニュー追加のXMLは次のとおりです。

<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:Android="http://schemas.Android.com/apk/res/Android">
    <item Android:id="@+id/italic"
          Android:showAsAction="always"
          Android:icon="@drawable/italic"
          Android:title="Italic"/>
    <item Android:id="@+id/bold"
          Android:showAsAction="always"
          Android:icon="@drawable/bold"
          Android:title="Bold"/>
    <item Android:id="@+id/underline"
          Android:showAsAction="always"
          Android:icon="@drawable/underline"
          Android:title="Underline"/>
</menu>
51
Clyde

上記の解決策は、アクションバーのオプションをカスタマイズする場合に適しています。しかし、アクションバーのコピー/貼り付けなどをオーバーライドしたい場合は、以下のコードを...

public class MainActivity extends Activity {
    EditText editText;
    private ClipboardManager myClipboard;
    private ClipData myClip;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        myClipboard = (ClipboardManager) getSystemService(CLIPBOARD_SERVICE);
        editText = (EditText) findViewById(R.id.editText3);

        myClipboard = (ClipboardManager) getSystemService(CLIPBOARD_SERVICE);
        editText = (EditText) findViewById(R.id.editText3);
        editText.setCustomSelectionActionModeCallback(new Callback() {

            @Override
            public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
                // TODO Auto-generated method stub
                return false;
            }

            @Override
            public void onDestroyActionMode(ActionMode mode) {
                // TODO Auto-generated method stub

            }

            @Override
            public boolean onCreateActionMode(ActionMode mode, Menu menu) {
                // TODO Auto-generated method stub
                return true;
            }

            @Override
            public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
                // TODO Auto-generated method stub
                switch (item.getItemId()) {
                case Android.R.id.copy:
                    int min = 0;
                    int max = editText.getText().length();
                    if (editText.isFocused()) {
                        final int selStart = editText.getSelectionStart();
                        final int selEnd = editText.getSelectionEnd();

                        min = Math.max(0, Math.min(selStart, selEnd));
                        max = Math.max(0, Math.max(selStart, selEnd));
                    }
                    // Perform your definition lookup with the selected text
                    final CharSequence selectedText = editText.getText()
                            .subSequence(min, max);
                    String text = selectedText.toString();

                    myClip = ClipData.newPlainText("text", text);
                    myClipboard.setPrimaryClip(myClip);
                    Toast.makeText(getApplicationContext(), "Text Copied",
                            Toast.LENGTH_SHORT).show();
                    // Finish and close the ActionMode
                    mode.finish();
                    return true;
                case Android.R.id.cut:
                    // add your custom code to get cut functionality according
                    // to your requirement
                    return true;
                case Android.R.id.paste:
                    // add your custom code to get paste functionality according
                    // to your requirement
                    return true;

                default:
                    break;
                }
                return false;
            }
        });         
    }    
}
9
SKG

最も簡単な方法は、applicationAndroidManifestタグで定義したメインテーマスタイルに行を追加することです。テーマスタイルを開き、以下を追加します。

<item name="actionModeBackground">@color/your_color</item>

OR

<item name="Android:actionModeBackground">@color/your_color</item>

例:定義した私のテーマスタイル:

<style name="AppTheme" parent="AppBaseTheme">

        <item name="calendarViewStyle">@style/Widget.Holo.CalendarView</item>
        <item name="Android:actionBarStyle">@style/AppTheme1</item>
        <!-- below is the line you have to add -->
        <item name="Android:actionModeBackground">@color/black_actionBar</item>
</style>
0
Pankaj