/*
* Copyright (C) 2015-2017 Emanuel Moecklin
*
* 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.onegravity.rteditor.effects;
import android.text.Spannable;
import com.onegravity.rteditor.spans.RTSpan;
import com.onegravity.rteditor.utils.Selection;
import java.lang.reflect.Array;
import java.util.List;
/**
* Spanned.getSpans(int, int, Class) unfortunately doesn't respect the mark/point flags
* (SPAN_EXCLUSIVE_EXCLUSIVE, SPAN_INCLUSIVE_EXCLUSIVE etc.).
*
* Android finds spans that precede or follow a selection (adjacent spans).
*
* - if the span is a point (start == end) --> every adjacent selection (cursor, range = selected text) will match:
* 01234[]|56789 point selection after point span --> getSpans returns span
* 01234|[]56789 point selection before point span --> getSpans returns span
* 01234[]|56|789 range selection after point span --> getSpans returns span
* 012|34|[]56789 range selection before point span --> getSpans returns span
*
* - if the span is a range (start < end) --> only point selections (cursors) will match:
* 01[234]|56789 point selection after range span --> getSpans returns span
* 01234|[567]89 point selection before range span --> getSpans returns span
* 01[234]|56|789 range selection after range span --> getSpans returns nothing
* 012|34|[567]89 range selection before range span --> getSpans returns nothing
*
* The span flags (SPAN_EXCLUSIVE_INCLUSIVE etc.) have no impact!
*
* (exception SPAN_EXCLUSIVE_EXCLUSIVE for point spans because those can't have a length of 0 and
* will be removed automatically:
* http://developer.android.com/reference/android/text/Spanned.html#SPAN_EXCLUSIVE_EXCLUSIVE)
*
* Sometimes we need to find or ignore adjacent spans depending on the span flags, the position of a
* span, the selection (last line, empty lines...) and the type of span (character, paragraph).
*
* This class allows us to implement different getSpan methods that honor the span flags.
*
* @param <V> the Effect's configuration information.
*/
abstract class SpanCollector<V> {
private Class<? extends RTSpan<V>> mSpanClazz;
protected SpanCollector(Class<? extends RTSpan<V>> spanClazz) {
mSpanClazz = spanClazz;
}
/**
* Equivalent to the Spanned.getSpans(int, int, Class<T>) method.
* Return the markup objects (spans) attached to the specified slice of a Spannable.
* The type of the spans is defined in the SpanCollector.
*
* @param str The Spannable to search for spans.
* @param selection The selection within the Spannable to search for spans.
* @param mode details see SpanCollectMode.
*
* @return the list of spans in this Spannable/Selection, never Null
*/
protected abstract List<RTSpan<V>> getSpans(Spannable str, Selection selection, SpanCollectMode mode);
/**
* Return an array of the markup objects attached to the specified slice of a Spannable and whose
* type is the specified type or a subclass of it (see Spanned.getSpans(int, int, Class<T>)).
*/
final protected RTSpan<V>[] getSpansAndroid(Spannable str, int selStart, int selEnd) {
RTSpan<V>[] spans = str.getSpans(selStart, selEnd, mSpanClazz);
return spans == null ? (RTSpan<V>[]) Array.newInstance(mSpanClazz) : spans;
}
/**
* @return True if the flags contain at least one of the values, False otherwise.
*/
final protected boolean isOneFlagSet(int flags, int...value) {
for (int flag : value) {
if ((flags & flag) == flag) {
return true;
}
}
return false;
}
}