2023-09-09 21:11:23 +02:00
|
|
|
/* Copyright 2021 Tusky Contributors
|
|
|
|
*
|
|
|
|
* This file is a part of Tusky.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it under the terms of the
|
|
|
|
* GNU General Public License as published by the Free Software Foundation; either version 3 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
|
|
|
|
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
|
|
|
|
* Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along with Tusky; if not,
|
|
|
|
* see <http://www.gnu.org/licenses>. */
|
|
|
|
|
|
|
|
package com.keylesspalace.tusky.adapter;
|
|
|
|
|
|
|
|
import android.content.Context;
|
|
|
|
import android.graphics.Color;
|
|
|
|
import android.graphics.PorterDuff;
|
|
|
|
import android.graphics.Typeface;
|
|
|
|
import android.graphics.drawable.Drawable;
|
|
|
|
import android.text.InputFilter;
|
|
|
|
import android.text.SpannableStringBuilder;
|
|
|
|
import android.text.Spanned;
|
|
|
|
import android.text.TextUtils;
|
|
|
|
import android.text.style.StyleSpan;
|
|
|
|
import android.view.LayoutInflater;
|
|
|
|
import android.view.View;
|
|
|
|
import android.view.ViewGroup;
|
|
|
|
import android.widget.Button;
|
|
|
|
import android.widget.ImageView;
|
|
|
|
import android.widget.TextView;
|
|
|
|
|
|
|
|
import androidx.annotation.ColorRes;
|
|
|
|
import androidx.annotation.DrawableRes;
|
|
|
|
import androidx.annotation.NonNull;
|
|
|
|
import androidx.annotation.Nullable;
|
|
|
|
import androidx.core.content.ContextCompat;
|
|
|
|
import androidx.recyclerview.widget.RecyclerView;
|
|
|
|
|
|
|
|
import com.bumptech.glide.Glide;
|
|
|
|
import com.keylesspalace.tusky.R;
|
|
|
|
import com.keylesspalace.tusky.databinding.ItemFollowRequestBinding;
|
|
|
|
import com.keylesspalace.tusky.databinding.ItemReportNotificationBinding;
|
|
|
|
import com.keylesspalace.tusky.entity.Emoji;
|
|
|
|
import com.keylesspalace.tusky.entity.Notification;
|
|
|
|
import com.keylesspalace.tusky.entity.Status;
|
|
|
|
import com.keylesspalace.tusky.entity.TimelineAccount;
|
|
|
|
import com.keylesspalace.tusky.interfaces.AccountActionListener;
|
|
|
|
import com.keylesspalace.tusky.interfaces.LinkListener;
|
|
|
|
import com.keylesspalace.tusky.interfaces.StatusActionListener;
|
|
|
|
import com.keylesspalace.tusky.util.AbsoluteTimeFormatter;
|
|
|
|
import com.keylesspalace.tusky.util.CardViewMode;
|
|
|
|
import com.keylesspalace.tusky.util.CustomEmojiHelper;
|
|
|
|
import com.keylesspalace.tusky.util.ImageLoadingHelper;
|
|
|
|
import com.keylesspalace.tusky.util.LinkHelper;
|
|
|
|
import com.keylesspalace.tusky.util.SmartLengthInputFilter;
|
|
|
|
import com.keylesspalace.tusky.util.StatusDisplayOptions;
|
|
|
|
import com.keylesspalace.tusky.util.StringUtils;
|
|
|
|
import com.keylesspalace.tusky.util.TimestampUtils;
|
|
|
|
import com.keylesspalace.tusky.viewdata.NotificationViewData;
|
|
|
|
import com.keylesspalace.tusky.viewdata.StatusViewData;
|
|
|
|
|
|
|
|
import java.util.Date;
|
|
|
|
import java.util.List;
|
|
|
|
|
|
|
|
import at.connyduck.sparkbutton.helpers.Utils;
|
|
|
|
|
|
|
|
public class NotificationsAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolder> implements LinkListener{
|
|
|
|
|
|
|
|
public interface AdapterDataSource<T> {
|
|
|
|
int getItemCount();
|
|
|
|
|
|
|
|
T getItemAt(int pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private static final int VIEW_TYPE_STATUS = 0;
|
|
|
|
private static final int VIEW_TYPE_STATUS_NOTIFICATION = 1;
|
|
|
|
private static final int VIEW_TYPE_FOLLOW = 2;
|
|
|
|
private static final int VIEW_TYPE_FOLLOW_REQUEST = 3;
|
|
|
|
private static final int VIEW_TYPE_PLACEHOLDER = 4;
|
|
|
|
private static final int VIEW_TYPE_REPORT = 5;
|
|
|
|
private static final int VIEW_TYPE_UNKNOWN = 6;
|
|
|
|
|
|
|
|
private static final InputFilter[] COLLAPSE_INPUT_FILTER = new InputFilter[]{SmartLengthInputFilter.INSTANCE};
|
|
|
|
private static final InputFilter[] NO_INPUT_FILTER = new InputFilter[0];
|
|
|
|
|
|
|
|
private final String accountId;
|
|
|
|
private StatusDisplayOptions statusDisplayOptions;
|
|
|
|
private final StatusActionListener statusListener;
|
|
|
|
private final NotificationActionListener notificationActionListener;
|
|
|
|
private final AccountActionListener accountActionListener;
|
|
|
|
private final AdapterDataSource<NotificationViewData> dataSource;
|
|
|
|
private final AbsoluteTimeFormatter absoluteTimeFormatter = new AbsoluteTimeFormatter();
|
|
|
|
|
|
|
|
public NotificationsAdapter(String accountId,
|
|
|
|
AdapterDataSource<NotificationViewData> dataSource,
|
|
|
|
StatusDisplayOptions statusDisplayOptions,
|
|
|
|
StatusActionListener statusListener,
|
|
|
|
NotificationActionListener notificationActionListener,
|
|
|
|
AccountActionListener accountActionListener) {
|
|
|
|
|
|
|
|
this.accountId = accountId;
|
|
|
|
this.dataSource = dataSource;
|
|
|
|
this.statusDisplayOptions = statusDisplayOptions;
|
|
|
|
this.statusListener = statusListener;
|
|
|
|
this.notificationActionListener = notificationActionListener;
|
|
|
|
this.accountActionListener = accountActionListener;
|
|
|
|
}
|
|
|
|
|
|
|
|
@NonNull
|
|
|
|
@Override
|
|
|
|
public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
|
|
|
|
LayoutInflater inflater = LayoutInflater.from(parent.getContext());
|
|
|
|
switch (viewType) {
|
|
|
|
case VIEW_TYPE_STATUS: {
|
|
|
|
View view = inflater
|
|
|
|
.inflate(R.layout.item_status, parent, false);
|
|
|
|
return new StatusViewHolder(view);
|
|
|
|
}
|
|
|
|
case VIEW_TYPE_STATUS_NOTIFICATION: {
|
|
|
|
View view = inflater
|
|
|
|
.inflate(R.layout.item_status_notification, parent, false);
|
|
|
|
return new StatusNotificationViewHolder(view, statusDisplayOptions, absoluteTimeFormatter);
|
|
|
|
}
|
|
|
|
case VIEW_TYPE_FOLLOW: {
|
|
|
|
View view = inflater
|
|
|
|
.inflate(R.layout.item_follow, parent, false);
|
|
|
|
return new FollowViewHolder(view, statusDisplayOptions);
|
|
|
|
}
|
|
|
|
case VIEW_TYPE_FOLLOW_REQUEST: {
|
|
|
|
ItemFollowRequestBinding binding = ItemFollowRequestBinding.inflate(inflater, parent, false);
|
|
|
|
return new FollowRequestViewHolder(binding, this, true);
|
|
|
|
}
|
|
|
|
case VIEW_TYPE_PLACEHOLDER: {
|
|
|
|
View view = inflater
|
|
|
|
.inflate(R.layout.item_status_placeholder, parent, false);
|
|
|
|
return new PlaceholderViewHolder(view);
|
|
|
|
}
|
|
|
|
case VIEW_TYPE_REPORT: {
|
|
|
|
ItemReportNotificationBinding binding = ItemReportNotificationBinding.inflate(inflater, parent, false);
|
|
|
|
return new ReportNotificationViewHolder(binding);
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
case VIEW_TYPE_UNKNOWN: {
|
|
|
|
View view = new View(parent.getContext());
|
|
|
|
view.setLayoutParams(
|
|
|
|
new ViewGroup.LayoutParams(
|
|
|
|
ViewGroup.LayoutParams.MATCH_PARENT,
|
|
|
|
Utils.dpToPx(parent.getContext(), 24)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
return new RecyclerView.ViewHolder(view) {
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onBindViewHolder(@NonNull RecyclerView.ViewHolder viewHolder, int position) {
|
|
|
|
bindViewHolder(viewHolder, position, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onBindViewHolder(@NonNull RecyclerView.ViewHolder viewHolder, int position, @NonNull List<Object> payloads) {
|
|
|
|
bindViewHolder(viewHolder, position, payloads);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void bindViewHolder(@NonNull RecyclerView.ViewHolder viewHolder, int position, @Nullable List<Object> payloads) {
|
|
|
|
Object payloadForHolder = payloads != null && !payloads.isEmpty() ? payloads.get(0) : null;
|
|
|
|
if (position < this.dataSource.getItemCount()) {
|
|
|
|
NotificationViewData notification = dataSource.getItemAt(position);
|
|
|
|
if (notification instanceof NotificationViewData.Placeholder) {
|
|
|
|
if (payloadForHolder == null) {
|
|
|
|
NotificationViewData.Placeholder placeholder = ((NotificationViewData.Placeholder) notification);
|
|
|
|
PlaceholderViewHolder holder = (PlaceholderViewHolder) viewHolder;
|
|
|
|
holder.setup(statusListener, placeholder.isLoading());
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
NotificationViewData.Concrete concreteNotification =
|
|
|
|
(NotificationViewData.Concrete) notification;
|
|
|
|
switch (viewHolder.getItemViewType()) {
|
|
|
|
case VIEW_TYPE_STATUS: {
|
|
|
|
StatusViewHolder holder = (StatusViewHolder) viewHolder;
|
|
|
|
StatusViewData.Concrete status = concreteNotification.getStatusViewData();
|
|
|
|
if (status == null) {
|
|
|
|
/* in some very rare cases servers sends null status even though they should not,
|
|
|
|
* we have to handle it somehow */
|
|
|
|
holder.showStatusContent(false);
|
|
|
|
} else {
|
|
|
|
if (payloads == null) {
|
|
|
|
holder.showStatusContent(true);
|
|
|
|
}
|
|
|
|
holder.setupWithStatus(status, statusListener, statusDisplayOptions, payloadForHolder);
|
|
|
|
}
|
|
|
|
if (concreteNotification.getType() == Notification.Type.POLL) {
|
|
|
|
holder.setPollInfo(accountId.equals(concreteNotification.getAccount().getId()));
|
|
|
|
} else {
|
|
|
|
holder.hideStatusInfo();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case VIEW_TYPE_STATUS_NOTIFICATION: {
|
|
|
|
StatusNotificationViewHolder holder = (StatusNotificationViewHolder) viewHolder;
|
|
|
|
StatusViewData.Concrete statusViewData = concreteNotification.getStatusViewData();
|
|
|
|
if (payloadForHolder == null) {
|
|
|
|
if (statusViewData == null) {
|
|
|
|
/* in some very rare cases servers sends null status even though they should not,
|
|
|
|
* we have to handle it somehow */
|
|
|
|
holder.showNotificationContent(false);
|
|
|
|
} else {
|
|
|
|
holder.showNotificationContent(true);
|
|
|
|
|
|
|
|
Status status = statusViewData.getActionable();
|
|
|
|
holder.setDisplayName(status.getAccount().getDisplayName(), status.getAccount().getEmojis());
|
|
|
|
holder.setUsername(status.getAccount().getUsername());
|
|
|
|
holder.setCreatedAt(status.getCreatedAt());
|
|
|
|
|
|
|
|
if (concreteNotification.getType() == Notification.Type.STATUS ||
|
|
|
|
concreteNotification.getType() == Notification.Type.UPDATE) {
|
|
|
|
holder.setAvatar(status.getAccount().getAvatar(), status.getAccount().getBot());
|
|
|
|
} else {
|
|
|
|
holder.setAvatars(status.getAccount().getAvatar(),
|
|
|
|
concreteNotification.getAccount().getAvatar());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
holder.setMessage(concreteNotification, statusListener);
|
|
|
|
holder.setupButtons(notificationActionListener,
|
|
|
|
concreteNotification.getAccount().getId(),
|
|
|
|
concreteNotification.getId());
|
|
|
|
} else {
|
|
|
|
if (payloadForHolder instanceof List)
|
|
|
|
for (Object item : (List<?>) payloadForHolder) {
|
|
|
|
if (StatusBaseViewHolder.Key.KEY_CREATED.equals(item) && statusViewData != null) {
|
|
|
|
holder.setCreatedAt(statusViewData.getStatus().getActionableStatus().getCreatedAt());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case VIEW_TYPE_FOLLOW: {
|
|
|
|
if (payloadForHolder == null) {
|
|
|
|
FollowViewHolder holder = (FollowViewHolder) viewHolder;
|
|
|
|
holder.setMessage(concreteNotification.getAccount(), concreteNotification.getType() == Notification.Type.SIGN_UP);
|
|
|
|
holder.setupButtons(notificationActionListener, concreteNotification.getAccount().getId());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case VIEW_TYPE_FOLLOW_REQUEST: {
|
|
|
|
if (payloadForHolder == null) {
|
|
|
|
FollowRequestViewHolder holder = (FollowRequestViewHolder) viewHolder;
|
|
|
|
holder.setupWithAccount(concreteNotification.getAccount(), statusDisplayOptions.animateAvatars(), statusDisplayOptions.animateEmojis(), statusDisplayOptions.showBotOverlay());
|
|
|
|
holder.setupActionListener(accountActionListener, concreteNotification.getAccount().getId());
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case VIEW_TYPE_REPORT: {
|
|
|
|
if (payloadForHolder == null) {
|
|
|
|
ReportNotificationViewHolder holder = (ReportNotificationViewHolder) viewHolder;
|
|
|
|
holder.setupWithReport(concreteNotification.getAccount(), concreteNotification.getReport(), statusDisplayOptions.animateAvatars(), statusDisplayOptions.animateEmojis());
|
|
|
|
holder.setupActionListener(notificationActionListener, concreteNotification.getReport().getTargetAccount().getId(), concreteNotification.getAccount().getId(), concreteNotification.getReport().getId());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public int getItemCount() {
|
|
|
|
return dataSource.getItemCount();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setMediaPreviewEnabled(boolean mediaPreviewEnabled) {
|
|
|
|
this.statusDisplayOptions = statusDisplayOptions.copy(
|
|
|
|
statusDisplayOptions.animateAvatars(),
|
|
|
|
mediaPreviewEnabled,
|
|
|
|
statusDisplayOptions.useAbsoluteTime(),
|
|
|
|
statusDisplayOptions.showBotOverlay(),
|
|
|
|
statusDisplayOptions.useBlurhash(),
|
|
|
|
CardViewMode.NONE,
|
|
|
|
statusDisplayOptions.confirmReblogs(),
|
|
|
|
statusDisplayOptions.confirmFavourites(),
|
|
|
|
statusDisplayOptions.hideStats(),
|
|
|
|
statusDisplayOptions.animateEmojis(),
|
|
|
|
statusDisplayOptions.showStatsInline(),
|
|
|
|
statusDisplayOptions.showSensitiveMedia(),
|
|
|
|
statusDisplayOptions.openSpoiler()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isMediaPreviewEnabled() {
|
|
|
|
return this.statusDisplayOptions.mediaPreviewEnabled();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public int getItemViewType(int position) {
|
|
|
|
NotificationViewData notification = dataSource.getItemAt(position);
|
|
|
|
if (notification instanceof NotificationViewData.Concrete) {
|
|
|
|
NotificationViewData.Concrete concrete = ((NotificationViewData.Concrete) notification);
|
|
|
|
switch (concrete.getType()) {
|
|
|
|
case MENTION:
|
|
|
|
case POLL: {
|
|
|
|
return VIEW_TYPE_STATUS;
|
|
|
|
}
|
|
|
|
case STATUS:
|
|
|
|
case FAVOURITE:
|
|
|
|
case REBLOG:
|
|
|
|
case UPDATE: {
|
|
|
|
return VIEW_TYPE_STATUS_NOTIFICATION;
|
|
|
|
}
|
|
|
|
case FOLLOW:
|
|
|
|
case SIGN_UP: {
|
|
|
|
return VIEW_TYPE_FOLLOW;
|
|
|
|
}
|
|
|
|
case FOLLOW_REQUEST: {
|
|
|
|
return VIEW_TYPE_FOLLOW_REQUEST;
|
|
|
|
}
|
|
|
|
case REPORT: {
|
|
|
|
return VIEW_TYPE_REPORT;
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
return VIEW_TYPE_UNKNOWN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (notification instanceof NotificationViewData.Placeholder) {
|
|
|
|
return VIEW_TYPE_PLACEHOLDER;
|
|
|
|
} else {
|
|
|
|
throw new AssertionError("Unknown notification type");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public interface NotificationActionListener {
|
|
|
|
void onViewAccount(String id);
|
|
|
|
|
|
|
|
void onViewStatusForNotificationId(String notificationId);
|
|
|
|
|
|
|
|
void onViewReport(String reportId);
|
|
|
|
|
|
|
|
void onExpandedChange(boolean expanded, int position);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when the status {@link android.widget.ToggleButton} responsible for collapsing long
|
|
|
|
* status content is interacted with.
|
|
|
|
*
|
|
|
|
* @param isCollapsed Whether the status content is shown in a collapsed state or fully.
|
|
|
|
* @param position The position of the status in the list.
|
|
|
|
*/
|
|
|
|
void onNotificationContentCollapsedChange(boolean isCollapsed, int position);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static class FollowViewHolder extends RecyclerView.ViewHolder {
|
|
|
|
private final TextView message;
|
|
|
|
private final TextView usernameView;
|
|
|
|
private final TextView displayNameView;
|
|
|
|
private final ImageView avatar;
|
|
|
|
private final StatusDisplayOptions statusDisplayOptions;
|
|
|
|
|
|
|
|
FollowViewHolder(View itemView, StatusDisplayOptions statusDisplayOptions) {
|
|
|
|
super(itemView);
|
|
|
|
message = itemView.findViewById(R.id.notification_text);
|
|
|
|
usernameView = itemView.findViewById(R.id.notification_username);
|
|
|
|
displayNameView = itemView.findViewById(R.id.notification_display_name);
|
|
|
|
avatar = itemView.findViewById(R.id.notification_avatar);
|
|
|
|
this.statusDisplayOptions = statusDisplayOptions;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setMessage(TimelineAccount account, Boolean isSignUp) {
|
|
|
|
Context context = message.getContext();
|
|
|
|
|
|
|
|
String format = context.getString(isSignUp ? R.string.notification_sign_up_format : R.string.notification_follow_format);
|
|
|
|
String wrappedDisplayName = StringUtils.unicodeWrap(account.getName());
|
|
|
|
String wholeMessage = String.format(format, wrappedDisplayName);
|
|
|
|
CharSequence emojifiedMessage = CustomEmojiHelper.emojify(
|
|
|
|
wholeMessage, account.getEmojis(), message, statusDisplayOptions.animateEmojis()
|
|
|
|
);
|
|
|
|
message.setText(emojifiedMessage);
|
|
|
|
|
|
|
|
String username = context.getString(R.string.post_username_format, account.getUsername());
|
|
|
|
usernameView.setText(username);
|
|
|
|
|
|
|
|
CharSequence emojifiedDisplayName = CustomEmojiHelper.emojify(
|
|
|
|
wrappedDisplayName, account.getEmojis(), usernameView, statusDisplayOptions.animateEmojis()
|
|
|
|
);
|
|
|
|
|
|
|
|
displayNameView.setText(emojifiedDisplayName);
|
|
|
|
|
|
|
|
int avatarRadius = avatar.getContext().getResources()
|
|
|
|
.getDimensionPixelSize(R.dimen.avatar_radius_42dp);
|
|
|
|
|
|
|
|
ImageLoadingHelper.loadAvatar(account.getAvatar(), avatar, avatarRadius,
|
|
|
|
statusDisplayOptions.animateAvatars(), null);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void setupButtons(final NotificationActionListener listener, final String accountId) {
|
|
|
|
itemView.setOnClickListener(v -> listener.onViewAccount(accountId));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static class StatusNotificationViewHolder extends RecyclerView.ViewHolder
|
|
|
|
implements View.OnClickListener {
|
|
|
|
|
|
|
|
private final View container;
|
|
|
|
private final TextView message;
|
|
|
|
// private final View statusNameBar;
|
|
|
|
private final TextView displayName;
|
|
|
|
private final TextView username;
|
|
|
|
private final TextView timestampInfo;
|
|
|
|
private final TextView statusContent;
|
|
|
|
private final ImageView statusAvatar;
|
|
|
|
private final ImageView notificationAvatar;
|
|
|
|
private final TextView contentWarningDescriptionTextView;
|
|
|
|
private final Button contentWarningButton;
|
|
|
|
private final Button contentCollapseButton; // TODO: This code SHOULD be based on StatusBaseViewHolder
|
|
|
|
private final StatusDisplayOptions statusDisplayOptions;
|
|
|
|
private final AbsoluteTimeFormatter absoluteTimeFormatter;
|
|
|
|
|
|
|
|
private String accountId;
|
|
|
|
private String notificationId;
|
|
|
|
private NotificationActionListener notificationActionListener;
|
|
|
|
private StatusViewData.Concrete statusViewData;
|
|
|
|
|
|
|
|
private final int avatarRadius48dp;
|
|
|
|
private final int avatarRadius36dp;
|
|
|
|
private final int avatarRadius24dp;
|
|
|
|
|
|
|
|
StatusNotificationViewHolder(
|
|
|
|
View itemView,
|
|
|
|
StatusDisplayOptions statusDisplayOptions,
|
|
|
|
AbsoluteTimeFormatter absoluteTimeFormatter
|
|
|
|
) {
|
|
|
|
super(itemView);
|
|
|
|
message = itemView.findViewById(R.id.notification_top_text);
|
|
|
|
// statusNameBar = itemView.findViewById(R.id.status_name_bar);
|
|
|
|
displayName = itemView.findViewById(R.id.status_display_name);
|
|
|
|
username = itemView.findViewById(R.id.status_username);
|
|
|
|
timestampInfo = itemView.findViewById(R.id.status_meta_info);
|
|
|
|
statusContent = itemView.findViewById(R.id.notification_content);
|
|
|
|
statusAvatar = itemView.findViewById(R.id.notification_status_avatar);
|
|
|
|
notificationAvatar = itemView.findViewById(R.id.notification_notification_avatar);
|
|
|
|
contentWarningDescriptionTextView = itemView.findViewById(R.id.notification_content_warning_description);
|
|
|
|
contentWarningButton = itemView.findViewById(R.id.notification_content_warning_button);
|
|
|
|
contentCollapseButton = itemView.findViewById(R.id.button_toggle_notification_content);
|
|
|
|
|
|
|
|
container = itemView.findViewById(R.id.notification_container);
|
|
|
|
|
|
|
|
this.statusDisplayOptions = statusDisplayOptions;
|
|
|
|
this.absoluteTimeFormatter = absoluteTimeFormatter;
|
|
|
|
|
|
|
|
int darkerFilter = Color.rgb(123, 123, 123);
|
|
|
|
statusAvatar.setColorFilter(darkerFilter, PorterDuff.Mode.MULTIPLY);
|
|
|
|
notificationAvatar.setColorFilter(darkerFilter, PorterDuff.Mode.MULTIPLY);
|
|
|
|
|
|
|
|
itemView.setOnClickListener(this);
|
|
|
|
message.setOnClickListener(this);
|
|
|
|
statusContent.setOnClickListener(this);
|
|
|
|
|
|
|
|
this.avatarRadius48dp = itemView.getContext().getResources().getDimensionPixelSize(R.dimen.avatar_radius_48dp);
|
|
|
|
this.avatarRadius36dp = itemView.getContext().getResources().getDimensionPixelSize(R.dimen.avatar_radius_36dp);
|
|
|
|
this.avatarRadius24dp = itemView.getContext().getResources().getDimensionPixelSize(R.dimen.avatar_radius_24dp);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void showNotificationContent(boolean show) {
|
|
|
|
// statusNameBar.setVisibility(show ? View.VISIBLE : View.GONE);
|
|
|
|
contentWarningDescriptionTextView.setVisibility(show ? View.VISIBLE : View.GONE);
|
|
|
|
contentWarningButton.setVisibility(show ? View.VISIBLE : View.GONE);
|
|
|
|
statusContent.setVisibility(show ? View.VISIBLE : View.GONE);
|
|
|
|
statusAvatar.setVisibility(show ? View.VISIBLE : View.GONE);
|
|
|
|
notificationAvatar.setVisibility(show ? View.VISIBLE : View.GONE);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setDisplayName(String name, List<Emoji> emojis) {
|
|
|
|
CharSequence emojifiedName = CustomEmojiHelper.emojify(name, emojis, displayName, statusDisplayOptions.animateEmojis());
|
|
|
|
displayName.setText(emojifiedName);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setUsername(String name) {
|
|
|
|
Context context = username.getContext();
|
|
|
|
String format = context.getString(R.string.post_username_format);
|
|
|
|
String usernameText = String.format(format, name);
|
|
|
|
username.setText(usernameText);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void setCreatedAt(@Nullable Date createdAt) {
|
|
|
|
if (statusDisplayOptions.useAbsoluteTime()) {
|
|
|
|
timestampInfo.setText(absoluteTimeFormatter.format(createdAt, true));
|
|
|
|
} else {
|
|
|
|
// This is the visible timestampInfo.
|
|
|
|
String readout;
|
|
|
|
/* This one is for screen-readers. Frequently, they would mispronounce timestamps like "17m"
|
|
|
|
* as 17 meters instead of minutes. */
|
|
|
|
CharSequence readoutAloud;
|
|
|
|
if (createdAt != null) {
|
|
|
|
long then = createdAt.getTime();
|
|
|
|
long now = new Date().getTime();
|
|
|
|
readout = TimestampUtils.getRelativeTimeSpanString(timestampInfo.getContext(), then, now);
|
|
|
|
readoutAloud = android.text.format.DateUtils.getRelativeTimeSpanString(then, now,
|
|
|
|
android.text.format.DateUtils.SECOND_IN_MILLIS,
|
|
|
|
android.text.format.DateUtils.FORMAT_ABBREV_RELATIVE);
|
|
|
|
} else {
|
|
|
|
// unknown minutes~
|
|
|
|
readout = "?m";
|
|
|
|
readoutAloud = "? minutes";
|
|
|
|
}
|
|
|
|
timestampInfo.setText(readout);
|
|
|
|
timestampInfo.setContentDescription(readoutAloud);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Drawable getIconWithColor(Context context, @DrawableRes int drawable, @ColorRes int color) {
|
|
|
|
Drawable icon = ContextCompat.getDrawable(context, drawable);
|
|
|
|
if (icon != null) {
|
|
|
|
icon.setColorFilter(context.getColor(color), PorterDuff.Mode.SRC_ATOP);
|
|
|
|
}
|
|
|
|
return icon;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setMessage(NotificationViewData.Concrete notificationViewData, LinkListener listener) {
|
|
|
|
this.statusViewData = notificationViewData.getStatusViewData();
|
|
|
|
|
|
|
|
String displayName = StringUtils.unicodeWrap(notificationViewData.getAccount().getName());
|
|
|
|
Notification.Type type = notificationViewData.getType();
|
|
|
|
|
|
|
|
Context context = message.getContext();
|
|
|
|
String format;
|
|
|
|
Drawable icon;
|
|
|
|
switch (type) {
|
|
|
|
default:
|
|
|
|
case FAVOURITE: {
|
|
|
|
icon = getIconWithColor(context, R.drawable.ic_star_24dp, R.color.tusky_orange);
|
|
|
|
format = context.getString(R.string.notification_favourite_format);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case REBLOG: {
|
|
|
|
icon = getIconWithColor(context, R.drawable.ic_repeat_24dp, R.color.tusky_blue);
|
|
|
|
format = context.getString(R.string.notification_reblog_format);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case STATUS: {
|
|
|
|
icon = getIconWithColor(context, R.drawable.ic_home_24dp, R.color.tusky_blue);
|
|
|
|
format = context.getString(R.string.notification_subscription_format);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case UPDATE: {
|
|
|
|
icon = getIconWithColor(context, R.drawable.ic_edit_24dp, R.color.tusky_blue);
|
|
|
|
format = context.getString(R.string.notification_update_format);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
message.setCompoundDrawablesWithIntrinsicBounds(icon, null, null, null);
|
|
|
|
String wholeMessage = String.format(format, displayName);
|
|
|
|
final SpannableStringBuilder str = new SpannableStringBuilder(wholeMessage);
|
|
|
|
int displayNameIndex = format.indexOf("%s");
|
|
|
|
str.setSpan(
|
|
|
|
new StyleSpan(Typeface.BOLD),
|
|
|
|
displayNameIndex,
|
|
|
|
displayNameIndex + displayName.length(),
|
|
|
|
Spanned.SPAN_EXCLUSIVE_EXCLUSIVE
|
|
|
|
);
|
|
|
|
CharSequence emojifiedText = CustomEmojiHelper.emojify(
|
|
|
|
str, notificationViewData.getAccount().getEmojis(), message, statusDisplayOptions.animateEmojis()
|
|
|
|
);
|
|
|
|
message.setText(emojifiedText);
|
|
|
|
|
|
|
|
if (statusViewData != null) {
|
|
|
|
boolean hasSpoiler = !TextUtils.isEmpty(statusViewData.getStatus().getSpoilerText());
|
|
|
|
contentWarningDescriptionTextView.setVisibility(hasSpoiler ? View.VISIBLE : View.GONE);
|
|
|
|
contentWarningButton.setVisibility(hasSpoiler ? View.VISIBLE : View.GONE);
|
|
|
|
if (statusViewData.isExpanded()) {
|
|
|
|
contentWarningButton.setText(R.string.post_content_warning_show_less);
|
|
|
|
} else {
|
|
|
|
contentWarningButton.setText(R.string.post_content_warning_show_more);
|
|
|
|
}
|
|
|
|
|
|
|
|
contentWarningButton.setOnClickListener(view -> {
|
|
|
|
if (getBindingAdapterPosition() != RecyclerView.NO_POSITION) {
|
|
|
|
notificationActionListener.onExpandedChange(!statusViewData.isExpanded(), getBindingAdapterPosition());
|
|
|
|
}
|
|
|
|
statusContent.setVisibility(statusViewData.isExpanded() ? View.GONE : View.VISIBLE);
|
|
|
|
});
|
|
|
|
|
|
|
|
setupContentAndSpoiler(listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void setupButtons(final NotificationActionListener listener, final String accountId,
|
|
|
|
final String notificationId) {
|
|
|
|
this.notificationActionListener = listener;
|
|
|
|
this.accountId = accountId;
|
|
|
|
this.notificationId = notificationId;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setAvatar(@Nullable String statusAvatarUrl, boolean isBot) {
|
|
|
|
statusAvatar.setPaddingRelative(0, 0, 0, 0);
|
|
|
|
|
|
|
|
ImageLoadingHelper.loadAvatar(statusAvatarUrl,
|
|
|
|
statusAvatar, avatarRadius48dp, statusDisplayOptions.animateAvatars(), null);
|
|
|
|
|
|
|
|
if (statusDisplayOptions.showBotOverlay() && isBot) {
|
|
|
|
notificationAvatar.setVisibility(View.VISIBLE);
|
|
|
|
Glide.with(notificationAvatar)
|
|
|
|
.load(R.drawable.bot_badge)
|
|
|
|
.into(notificationAvatar);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
notificationAvatar.setVisibility(View.GONE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void setAvatars(@Nullable String statusAvatarUrl, @Nullable String notificationAvatarUrl) {
|
|
|
|
int padding = Utils.dpToPx(statusAvatar.getContext(), 12);
|
|
|
|
statusAvatar.setPaddingRelative(0, 0, padding, padding);
|
|
|
|
|
|
|
|
ImageLoadingHelper.loadAvatar(statusAvatarUrl,
|
|
|
|
statusAvatar, avatarRadius36dp, statusDisplayOptions.animateAvatars(), null);
|
|
|
|
|
|
|
|
notificationAvatar.setVisibility(View.VISIBLE);
|
|
|
|
ImageLoadingHelper.loadAvatar(notificationAvatarUrl, notificationAvatar,
|
|
|
|
avatarRadius24dp, statusDisplayOptions.animateAvatars(), null);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onClick(View v) {
|
|
|
|
if (notificationActionListener == null)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (v == container || v == statusContent) {
|
|
|
|
notificationActionListener.onViewStatusForNotificationId(notificationId);
|
|
|
|
}
|
|
|
|
else if (v == message) {
|
|
|
|
notificationActionListener.onViewAccount(accountId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setupContentAndSpoiler(final LinkListener listener) {
|
|
|
|
|
|
|
|
boolean shouldShowContentIfSpoiler = statusViewData.isExpanded();
|
|
|
|
boolean hasSpoiler = !TextUtils.isEmpty(statusViewData.getStatus(). getSpoilerText());
|
|
|
|
if (!shouldShowContentIfSpoiler && hasSpoiler) {
|
|
|
|
statusContent.setVisibility(View.GONE);
|
|
|
|
} else {
|
|
|
|
statusContent.setVisibility(View.VISIBLE);
|
|
|
|
}
|
|
|
|
|
|
|
|
Spanned content = statusViewData.getContent();
|
|
|
|
List<Emoji> emojis = statusViewData.getActionable().getEmojis();
|
|
|
|
|
|
|
|
if (statusViewData.isCollapsible() && (statusViewData.isExpanded() || !hasSpoiler)) {
|
|
|
|
contentCollapseButton.setOnClickListener(view -> {
|
|
|
|
int position = getBindingAdapterPosition();
|
|
|
|
if (position != RecyclerView.NO_POSITION && notificationActionListener != null) {
|
|
|
|
notificationActionListener.onNotificationContentCollapsedChange(!statusViewData.isCollapsed(), position);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
contentCollapseButton.setVisibility(View.VISIBLE);
|
|
|
|
if (statusViewData.isCollapsed()) {
|
|
|
|
contentCollapseButton.setText(R.string.post_content_warning_show_more);
|
|
|
|
statusContent.setFilters(COLLAPSE_INPUT_FILTER);
|
|
|
|
} else {
|
|
|
|
contentCollapseButton.setText(R.string.post_content_warning_show_less);
|
|
|
|
statusContent.setFilters(NO_INPUT_FILTER);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
contentCollapseButton.setVisibility(View.GONE);
|
|
|
|
statusContent.setFilters(NO_INPUT_FILTER);
|
|
|
|
}
|
|
|
|
|
|
|
|
CharSequence emojifiedText = CustomEmojiHelper.emojify(
|
|
|
|
content, emojis, statusContent, statusDisplayOptions.animateEmojis()
|
|
|
|
);
|
|
|
|
LinkHelper.setClickableText(statusContent, emojifiedText, statusViewData.getActionable().getMentions(), statusViewData.getActionable().getTags(), listener);
|
|
|
|
|
2023-09-12 19:25:45 +02:00
|
|
|
CharSequence emojifiedContentWarning = CustomEmojiHelper.emojify(
|
|
|
|
statusViewData.getStatus().getSpoilerText(),
|
2023-09-09 21:11:23 +02:00
|
|
|
statusViewData.getActionable().getEmojis(),
|
|
|
|
contentWarningDescriptionTextView,
|
|
|
|
statusDisplayOptions.animateEmojis()
|
|
|
|
);
|
|
|
|
contentWarningDescriptionTextView.setText(emojifiedContentWarning);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onViewTag(@NonNull String tag) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onViewAccount(@NonNull String id) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onViewUrl(@NonNull String url) {
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|