/* * Copyright (C) 2017 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 android.support.text.emoji.widget; import static android.support.annotation.RestrictTo.Scope.LIBRARY_GROUP; import android.support.annotation.NonNull; import android.support.annotation.Nullable; import android.support.annotation.RestrictTo; import android.support.text.emoji.EmojiSpan; import android.support.v4.util.Preconditions; import android.text.Editable; import android.text.SpanWatcher; import android.text.Spannable; import android.text.SpannableStringBuilder; import android.text.TextWatcher; import java.lang.reflect.Array; import java.util.ArrayList; import java.util.List; import java.util.concurrent.atomic.AtomicInteger; /** * When setSpan functions is called on EmojiSpannableBuilder, it checks if the mObject is instance * of the DynamicLayout$ChangeWatcher. if so, it wraps it into another listener mObject * (WatcherWrapper) that implements the same interfaces. *

* During a span change event WatcherWrapper’s functions are fired, it checks if the span is an * EmojiSpan, and prevents the ChangeWatcher being fired for that span. WatcherWrapper informs * ChangeWatcher only once at the end of the edit. Important point is, the block operation is * applied only for EmojiSpans. Therefore any other span change operation works the same way as in * the framework. * * @hide * @see EmojiEditableFactory */ @RestrictTo(LIBRARY_GROUP) public final class SpannableBuilder extends SpannableStringBuilder { /** * DynamicLayout$ChangeWatcher class. */ private final Class mWatcherClass; /** * All WatcherWrappers. */ private final List mWatchers = new ArrayList<>(); /** * @hide */ @RestrictTo(LIBRARY_GROUP) SpannableBuilder(@NonNull Class watcherClass) { Preconditions.checkNotNull(watcherClass, "watcherClass cannot be null"); mWatcherClass = watcherClass; } /** * @hide */ @RestrictTo(LIBRARY_GROUP) SpannableBuilder(@NonNull Class watcherClass, @NonNull CharSequence text) { super(text); Preconditions.checkNotNull(watcherClass, "watcherClass cannot be null"); mWatcherClass = watcherClass; } /** * @hide */ @RestrictTo(LIBRARY_GROUP) SpannableBuilder(@NonNull Class watcherClass, @NonNull CharSequence text, int start, int end) { super(text, start, end); Preconditions.checkNotNull(watcherClass, "watcherClass cannot be null"); mWatcherClass = watcherClass; } /** * @hide */ @RestrictTo(LIBRARY_GROUP) static SpannableBuilder create(@NonNull Class clazz, @NonNull CharSequence text) { return new SpannableBuilder(clazz, text); } /** * Checks whether the mObject is instance of the DynamicLayout$ChangeWatcher. * * @param object mObject to be checked * * @return true if mObject is instance of the DynamicLayout$ChangeWatcher. */ private boolean isWatcher(@Nullable Object object) { return object != null && isWatcher(object.getClass()); } /** * Checks whether the class is DynamicLayout$ChangeWatcher. * * @param clazz class to be checked * * @return true if class is DynamicLayout$ChangeWatcher. */ private boolean isWatcher(@NonNull Class clazz) { return mWatcherClass == clazz; } @Override public CharSequence subSequence(int start, int end) { return new SpannableBuilder(mWatcherClass, this, start, end); } /** * If the span being added is instance of DynamicLayout$ChangeWatcher, wrap the watcher in * another internal watcher that will prevent EmojiSpan events to be fired to DynamicLayout. Set * this new mObject as the span. */ @Override public void setSpan(Object what, int start, int end, int flags) { if (isWatcher(what)) { final WatcherWrapper span = new WatcherWrapper(what); mWatchers.add(span); what = span; } super.setSpan(what, start, end, flags); } /** * If previously a DynamicLayout$ChangeWatcher was wrapped in a WatcherWrapper, return the * correct Object that the client has set. */ @SuppressWarnings("unchecked") @Override public T[] getSpans(int queryStart, int queryEnd, Class kind) { if (isWatcher(kind)) { final WatcherWrapper[] spans = super.getSpans(queryStart, queryEnd, WatcherWrapper.class); final T[] result = (T[]) Array.newInstance(kind, spans.length); for (int i = 0; i < spans.length; i++) { result[i] = (T) spans[i].mObject; } return result; } return super.getSpans(queryStart, queryEnd, kind); } /** * If the client wants to remove the DynamicLayout$ChangeWatcher span, remove the WatcherWrapper * instead. */ @Override public void removeSpan(Object what) { final WatcherWrapper watcher; if (isWatcher(what)) { watcher = getWatcherFor(what); if (watcher != null) { what = watcher; } } else { watcher = null; } super.removeSpan(what); if (watcher != null) { mWatchers.remove(watcher); } } /** * Return the correct start for the DynamicLayout$ChangeWatcher span. */ @Override public int getSpanStart(Object tag) { if (isWatcher(tag)) { final WatcherWrapper watcher = getWatcherFor(tag); if (watcher != null) { tag = watcher; } } return super.getSpanStart(tag); } /** * Return the correct end for the DynamicLayout$ChangeWatcher span. */ @Override public int getSpanEnd(Object tag) { if (isWatcher(tag)) { final WatcherWrapper watcher = getWatcherFor(tag); if (watcher != null) { tag = watcher; } } return super.getSpanEnd(tag); } /** * Return the correct flags for the DynamicLayout$ChangeWatcher span. */ @Override public int getSpanFlags(Object tag) { if (isWatcher(tag)) { final WatcherWrapper watcher = getWatcherFor(tag); if (watcher != null) { tag = watcher; } } return super.getSpanFlags(tag); } /** * Return the correct transition for the DynamicLayout$ChangeWatcher span. */ @Override public int nextSpanTransition(int start, int limit, Class type) { if (isWatcher(type)) { type = WatcherWrapper.class; } return super.nextSpanTransition(start, limit, type); } /** * Find the WatcherWrapper for a given DynamicLayout$ChangeWatcher. * * @param object DynamicLayout$ChangeWatcher mObject * * @return WatcherWrapper that wraps the mObject. */ private WatcherWrapper getWatcherFor(Object object) { for (int i = 0; i < mWatchers.size(); i++) { WatcherWrapper watcher = mWatchers.get(i); if (watcher.mObject == object) { return watcher; } } return null; } /** * @hide */ @RestrictTo(LIBRARY_GROUP) public void beginBatchEdit() { blockWatchers(); } /** * @hide */ @RestrictTo(LIBRARY_GROUP) public void endBatchEdit() { unblockwatchers(); fireWatchers(); } /** * Block all watcher wrapper events. */ private void blockWatchers() { for (int i = 0; i < mWatchers.size(); i++) { mWatchers.get(i).blockCalls(); } } /** * Unblock all watcher wrapper events. */ private void unblockwatchers() { for (int i = 0; i < mWatchers.size(); i++) { mWatchers.get(i).unblockCalls(); } } /** * Unblock all watcher wrapper events. Called by editing operations, namely * {@link SpannableStringBuilder#replace(int, int, CharSequence)}. */ private void fireWatchers() { for (int i = 0; i < mWatchers.size(); i++) { mWatchers.get(i).onTextChanged(this, 0, this.length(), this.length()); } } @Override public SpannableStringBuilder replace(int start, int end, CharSequence tb) { blockWatchers(); super.replace(start, end, tb); unblockwatchers(); return this; } @Override public SpannableStringBuilder replace(int start, int end, CharSequence tb, int tbstart, int tbend) { blockWatchers(); super.replace(start, end, tb, tbstart, tbend); unblockwatchers(); return this; } @Override public SpannableStringBuilder insert(int where, CharSequence tb) { super.insert(where, tb); return this; } @Override public SpannableStringBuilder insert(int where, CharSequence tb, int start, int end) { super.insert(where, tb, start, end); return this; } @Override public SpannableStringBuilder delete(int start, int end) { super.delete(start, end); return this; } @Override public SpannableStringBuilder append(CharSequence text) { super.append(text); return this; } @Override public SpannableStringBuilder append(char text) { super.append(text); return this; } @Override public SpannableStringBuilder append(CharSequence text, int start, int end) { super.append(text, start, end); return this; } @Override public SpannableStringBuilder append(CharSequence text, Object what, int flags) { super.append(text, what, flags); return this; } /** * Wraps a DynamicLayout$ChangeWatcher in order to prevent firing of events to DynamicLayout. */ private static class WatcherWrapper implements TextWatcher, SpanWatcher { private final Object mObject; private final AtomicInteger mBlockCalls = new AtomicInteger(0); WatcherWrapper(Object object) { this.mObject = object; } @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { ((TextWatcher) mObject).beforeTextChanged(s, start, count, after); } @Override public void onTextChanged(CharSequence s, int start, int before, int count) { ((TextWatcher) mObject).onTextChanged(s, start, before, count); } @Override public void afterTextChanged(Editable s) { ((TextWatcher) mObject).afterTextChanged(s); } /** * Prevent the onSpanAdded calls to DynamicLayout$ChangeWatcher if in a replace operation * (mBlockCalls is set) and the span that is added is an EmojiSpan. */ @Override public void onSpanAdded(Spannable text, Object what, int start, int end) { if (mBlockCalls.get() > 0 && isEmojiSpan(what)) { return; } ((SpanWatcher) mObject).onSpanAdded(text, what, start, end); } /** * Prevent the onSpanRemoved calls to DynamicLayout$ChangeWatcher if in a replace operation * (mBlockCalls is set) and the span that is added is an EmojiSpan. */ @Override public void onSpanRemoved(Spannable text, Object what, int start, int end) { if (mBlockCalls.get() > 0 && isEmojiSpan(what)) { return; } ((SpanWatcher) mObject).onSpanRemoved(text, what, start, end); } /** * Prevent the onSpanChanged calls to DynamicLayout$ChangeWatcher if in a replace operation * (mBlockCalls is set) and the span that is added is an EmojiSpan. */ @Override public void onSpanChanged(Spannable text, Object what, int ostart, int oend, int nstart, int nend) { if (mBlockCalls.get() > 0 && isEmojiSpan(what)) { return; } ((SpanWatcher) mObject).onSpanChanged(text, what, ostart, oend, nstart, nend); } final void blockCalls() { mBlockCalls.incrementAndGet(); } final void unblockCalls() { mBlockCalls.decrementAndGet(); } private boolean isEmojiSpan(final Object span) { return span instanceof EmojiSpan; } } }