feat: inline avatars for reblogline
This commit is contained in:
@@ -7,6 +7,7 @@ import android.content.Context;
|
|||||||
import android.graphics.drawable.Drawable;
|
import android.graphics.drawable.Drawable;
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
import android.text.SpannableStringBuilder;
|
import android.text.SpannableStringBuilder;
|
||||||
|
import android.text.Spanned;
|
||||||
import android.util.TypedValue;
|
import android.util.TypedValue;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.view.ViewGroup;
|
import android.view.ViewGroup;
|
||||||
@@ -18,7 +19,10 @@ import org.joinmastodon.android.fragments.BaseStatusListFragment;
|
|||||||
import org.joinmastodon.android.model.Emoji;
|
import org.joinmastodon.android.model.Emoji;
|
||||||
import org.joinmastodon.android.model.Status;
|
import org.joinmastodon.android.model.Status;
|
||||||
import org.joinmastodon.android.model.StatusPrivacy;
|
import org.joinmastodon.android.model.StatusPrivacy;
|
||||||
|
import org.joinmastodon.android.ui.text.AvatarSpan;
|
||||||
|
import org.joinmastodon.android.ui.text.CustomEmojiSpan;
|
||||||
import org.joinmastodon.android.ui.text.HtmlParser;
|
import org.joinmastodon.android.ui.text.HtmlParser;
|
||||||
|
import org.joinmastodon.android.ui.text.SpacerSpan;
|
||||||
import org.joinmastodon.android.ui.utils.CustomEmojiHelper;
|
import org.joinmastodon.android.ui.utils.CustomEmojiHelper;
|
||||||
import org.joinmastodon.android.ui.utils.UiUtils;
|
import org.joinmastodon.android.ui.utils.UiUtils;
|
||||||
|
|
||||||
@@ -53,6 +57,13 @@ public class ReblogOrReplyLineStatusDisplayItem extends StatusDisplayItem{
|
|||||||
SpannableStringBuilder ssb=new SpannableStringBuilder(text);
|
SpannableStringBuilder ssb=new SpannableStringBuilder(text);
|
||||||
if(AccountSessionManager.get(parentFragment.getAccountID()).getLocalPreferences().customEmojiInNames)
|
if(AccountSessionManager.get(parentFragment.getAccountID()).getLocalPreferences().customEmojiInNames)
|
||||||
HtmlParser.parseCustomEmoji(ssb, emojis);
|
HtmlParser.parseCustomEmoji(ssb, emojis);
|
||||||
|
|
||||||
|
if(status.reblog!=null){
|
||||||
|
//add temp chars for span replacement, should be same as spans added below
|
||||||
|
ssb.insert(0, " ");
|
||||||
|
ssb.setSpan(new AvatarSpan(status.account), 0, 1, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
|
||||||
|
ssb.setSpan(new SpacerSpan(15, 20), 1, 2, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
|
||||||
|
}
|
||||||
this.text=ssb;
|
this.text=ssb;
|
||||||
emojiHelper.setText(ssb);
|
emojiHelper.setText(ssb);
|
||||||
this.fullText=fullText;
|
this.fullText=fullText;
|
||||||
|
|||||||
@@ -0,0 +1,50 @@
|
|||||||
|
package org.joinmastodon.android.ui.text;
|
||||||
|
|
||||||
|
import android.graphics.Canvas;
|
||||||
|
import android.graphics.Paint;
|
||||||
|
import android.graphics.Path;
|
||||||
|
import android.graphics.Rect;
|
||||||
|
import android.graphics.drawable.Drawable;
|
||||||
|
import android.text.style.ReplacementSpan;
|
||||||
|
|
||||||
|
import org.joinmastodon.android.GlobalUserPreferences;
|
||||||
|
import org.joinmastodon.android.model.Account;
|
||||||
|
import org.joinmastodon.android.model.Emoji;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
import me.grishka.appkit.imageloader.requests.UrlImageLoaderRequest;
|
||||||
|
import me.grishka.appkit.utils.V;
|
||||||
|
|
||||||
|
public class AvatarSpan extends CustomEmojiSpan{
|
||||||
|
|
||||||
|
public AvatarSpan(Account account){
|
||||||
|
//this is a hacky solution to allow loading of avatars in the middle of strings,
|
||||||
|
//using already existing code for loading emojis
|
||||||
|
super(new Emoji(account.avatarStatic, account.avatar, account.avatarStatic));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void draw(@NonNull Canvas canvas, CharSequence text, int start, int end, float x, int top, int y, int bottom, @NonNull Paint paint){
|
||||||
|
//modified draw of a CustomEmojiSpan, drawing a circular image instead.
|
||||||
|
if(drawable==null)
|
||||||
|
return;
|
||||||
|
int size=Math.round(paint.descent()-paint.ascent());
|
||||||
|
Rect bounds=drawable.getBounds();
|
||||||
|
int dw=drawable.getIntrinsicWidth();
|
||||||
|
int dh=drawable.getIntrinsicHeight();
|
||||||
|
if(bounds.left!=0 || bounds.top!=0 || bounds.right!=dw || bounds.left!=dh){
|
||||||
|
drawable.setBounds(0, 0, dw, dh);
|
||||||
|
}
|
||||||
|
canvas.save();
|
||||||
|
float radius = size / 2f;
|
||||||
|
Path clipPath = new Path();
|
||||||
|
clipPath.addCircle(x + radius, top + radius, radius, Path.Direction.CW);
|
||||||
|
canvas.clipPath(clipPath);
|
||||||
|
canvas.translate(x, top);
|
||||||
|
canvas.scale(size/(float)dw, size/(float)dh, 0f, 0f);
|
||||||
|
drawable.draw(canvas);
|
||||||
|
canvas.restore();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -16,7 +16,7 @@ import me.grishka.appkit.utils.V;
|
|||||||
|
|
||||||
public class CustomEmojiSpan extends ReplacementSpan{
|
public class CustomEmojiSpan extends ReplacementSpan{
|
||||||
public final Emoji emoji;
|
public final Emoji emoji;
|
||||||
private Drawable drawable;
|
protected Drawable drawable;
|
||||||
|
|
||||||
public CustomEmojiSpan(Emoji emoji){
|
public CustomEmojiSpan(Emoji emoji){
|
||||||
this.emoji=emoji;
|
this.emoji=emoji;
|
||||||
|
|||||||
Reference in New Issue
Block a user