mirror of
https://github.com/oxen-io/session-android.git
synced 2025-06-09 09:18:49 +00:00
New group notifications management ui.
This commit is contained in:
parent
a2de8a2a05
commit
c3832cf8b1
@ -3,12 +3,16 @@ package org.thoughtcrime.securesms;
|
|||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
import androidx.appcompat.app.AlertDialog;
|
import androidx.appcompat.app.AlertDialog;
|
||||||
import androidx.fragment.app.FragmentActivity;
|
import androidx.fragment.app.FragmentActivity;
|
||||||
|
import androidx.lifecycle.LiveData;
|
||||||
|
|
||||||
import org.thoughtcrime.securesms.groups.LiveGroup;
|
import org.thoughtcrime.securesms.groups.LiveGroup;
|
||||||
|
import org.thoughtcrime.securesms.groups.ui.GroupMemberEntry;
|
||||||
import org.thoughtcrime.securesms.groups.ui.GroupMemberListView;
|
import org.thoughtcrime.securesms.groups.ui.GroupMemberListView;
|
||||||
import org.thoughtcrime.securesms.recipients.Recipient;
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
||||||
import org.thoughtcrime.securesms.recipients.ui.bottomsheet.RecipientBottomSheetDialogFragment;
|
import org.thoughtcrime.securesms.recipients.ui.bottomsheet.RecipientBottomSheetDialogFragment;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
public final class GroupMembersDialog {
|
public final class GroupMembersDialog {
|
||||||
|
|
||||||
private final FragmentActivity fragmentActivity;
|
private final FragmentActivity fragmentActivity;
|
||||||
@ -33,11 +37,12 @@ public final class GroupMembersDialog {
|
|||||||
GroupMemberListView memberListView = dialog.findViewById(R.id.list_members);
|
GroupMemberListView memberListView = dialog.findViewById(R.id.list_members);
|
||||||
|
|
||||||
LiveGroup liveGroup = new LiveGroup(groupRecipient.requireGroupId());
|
LiveGroup liveGroup = new LiveGroup(groupRecipient.requireGroupId());
|
||||||
|
LiveData<List<GroupMemberEntry.FullMember>> fullMembers = liveGroup.getFullMembers();
|
||||||
|
|
||||||
//noinspection ConstantConditions
|
//noinspection ConstantConditions
|
||||||
liveGroup.getFullMembers().observe(fragmentActivity, memberListView::setMembers);
|
fullMembers.observe(fragmentActivity, memberListView::setMembers);
|
||||||
|
|
||||||
dialog.setOnDismissListener(d -> liveGroup.removeObservers(fragmentActivity));
|
dialog.setOnDismissListener(d -> fullMembers.removeObservers(fragmentActivity));
|
||||||
|
|
||||||
memberListView.setRecipientClickListener(recipient -> {
|
memberListView.setRecipientClickListener(recipient -> {
|
||||||
dialog.dismiss();
|
dialog.dismiss();
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
package org.thoughtcrime.securesms;
|
package org.thoughtcrime.securesms;
|
||||||
|
|
||||||
|
import android.app.Dialog;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.DialogInterface;
|
import android.content.DialogInterface;
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
import androidx.appcompat.app.AlertDialog;
|
import androidx.appcompat.app.AlertDialog;
|
||||||
|
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
@ -23,6 +25,10 @@ public class MuteDialog extends AlertDialog {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public static void show(final Context context, final @NonNull MuteSelectionListener listener) {
|
public static void show(final Context context, final @NonNull MuteSelectionListener listener) {
|
||||||
|
show(context, listener, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void show(final Context context, final @NonNull MuteSelectionListener listener, @Nullable Runnable cancelListener) {
|
||||||
AlertDialog.Builder builder = new AlertDialog.Builder(context);
|
AlertDialog.Builder builder = new AlertDialog.Builder(context);
|
||||||
builder.setTitle(R.string.MuteDialog_mute_notifications);
|
builder.setTitle(R.string.MuteDialog_mute_notifications);
|
||||||
builder.setItems(R.array.mute_durations, new DialogInterface.OnClickListener() {
|
builder.setItems(R.array.mute_durations, new DialogInterface.OnClickListener() {
|
||||||
@ -43,6 +49,13 @@ public class MuteDialog extends AlertDialog {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (cancelListener != null) {
|
||||||
|
builder.setOnCancelListener(dialog -> {
|
||||||
|
cancelListener.run();
|
||||||
|
dialog.dismiss();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
builder.show();
|
builder.show();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,7 @@ import android.content.res.Resources;
|
|||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
import androidx.lifecycle.LiveData;
|
import androidx.lifecycle.LiveData;
|
||||||
import androidx.lifecycle.MediatorLiveData;
|
import androidx.lifecycle.MutableLiveData;
|
||||||
import androidx.lifecycle.Transformations;
|
import androidx.lifecycle.Transformations;
|
||||||
|
|
||||||
import com.annimon.stream.ComparatorCompat;
|
import com.annimon.stream.ComparatorCompat;
|
||||||
@ -16,16 +16,16 @@ import org.thoughtcrime.securesms.database.DatabaseFactory;
|
|||||||
import org.thoughtcrime.securesms.database.GroupDatabase;
|
import org.thoughtcrime.securesms.database.GroupDatabase;
|
||||||
import org.thoughtcrime.securesms.dependencies.ApplicationDependencies;
|
import org.thoughtcrime.securesms.dependencies.ApplicationDependencies;
|
||||||
import org.thoughtcrime.securesms.groups.ui.GroupMemberEntry;
|
import org.thoughtcrime.securesms.groups.ui.GroupMemberEntry;
|
||||||
|
import org.thoughtcrime.securesms.recipients.LiveRecipient;
|
||||||
import org.thoughtcrime.securesms.recipients.Recipient;
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
||||||
import org.thoughtcrime.securesms.recipients.RecipientId;
|
import org.thoughtcrime.securesms.recipients.RecipientId;
|
||||||
import org.thoughtcrime.securesms.util.concurrent.SignalExecutors;
|
import org.thoughtcrime.securesms.util.concurrent.SignalExecutors;
|
||||||
import org.thoughtcrime.securesms.util.livedata.LiveDataUtil;
|
import org.thoughtcrime.securesms.util.livedata.LiveDataUtil;
|
||||||
import org.whispersystems.libsignal.util.guava.Optional;
|
|
||||||
|
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public final class LiveGroup extends MediatorLiveData<GroupDatabase.GroupRecord> {
|
public final class LiveGroup {
|
||||||
|
|
||||||
private static final Comparator<GroupMemberEntry.FullMember> LOCAL_FIRST = (m1, m2) -> Boolean.compare(m2.getMember().isLocalNumber(), m1.getMember().isLocalNumber());
|
private static final Comparator<GroupMemberEntry.FullMember> LOCAL_FIRST = (m1, m2) -> Boolean.compare(m2.getMember().isLocalNumber(), m1.getMember().isLocalNumber());
|
||||||
private static final Comparator<GroupMemberEntry.FullMember> ADMIN_FIRST = (m1, m2) -> Boolean.compare(m2.isAdmin(), m1.isAdmin());
|
private static final Comparator<GroupMemberEntry.FullMember> ADMIN_FIRST = (m1, m2) -> Boolean.compare(m2.isAdmin(), m1.isAdmin());
|
||||||
@ -34,53 +34,49 @@ public final class LiveGroup extends MediatorLiveData<GroupDatabase.GroupRecord>
|
|||||||
|
|
||||||
private final GroupDatabase groupDatabase;
|
private final GroupDatabase groupDatabase;
|
||||||
private final LiveData<Recipient> recipient;
|
private final LiveData<Recipient> recipient;
|
||||||
|
private final LiveData<GroupDatabase.GroupRecord> groupRecord;
|
||||||
|
|
||||||
public LiveGroup(@NonNull GroupId groupId) {
|
public LiveGroup(@NonNull GroupId groupId) {
|
||||||
Context context = ApplicationDependencies.getApplication();
|
Context context = ApplicationDependencies.getApplication();
|
||||||
|
MutableLiveData<LiveRecipient> liveRecipient = new MutableLiveData<>();
|
||||||
|
|
||||||
this.groupDatabase = DatabaseFactory.getGroupDatabase(context);
|
this.groupDatabase = DatabaseFactory.getGroupDatabase(context);
|
||||||
|
this.recipient = Transformations.switchMap(liveRecipient, LiveRecipient::getLiveData);
|
||||||
|
this.groupRecord = LiveDataUtil.filterNotNull(LiveDataUtil.mapAsync(recipient, groupRecipient-> groupDatabase.getGroup(groupRecipient.getId()).orNull()));
|
||||||
|
|
||||||
recipient = Recipient.externalGroup(context, groupId).live().getLiveData();
|
SignalExecutors.BOUNDED.execute(() -> liveRecipient.postValue(Recipient.externalGroup(context, groupId).live()));
|
||||||
|
|
||||||
addSource(recipient, this::refresh);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void refresh(@NonNull Recipient groupRecipient) {
|
|
||||||
SignalExecutors.BOUNDED.execute(() -> {
|
|
||||||
Optional<GroupDatabase.GroupRecord> group = groupDatabase.getGroup(groupRecipient.getId());
|
|
||||||
if (group.isPresent()) {
|
|
||||||
postValue(group.get());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public LiveData<String> getTitle() {
|
public LiveData<String> getTitle() {
|
||||||
return Transformations.map(this, GroupDatabase.GroupRecord::getTitle);
|
return Transformations.map(groupRecord, GroupDatabase.GroupRecord::getTitle);
|
||||||
|
}
|
||||||
|
|
||||||
|
public LiveData<Recipient> getGroupRecipient() {
|
||||||
|
return recipient;
|
||||||
}
|
}
|
||||||
|
|
||||||
public LiveData<Boolean> isSelfAdmin() {
|
public LiveData<Boolean> isSelfAdmin() {
|
||||||
return Transformations.map(this, g -> g.isAdmin(Recipient.self()));
|
return Transformations.map(groupRecord, g -> g.isAdmin(Recipient.self()));
|
||||||
}
|
}
|
||||||
|
|
||||||
public LiveData<Boolean> getRecipientIsAdmin(@NonNull RecipientId recipientId) {
|
public LiveData<Boolean> getRecipientIsAdmin(@NonNull RecipientId recipientId) {
|
||||||
return LiveDataUtil.mapAsync(this, g -> g.isAdmin(Recipient.resolved(recipientId)));
|
return LiveDataUtil.mapAsync(groupRecord, g -> g.isAdmin(Recipient.resolved(recipientId)));
|
||||||
}
|
}
|
||||||
|
|
||||||
public LiveData<Integer> getPendingMemberCount() {
|
public LiveData<Integer> getPendingMemberCount() {
|
||||||
return Transformations.map(this, g -> g.isV2Group() ? g.requireV2GroupProperties().getDecryptedGroup().getPendingMembersCount() : 0);
|
return Transformations.map(groupRecord, g -> g.isV2Group() ? g.requireV2GroupProperties().getDecryptedGroup().getPendingMembersCount() : 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
public LiveData<GroupAccessControl> getMembershipAdditionAccessControl() {
|
public LiveData<GroupAccessControl> getMembershipAdditionAccessControl() {
|
||||||
return Transformations.map(this, GroupDatabase.GroupRecord::getMembershipAdditionAccessControl);
|
return Transformations.map(groupRecord, GroupDatabase.GroupRecord::getMembershipAdditionAccessControl);
|
||||||
}
|
}
|
||||||
|
|
||||||
public LiveData<GroupAccessControl> getAttributesAccessControl() {
|
public LiveData<GroupAccessControl> getAttributesAccessControl() {
|
||||||
return Transformations.map(this, GroupDatabase.GroupRecord::getAttributesAccessControl);
|
return Transformations.map(groupRecord, GroupDatabase.GroupRecord::getAttributesAccessControl);
|
||||||
}
|
}
|
||||||
|
|
||||||
public LiveData<List<GroupMemberEntry.FullMember>> getFullMembers() {
|
public LiveData<List<GroupMemberEntry.FullMember>> getFullMembers() {
|
||||||
return LiveDataUtil.mapAsync(this,
|
return LiveDataUtil.mapAsync(groupRecord,
|
||||||
g -> Stream.of(g.getMembers())
|
g -> Stream.of(g.getMembers())
|
||||||
.map(m -> {
|
.map(m -> {
|
||||||
Recipient recipient = Recipient.resolved(m);
|
Recipient recipient = Recipient.resolved(m);
|
||||||
|
@ -11,16 +11,20 @@ import android.view.MenuItem;
|
|||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.view.ViewGroup;
|
import android.view.ViewGroup;
|
||||||
import android.widget.Button;
|
import android.widget.Button;
|
||||||
|
import android.widget.CompoundButton;
|
||||||
|
import android.widget.Switch;
|
||||||
import android.widget.TextView;
|
import android.widget.TextView;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
import androidx.annotation.Nullable;
|
import androidx.annotation.Nullable;
|
||||||
|
import androidx.constraintlayout.widget.Group;
|
||||||
import androidx.core.view.ViewCompat;
|
import androidx.core.view.ViewCompat;
|
||||||
import androidx.fragment.app.Fragment;
|
import androidx.fragment.app.Fragment;
|
||||||
import androidx.fragment.app.FragmentActivity;
|
import androidx.fragment.app.FragmentActivity;
|
||||||
import androidx.lifecycle.ViewModelProviders;
|
import androidx.lifecycle.ViewModelProviders;
|
||||||
|
|
||||||
import org.thoughtcrime.securesms.MediaPreviewActivity;
|
import org.thoughtcrime.securesms.MediaPreviewActivity;
|
||||||
|
import org.thoughtcrime.securesms.MuteDialog;
|
||||||
import org.thoughtcrime.securesms.R;
|
import org.thoughtcrime.securesms.R;
|
||||||
import org.thoughtcrime.securesms.components.AvatarImageView;
|
import org.thoughtcrime.securesms.components.AvatarImageView;
|
||||||
import org.thoughtcrime.securesms.components.ThreadPhotoRailView;
|
import org.thoughtcrime.securesms.components.ThreadPhotoRailView;
|
||||||
@ -28,13 +32,17 @@ import org.thoughtcrime.securesms.groups.GroupId;
|
|||||||
import org.thoughtcrime.securesms.groups.ui.GroupMemberListView;
|
import org.thoughtcrime.securesms.groups.ui.GroupMemberListView;
|
||||||
import org.thoughtcrime.securesms.groups.ui.LeaveGroupDialog;
|
import org.thoughtcrime.securesms.groups.ui.LeaveGroupDialog;
|
||||||
import org.thoughtcrime.securesms.groups.ui.managegroup.dialogs.GroupRightsDialog;
|
import org.thoughtcrime.securesms.groups.ui.managegroup.dialogs.GroupRightsDialog;
|
||||||
|
import org.thoughtcrime.securesms.groups.ui.notifications.CustomNotificationsDialogFragment;
|
||||||
import org.thoughtcrime.securesms.groups.ui.pendingmemberinvites.PendingMemberInvitesActivity;
|
import org.thoughtcrime.securesms.groups.ui.pendingmemberinvites.PendingMemberInvitesActivity;
|
||||||
import org.thoughtcrime.securesms.logging.Log;
|
import org.thoughtcrime.securesms.logging.Log;
|
||||||
import org.thoughtcrime.securesms.mediaoverview.MediaOverviewActivity;
|
import org.thoughtcrime.securesms.mediaoverview.MediaOverviewActivity;
|
||||||
import org.thoughtcrime.securesms.mms.GlideApp;
|
import org.thoughtcrime.securesms.mms.GlideApp;
|
||||||
|
import org.thoughtcrime.securesms.notifications.NotificationChannels;
|
||||||
import org.thoughtcrime.securesms.profiles.edit.EditProfileActivity;
|
import org.thoughtcrime.securesms.profiles.edit.EditProfileActivity;
|
||||||
import org.thoughtcrime.securesms.recipients.ui.bottomsheet.RecipientBottomSheetDialogFragment;
|
import org.thoughtcrime.securesms.recipients.ui.bottomsheet.RecipientBottomSheetDialogFragment;
|
||||||
|
import org.thoughtcrime.securesms.util.DateUtils;
|
||||||
|
|
||||||
|
import java.util.Locale;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
|
||||||
public class ManageGroupFragment extends Fragment {
|
public class ManageGroupFragment extends Fragment {
|
||||||
@ -61,6 +69,10 @@ public class ManageGroupFragment extends Fragment {
|
|||||||
private Button disappearingMessages;
|
private Button disappearingMessages;
|
||||||
private Button blockGroup;
|
private Button blockGroup;
|
||||||
private Button leaveGroup;
|
private Button leaveGroup;
|
||||||
|
private Switch muteNotificationsSwitch;
|
||||||
|
private TextView muteNotificationsUntilLabel;
|
||||||
|
private TextView customNotificationsButton;
|
||||||
|
private Group customNotificationsControls;
|
||||||
|
|
||||||
static ManageGroupFragment newInstance(@NonNull String groupId) {
|
static ManageGroupFragment newInstance(@NonNull String groupId) {
|
||||||
ManageGroupFragment fragment = new ManageGroupFragment();
|
ManageGroupFragment fragment = new ManageGroupFragment();
|
||||||
@ -100,6 +112,10 @@ public class ManageGroupFragment extends Fragment {
|
|||||||
disappearingMessages = view.findViewById(R.id.disappearing_messages);
|
disappearingMessages = view.findViewById(R.id.disappearing_messages);
|
||||||
blockGroup = view.findViewById(R.id.blockGroup);
|
blockGroup = view.findViewById(R.id.blockGroup);
|
||||||
leaveGroup = view.findViewById(R.id.leaveGroup);
|
leaveGroup = view.findViewById(R.id.leaveGroup);
|
||||||
|
muteNotificationsUntilLabel = view.findViewById(R.id.group_mute_notifications_until);
|
||||||
|
muteNotificationsSwitch = view.findViewById(R.id.group_mute_notifications_switch);
|
||||||
|
customNotificationsButton = view.findViewById(R.id.group_custom_notifications_button);
|
||||||
|
customNotificationsControls = view.findViewById(R.id.group_custom_notifications_controls);
|
||||||
|
|
||||||
return view;
|
return view;
|
||||||
}
|
}
|
||||||
@ -185,6 +201,47 @@ public class ManageGroupFragment extends Fragment {
|
|||||||
viewModel.getCanEditGroupAttributes().observe(getViewLifecycleOwner(), canEdit -> disappearingMessages.setEnabled(canEdit));
|
viewModel.getCanEditGroupAttributes().observe(getViewLifecycleOwner(), canEdit -> disappearingMessages.setEnabled(canEdit));
|
||||||
|
|
||||||
groupMemberList.setRecipientClickListener(recipient -> RecipientBottomSheetDialogFragment.create(recipient.getId(), groupId).show(requireFragmentManager(), "BOTTOM"));
|
groupMemberList.setRecipientClickListener(recipient -> RecipientBottomSheetDialogFragment.create(recipient.getId(), groupId).show(requireFragmentManager(), "BOTTOM"));
|
||||||
|
|
||||||
|
final CompoundButton.OnCheckedChangeListener muteSwitchListener = (buttonView, isChecked) -> {
|
||||||
|
if (isChecked) {
|
||||||
|
MuteDialog.show(context, viewModel::setMuteUntil, () -> muteNotificationsSwitch.setChecked(false));
|
||||||
|
} else {
|
||||||
|
viewModel.clearMuteUntil();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
viewModel.getMuteState().observe(getViewLifecycleOwner(), muteState -> {
|
||||||
|
if (muteNotificationsSwitch.isChecked() != muteState.isMuted()) {
|
||||||
|
muteNotificationsSwitch.setOnCheckedChangeListener(null);
|
||||||
|
muteNotificationsSwitch.setChecked(muteState.isMuted());
|
||||||
|
}
|
||||||
|
|
||||||
|
muteNotificationsSwitch.setEnabled(true);
|
||||||
|
muteNotificationsSwitch.setOnCheckedChangeListener(muteSwitchListener);
|
||||||
|
muteNotificationsUntilLabel.setVisibility(muteState.isMuted() ? View.VISIBLE : View.GONE);
|
||||||
|
|
||||||
|
if (muteState.isMuted()) {
|
||||||
|
muteNotificationsUntilLabel.setText(getString(R.string.ManageGroupActivity_until_s,
|
||||||
|
DateUtils.getTimeString(requireContext(),
|
||||||
|
Locale.getDefault(),
|
||||||
|
muteState.getMutedUntil())));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (NotificationChannels.supported()) {
|
||||||
|
customNotificationsControls.setVisibility(View.VISIBLE);
|
||||||
|
|
||||||
|
customNotificationsButton.setOnClickListener(v -> CustomNotificationsDialogFragment.create(groupId)
|
||||||
|
.show(requireFragmentManager(), "CUSTOM_NOTIFICATIONS"));
|
||||||
|
|
||||||
|
//noinspection CodeBlock2Expr
|
||||||
|
viewModel.hasCustomNotifications().observe(getViewLifecycleOwner(), hasCustomNotifications -> {
|
||||||
|
customNotificationsButton.setText(hasCustomNotifications ? R.string.ManageGroupActivity_on
|
||||||
|
: R.string.ManageGroupActivity_off);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
customNotificationsControls.setVisibility(View.GONE);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -17,6 +17,7 @@ import org.thoughtcrime.securesms.groups.GroupInsufficientRightsException;
|
|||||||
import org.thoughtcrime.securesms.groups.GroupManager;
|
import org.thoughtcrime.securesms.groups.GroupManager;
|
||||||
import org.thoughtcrime.securesms.logging.Log;
|
import org.thoughtcrime.securesms.logging.Log;
|
||||||
import org.thoughtcrime.securesms.recipients.Recipient;
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
||||||
|
import org.thoughtcrime.securesms.recipients.RecipientId;
|
||||||
import org.thoughtcrime.securesms.util.concurrent.SignalExecutors;
|
import org.thoughtcrime.securesms.util.concurrent.SignalExecutors;
|
||||||
import org.thoughtcrime.securesms.util.concurrent.SimpleTask;
|
import org.thoughtcrime.securesms.util.concurrent.SimpleTask;
|
||||||
|
|
||||||
@ -101,6 +102,13 @@ final class ManageGroupRepository {
|
|||||||
recipientCallback::accept);
|
recipientCallback::accept);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void setMuteUntil(long until) {
|
||||||
|
SignalExecutors.BOUNDED.execute(() -> {
|
||||||
|
RecipientId recipientId = Recipient.externalGroup(context, groupId).getId();
|
||||||
|
DatabaseFactory.getRecipientDatabase(context).setMuted(recipientId, until);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
static final class GroupStateResult {
|
static final class GroupStateResult {
|
||||||
|
|
||||||
private final long threadId;
|
private final long threadId;
|
||||||
|
@ -26,7 +26,6 @@ import org.thoughtcrime.securesms.recipients.Recipient;
|
|||||||
import org.thoughtcrime.securesms.recipients.RecipientUtil;
|
import org.thoughtcrime.securesms.recipients.RecipientUtil;
|
||||||
import org.thoughtcrime.securesms.util.ExpirationUtil;
|
import org.thoughtcrime.securesms.util.ExpirationUtil;
|
||||||
import org.thoughtcrime.securesms.util.Util;
|
import org.thoughtcrime.securesms.util.Util;
|
||||||
import org.thoughtcrime.securesms.util.livedata.LiveDataUtil;
|
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
@ -44,6 +43,8 @@ public class ManageGroupViewModel extends ViewModel {
|
|||||||
private final LiveData<GroupAccessControl> editMembershipRights;
|
private final LiveData<GroupAccessControl> editMembershipRights;
|
||||||
private final LiveData<GroupAccessControl> editGroupAttributesRights;
|
private final LiveData<GroupAccessControl> editGroupAttributesRights;
|
||||||
private final MutableLiveData<GroupViewState> groupViewState = new MutableLiveData<>(null);
|
private final MutableLiveData<GroupViewState> groupViewState = new MutableLiveData<>(null);
|
||||||
|
private final LiveData<MuteState> muteState;
|
||||||
|
private final LiveData<Boolean> hasCustomNotifications;
|
||||||
|
|
||||||
private ManageGroupViewModel(@NonNull Context context, @NonNull ManageGroupRepository manageGroupRepository) {
|
private ManageGroupViewModel(@NonNull Context context, @NonNull ManageGroupRepository manageGroupRepository) {
|
||||||
this.context = context;
|
this.context = context;
|
||||||
@ -62,6 +63,10 @@ public class ManageGroupViewModel extends ViewModel {
|
|||||||
this.editGroupAttributesRights = liveGroup.getAttributesAccessControl();
|
this.editGroupAttributesRights = liveGroup.getAttributesAccessControl();
|
||||||
this.disappearingMessageTimer = Transformations.map(liveGroup.getExpireMessages(), expiration -> ExpirationUtil.getExpirationDisplayValue(context, expiration));
|
this.disappearingMessageTimer = Transformations.map(liveGroup.getExpireMessages(), expiration -> ExpirationUtil.getExpirationDisplayValue(context, expiration));
|
||||||
this.canEditGroupAttributes = liveGroup.selfCanEditGroupAttributes();
|
this.canEditGroupAttributes = liveGroup.selfCanEditGroupAttributes();
|
||||||
|
this.muteState = Transformations.map(liveGroup.getGroupRecipient(),
|
||||||
|
recipient -> new MuteState(recipient.getMuteUntil(), recipient.isMuted()));
|
||||||
|
this.hasCustomNotifications = Transformations.map(liveGroup.getGroupRecipient(),
|
||||||
|
recipient -> recipient.getNotificationChannel() != null);
|
||||||
}
|
}
|
||||||
|
|
||||||
@WorkerThread
|
@WorkerThread
|
||||||
@ -91,6 +96,10 @@ public class ManageGroupViewModel extends ViewModel {
|
|||||||
return title;
|
return title;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LiveData<MuteState> getMuteState() {
|
||||||
|
return muteState;
|
||||||
|
}
|
||||||
|
|
||||||
LiveData<GroupAccessControl> getMembershipRights() {
|
LiveData<GroupAccessControl> getMembershipRights() {
|
||||||
return editMembershipRights;
|
return editMembershipRights;
|
||||||
}
|
}
|
||||||
@ -111,6 +120,10 @@ public class ManageGroupViewModel extends ViewModel {
|
|||||||
return disappearingMessageTimer;
|
return disappearingMessageTimer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LiveData<Boolean> hasCustomNotifications() {
|
||||||
|
return hasCustomNotifications;
|
||||||
|
}
|
||||||
|
|
||||||
void handleExpirationSelection() {
|
void handleExpirationSelection() {
|
||||||
manageGroupRepository.getRecipient(groupRecipient ->
|
manageGroupRepository.getRecipient(groupRecipient ->
|
||||||
ExpirationDialog.show(context,
|
ExpirationDialog.show(context,
|
||||||
@ -131,6 +144,14 @@ public class ManageGroupViewModel extends ViewModel {
|
|||||||
() -> RecipientUtil.block(context, recipient)));
|
() -> RecipientUtil.block(context, recipient)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setMuteUntil(long muteUntil) {
|
||||||
|
manageGroupRepository.setMuteUntil(muteUntil);
|
||||||
|
}
|
||||||
|
|
||||||
|
void clearMuteUntil() {
|
||||||
|
manageGroupRepository.setMuteUntil(0);
|
||||||
|
}
|
||||||
|
|
||||||
@WorkerThread
|
@WorkerThread
|
||||||
private void showErrorToast(@NonNull ManageGroupRepository.FailureReason e) {
|
private void showErrorToast(@NonNull ManageGroupRepository.FailureReason e) {
|
||||||
Util.runOnMain(() -> Toast.makeText(context, e.getToastMessage(), Toast.LENGTH_SHORT).show());
|
Util.runOnMain(() -> Toast.makeText(context, e.getToastMessage(), Toast.LENGTH_SHORT).show());
|
||||||
@ -163,6 +184,24 @@ public class ManageGroupViewModel extends ViewModel {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static final class MuteState {
|
||||||
|
private final long mutedUntil;
|
||||||
|
private final boolean isMuted;
|
||||||
|
|
||||||
|
MuteState(long mutedUntil, boolean isMuted) {
|
||||||
|
this.mutedUntil = mutedUntil;
|
||||||
|
this.isMuted = isMuted;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getMutedUntil() {
|
||||||
|
return mutedUntil;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isMuted() {
|
||||||
|
return isMuted;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
interface CursorFactory {
|
interface CursorFactory {
|
||||||
Cursor create();
|
Cursor create();
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,176 @@
|
|||||||
|
package org.thoughtcrime.securesms.groups.ui.notifications;
|
||||||
|
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.media.Ringtone;
|
||||||
|
import android.media.RingtoneManager;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.provider.Settings;
|
||||||
|
import android.view.LayoutInflater;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.widget.CompoundButton;
|
||||||
|
import android.widget.Switch;
|
||||||
|
import android.widget.TextView;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
import androidx.appcompat.widget.Toolbar;
|
||||||
|
import androidx.fragment.app.DialogFragment;
|
||||||
|
import androidx.lifecycle.ViewModelProviders;
|
||||||
|
|
||||||
|
import org.thoughtcrime.securesms.R;
|
||||||
|
import org.thoughtcrime.securesms.database.RecipientDatabase;
|
||||||
|
import org.thoughtcrime.securesms.groups.GroupId;
|
||||||
|
import org.thoughtcrime.securesms.util.ThemeUtil;
|
||||||
|
|
||||||
|
public class CustomNotificationsDialogFragment extends DialogFragment {
|
||||||
|
|
||||||
|
private static final short RINGTONE_PICKER_REQUEST_CODE = 13562;
|
||||||
|
|
||||||
|
private static final String ARG_GROUP_ID = "group_id";
|
||||||
|
|
||||||
|
private Switch customNotificationsSwitch;
|
||||||
|
private View soundLabel;
|
||||||
|
private TextView soundSelector;
|
||||||
|
private View vibrateLabel;
|
||||||
|
private Switch vibrateSwitch;
|
||||||
|
|
||||||
|
private CustomNotificationsViewModel viewModel;
|
||||||
|
|
||||||
|
public static DialogFragment create(@NonNull GroupId groupId) {
|
||||||
|
DialogFragment fragment = new CustomNotificationsDialogFragment();
|
||||||
|
Bundle args = new Bundle();
|
||||||
|
|
||||||
|
args.putString(ARG_GROUP_ID, groupId.toString());
|
||||||
|
fragment.setArguments(args);
|
||||||
|
|
||||||
|
return fragment;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onCreate(@Nullable Bundle savedInstanceState) {
|
||||||
|
super.onCreate(savedInstanceState);
|
||||||
|
if (ThemeUtil.isDarkTheme(requireActivity())) {
|
||||||
|
setStyle(STYLE_NO_FRAME, R.style.TextSecure_DarkTheme);
|
||||||
|
} else {
|
||||||
|
setStyle(STYLE_NO_FRAME, R.style.TextSecure_LightTheme);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public @Nullable View onCreateView(@NonNull LayoutInflater inflater,
|
||||||
|
@Nullable ViewGroup container,
|
||||||
|
@Nullable Bundle savedInstanceState)
|
||||||
|
{
|
||||||
|
return inflater.inflate(R.layout.custom_notifications_dialog_fragment, container, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
|
||||||
|
initializeViewModel();
|
||||||
|
initializeViews(view);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
|
||||||
|
if (requestCode == RINGTONE_PICKER_REQUEST_CODE && resultCode == Activity.RESULT_OK && data != null) {
|
||||||
|
Uri uri = data.getParcelableExtra(RingtoneManager.EXTRA_RINGTONE_PICKED_URI);
|
||||||
|
|
||||||
|
viewModel.setMessageSound(uri);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void initializeViewModel() {
|
||||||
|
Bundle arguments = requireArguments();
|
||||||
|
GroupId groupId = GroupId.parseOrThrow(arguments.getString(ARG_GROUP_ID, ""));
|
||||||
|
CustomNotificationsRepository repository = new CustomNotificationsRepository(requireContext(), groupId);
|
||||||
|
CustomNotificationsViewModel.Factory factory = new CustomNotificationsViewModel.Factory(groupId, repository);
|
||||||
|
|
||||||
|
viewModel = ViewModelProviders.of(this, factory).get(CustomNotificationsViewModel.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void initializeViews(@NonNull View view) {
|
||||||
|
customNotificationsSwitch = view.findViewById(R.id.custom_notifications_enable_switch);
|
||||||
|
soundLabel = view.findViewById(R.id.custom_notifications_sound_label);
|
||||||
|
soundSelector = view.findViewById(R.id.custom_notifications_sound_selection);
|
||||||
|
vibrateLabel = view.findViewById(R.id.custom_notifications_vibrate_label);
|
||||||
|
vibrateSwitch = view.findViewById(R.id.custom_notifications_vibrate_switch);
|
||||||
|
|
||||||
|
Toolbar toolbar = view.findViewById(R.id.custom_notifications_toolbar);
|
||||||
|
|
||||||
|
toolbar.setNavigationOnClickListener(v -> requireActivity().finish());
|
||||||
|
|
||||||
|
CompoundButton.OnCheckedChangeListener onCustomNotificationsSwitchCheckChangedListener = (buttonView, isChecked) -> {
|
||||||
|
viewModel.setHasCustomNotifications(isChecked);
|
||||||
|
};
|
||||||
|
|
||||||
|
viewModel.isInitialLoadComplete().observe(getViewLifecycleOwner(), customNotificationsSwitch::setEnabled);
|
||||||
|
|
||||||
|
viewModel.hasCustomNotifications().observe(getViewLifecycleOwner(), hasCustomNotifications -> {
|
||||||
|
if (customNotificationsSwitch.isChecked() != hasCustomNotifications) {
|
||||||
|
customNotificationsSwitch.setOnCheckedChangeListener(null);
|
||||||
|
customNotificationsSwitch.setChecked(hasCustomNotifications);
|
||||||
|
}
|
||||||
|
|
||||||
|
customNotificationsSwitch.setOnCheckedChangeListener(onCustomNotificationsSwitchCheckChangedListener);
|
||||||
|
|
||||||
|
soundLabel.setEnabled(hasCustomNotifications);
|
||||||
|
vibrateLabel.setEnabled(hasCustomNotifications);
|
||||||
|
soundSelector.setVisibility(hasCustomNotifications ? View.VISIBLE : View.GONE);
|
||||||
|
vibrateSwitch.setVisibility(hasCustomNotifications ? View.VISIBLE : View.GONE);
|
||||||
|
});
|
||||||
|
|
||||||
|
CompoundButton.OnCheckedChangeListener onVibrateSwitchCheckChangedListener = (buttonView, isChecked) -> {
|
||||||
|
viewModel.setMessageVibrate(isChecked ? RecipientDatabase.VibrateState.ENABLED : RecipientDatabase.VibrateState.DISABLED);
|
||||||
|
};
|
||||||
|
|
||||||
|
viewModel.getVibrateState().observe(getViewLifecycleOwner(), vibrateState -> {
|
||||||
|
boolean vibrateEnabled = vibrateState != RecipientDatabase.VibrateState.DISABLED;
|
||||||
|
|
||||||
|
if (vibrateSwitch.isChecked() != vibrateEnabled) {
|
||||||
|
vibrateSwitch.setOnCheckedChangeListener(null);
|
||||||
|
vibrateSwitch.setChecked(vibrateEnabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
vibrateSwitch.setOnCheckedChangeListener(onVibrateSwitchCheckChangedListener);
|
||||||
|
});
|
||||||
|
|
||||||
|
viewModel.getNotificationSound().observe(getViewLifecycleOwner(), sound -> {
|
||||||
|
soundSelector.setText(getRingtoneSummary(requireContext(), sound));
|
||||||
|
soundSelector.setTag(sound);
|
||||||
|
});
|
||||||
|
|
||||||
|
soundSelector.setOnClickListener(v -> launchSoundSelector(viewModel.getNotificationSound().getValue()));
|
||||||
|
}
|
||||||
|
|
||||||
|
private @NonNull String getRingtoneSummary(@NonNull Context context, @Nullable Uri ringtone) {
|
||||||
|
if (ringtone == null) {
|
||||||
|
return context.getString(R.string.preferences__default);
|
||||||
|
} else if (ringtone.toString().isEmpty()) {
|
||||||
|
return context.getString(R.string.preferences__silent);
|
||||||
|
} else {
|
||||||
|
Ringtone tone = RingtoneManager.getRingtone(getActivity(), ringtone);
|
||||||
|
|
||||||
|
if (tone != null) {
|
||||||
|
return tone.getTitle(context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return context.getString(R.string.preferences__default);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void launchSoundSelector(@Nullable Uri current) {
|
||||||
|
Intent intent = new Intent(RingtoneManager.ACTION_RINGTONE_PICKER);
|
||||||
|
|
||||||
|
intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_DEFAULT, true);
|
||||||
|
intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_SILENT, true);
|
||||||
|
intent.putExtra(RingtoneManager.EXTRA_RINGTONE_TYPE, RingtoneManager.TYPE_NOTIFICATION);
|
||||||
|
intent.putExtra(RingtoneManager.EXTRA_RINGTONE_DEFAULT_URI, Settings.System.DEFAULT_NOTIFICATION_URI);
|
||||||
|
intent.putExtra(RingtoneManager.EXTRA_RINGTONE_EXISTING_URI, current);
|
||||||
|
|
||||||
|
startActivityForResult(intent, RINGTONE_PICKER_REQUEST_CODE);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,97 @@
|
|||||||
|
package org.thoughtcrime.securesms.groups.ui.notifications;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.net.Uri;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
import androidx.annotation.WorkerThread;
|
||||||
|
|
||||||
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
||||||
|
import org.thoughtcrime.securesms.database.RecipientDatabase;
|
||||||
|
import org.thoughtcrime.securesms.groups.GroupId;
|
||||||
|
import org.thoughtcrime.securesms.notifications.NotificationChannels;
|
||||||
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
||||||
|
import org.thoughtcrime.securesms.util.TextSecurePreferences;
|
||||||
|
import org.thoughtcrime.securesms.util.concurrent.SignalExecutors;
|
||||||
|
|
||||||
|
class CustomNotificationsRepository {
|
||||||
|
|
||||||
|
private final Context context;
|
||||||
|
private final GroupId groupId;
|
||||||
|
|
||||||
|
CustomNotificationsRepository(@NonNull Context context, @NonNull GroupId groupId) {
|
||||||
|
this.context = context;
|
||||||
|
this.groupId = groupId;
|
||||||
|
}
|
||||||
|
|
||||||
|
void onLoad(@NonNull Runnable onLoaded) {
|
||||||
|
SignalExecutors.SERIAL.execute(() -> {
|
||||||
|
Recipient recipient = getRecipient();
|
||||||
|
RecipientDatabase recipientDatabase = DatabaseFactory.getRecipientDatabase(context);
|
||||||
|
|
||||||
|
recipientDatabase.setMessageRingtone(recipient.getId(), NotificationChannels.getMessageRingtone(context, recipient));
|
||||||
|
recipientDatabase.setMessageVibrate(recipient.getId(), NotificationChannels.getMessageVibrate(context, recipient) ? RecipientDatabase.VibrateState.ENABLED
|
||||||
|
: RecipientDatabase.VibrateState.DISABLED);
|
||||||
|
|
||||||
|
NotificationChannels.ensureCustomChannelConsistency(context);
|
||||||
|
|
||||||
|
onLoaded.run();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void setHasCustomNotifications(final boolean hasCustomNotifications) {
|
||||||
|
SignalExecutors.SERIAL.execute(() -> {
|
||||||
|
if (hasCustomNotifications) {
|
||||||
|
createCustomNotificationChannel();
|
||||||
|
} else {
|
||||||
|
deleteCustomNotificationChannel();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void setMessageVibrate(final RecipientDatabase.VibrateState vibrateState) {
|
||||||
|
SignalExecutors.SERIAL.execute(() -> {
|
||||||
|
Recipient recipient = getRecipient();
|
||||||
|
|
||||||
|
DatabaseFactory.getRecipientDatabase(context).setMessageVibrate(recipient.getId(), vibrateState);
|
||||||
|
NotificationChannels.updateMessageVibrate(context, recipient, vibrateState);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void setMessageSound(@Nullable Uri sound) {
|
||||||
|
SignalExecutors.SERIAL.execute(() -> {
|
||||||
|
Recipient recipient = getRecipient();
|
||||||
|
Uri defaultValue = TextSecurePreferences.getNotificationRingtone(context);
|
||||||
|
Uri newValue;
|
||||||
|
|
||||||
|
if (defaultValue.equals(sound)) newValue = null;
|
||||||
|
else if (sound == null) newValue = Uri.EMPTY;
|
||||||
|
else newValue = sound;
|
||||||
|
|
||||||
|
DatabaseFactory.getRecipientDatabase(context).setMessageRingtone(recipient.getId(), newValue);
|
||||||
|
NotificationChannels.updateMessageRingtone(context, recipient, newValue);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@WorkerThread
|
||||||
|
private void createCustomNotificationChannel() {
|
||||||
|
Recipient recipient = getRecipient();
|
||||||
|
String channelId = NotificationChannels.createChannelFor(context, recipient);
|
||||||
|
|
||||||
|
DatabaseFactory.getRecipientDatabase(context).setNotificationChannel(recipient.getId(), channelId);
|
||||||
|
}
|
||||||
|
|
||||||
|
@WorkerThread
|
||||||
|
private void deleteCustomNotificationChannel() {
|
||||||
|
Recipient recipient = getRecipient();
|
||||||
|
|
||||||
|
DatabaseFactory.getRecipientDatabase(context).setNotificationChannel(recipient.getId(), null);
|
||||||
|
NotificationChannels.deleteChannelFor(context, recipient);
|
||||||
|
}
|
||||||
|
|
||||||
|
@WorkerThread
|
||||||
|
private @NonNull Recipient getRecipient() {
|
||||||
|
return Recipient.externalGroup(context, groupId).resolve();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,81 @@
|
|||||||
|
package org.thoughtcrime.securesms.groups.ui.notifications;
|
||||||
|
|
||||||
|
import android.net.Uri;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
import androidx.lifecycle.LiveData;
|
||||||
|
import androidx.lifecycle.MutableLiveData;
|
||||||
|
import androidx.lifecycle.Transformations;
|
||||||
|
import androidx.lifecycle.ViewModel;
|
||||||
|
import androidx.lifecycle.ViewModelProvider;
|
||||||
|
|
||||||
|
import org.thoughtcrime.securesms.database.RecipientDatabase;
|
||||||
|
import org.thoughtcrime.securesms.groups.GroupId;
|
||||||
|
import org.thoughtcrime.securesms.groups.LiveGroup;
|
||||||
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
||||||
|
|
||||||
|
public final class CustomNotificationsViewModel extends ViewModel {
|
||||||
|
|
||||||
|
private final LiveGroup liveGroup;
|
||||||
|
private final LiveData<Boolean> hasCustomNotifications;
|
||||||
|
private final LiveData<RecipientDatabase.VibrateState> isVibrateEnabled;
|
||||||
|
private final LiveData<Uri> notificationSound;
|
||||||
|
private final CustomNotificationsRepository repository;
|
||||||
|
private final MutableLiveData<Boolean> isInitialLoadComplete = new MutableLiveData<>();
|
||||||
|
|
||||||
|
private CustomNotificationsViewModel(@NonNull GroupId groupId, @NonNull CustomNotificationsRepository repository) {
|
||||||
|
this.liveGroup = new LiveGroup(groupId);
|
||||||
|
this.repository = repository;
|
||||||
|
this.hasCustomNotifications = Transformations.map(liveGroup.getGroupRecipient(), recipient -> recipient.getNotificationChannel() != null);
|
||||||
|
this.isVibrateEnabled = Transformations.map(liveGroup.getGroupRecipient(), Recipient::getMessageVibrate);
|
||||||
|
this.notificationSound = Transformations.map(liveGroup.getGroupRecipient(), Recipient::getMessageRingtone);
|
||||||
|
|
||||||
|
repository.onLoad(() -> isInitialLoadComplete.postValue(true));
|
||||||
|
}
|
||||||
|
|
||||||
|
public LiveData<Boolean> isInitialLoadComplete() {
|
||||||
|
return isInitialLoadComplete;
|
||||||
|
}
|
||||||
|
|
||||||
|
public LiveData<Boolean> hasCustomNotifications() {
|
||||||
|
return hasCustomNotifications;
|
||||||
|
}
|
||||||
|
|
||||||
|
public LiveData<RecipientDatabase.VibrateState> getVibrateState() {
|
||||||
|
return isVibrateEnabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
public LiveData<Uri> getNotificationSound() {
|
||||||
|
return notificationSound;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setHasCustomNotifications(boolean hasCustomNotifications) {
|
||||||
|
repository.setHasCustomNotifications(hasCustomNotifications);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMessageVibrate(@NonNull RecipientDatabase.VibrateState vibrateState) {
|
||||||
|
repository.setMessageVibrate(vibrateState);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMessageSound(@Nullable Uri sound) {
|
||||||
|
repository.setMessageSound(sound);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final class Factory implements ViewModelProvider.Factory {
|
||||||
|
|
||||||
|
private final GroupId groupId;
|
||||||
|
private final CustomNotificationsRepository repository;
|
||||||
|
|
||||||
|
public Factory(@NonNull GroupId groupId, @NonNull CustomNotificationsRepository repository) {
|
||||||
|
this.groupId = groupId;
|
||||||
|
this.repository = repository;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public @NonNull <T extends ViewModel> T create(@NonNull Class<T> modelClass) {
|
||||||
|
//noinspection ConstantConditions
|
||||||
|
return modelClass.cast(new CustomNotificationsViewModel(groupId, repository));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -669,6 +669,10 @@ public class Recipient {
|
|||||||
return System.currentTimeMillis() <= muteUntil;
|
return System.currentTimeMillis() <= muteUntil;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public long getMuteUntil() {
|
||||||
|
return muteUntil;
|
||||||
|
}
|
||||||
|
|
||||||
public boolean isBlocked() {
|
public boolean isBlocked() {
|
||||||
return blocked;
|
return blocked;
|
||||||
}
|
}
|
||||||
|
@ -42,6 +42,10 @@ public class DateUtils extends android.text.format.DateUtils {
|
|||||||
return System.currentTimeMillis() - millis <= unit.toMillis(span);
|
return System.currentTimeMillis() - millis <= unit.toMillis(span);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static boolean isWithinAbs(final long millis, final long span, final TimeUnit unit) {
|
||||||
|
return Math.abs(System.currentTimeMillis() - millis) <= unit.toMillis(span);
|
||||||
|
}
|
||||||
|
|
||||||
private static boolean isYesterday(final long when) {
|
private static boolean isYesterday(final long when) {
|
||||||
return DateUtils.isToday(when + TimeUnit.DAYS.toMillis(1));
|
return DateUtils.isToday(when + TimeUnit.DAYS.toMillis(1));
|
||||||
}
|
}
|
||||||
@ -92,6 +96,20 @@ public class DateUtils extends android.text.format.DateUtils {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static String getTimeString(final Context c, final Locale locale, final long timestamp) {
|
||||||
|
StringBuilder format = new StringBuilder();
|
||||||
|
|
||||||
|
if (isSameDay(System.currentTimeMillis(), timestamp)) format.append("");
|
||||||
|
else if (isWithinAbs(timestamp, 6, TimeUnit.DAYS)) format.append("EEE ");
|
||||||
|
else if (isWithinAbs(timestamp, 364, TimeUnit.DAYS)) format.append("MMM d, ");
|
||||||
|
else format.append("MMM d, yyyy, ");
|
||||||
|
|
||||||
|
if (DateFormat.is24HourFormat(c)) format.append("HH:mm");
|
||||||
|
else format.append("hh:mm a");
|
||||||
|
|
||||||
|
return getFormattedDateTime(timestamp, format.toString(), locale);
|
||||||
|
}
|
||||||
|
|
||||||
public static String getDayPrecisionTimeSpanString(Context context, Locale locale, long timestamp) {
|
public static String getDayPrecisionTimeSpanString(Context context, Locale locale, long timestamp) {
|
||||||
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyyMMdd");
|
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyyMMdd");
|
||||||
|
|
||||||
|
@ -3,6 +3,9 @@ package org.thoughtcrime.securesms.util.livedata;
|
|||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
import androidx.lifecycle.LiveData;
|
import androidx.lifecycle.LiveData;
|
||||||
import androidx.lifecycle.MediatorLiveData;
|
import androidx.lifecycle.MediatorLiveData;
|
||||||
|
import androidx.lifecycle.MutableLiveData;
|
||||||
|
|
||||||
|
import com.annimon.stream.function.Predicate;
|
||||||
|
|
||||||
import org.thoughtcrime.securesms.util.concurrent.SignalExecutors;
|
import org.thoughtcrime.securesms.util.concurrent.SignalExecutors;
|
||||||
import org.whispersystems.libsignal.util.guava.Function;
|
import org.whispersystems.libsignal.util.guava.Function;
|
||||||
@ -14,6 +17,26 @@ public final class LiveDataUtil {
|
|||||||
private LiveDataUtil() {
|
private LiveDataUtil() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static @NonNull <A> LiveData<A> filterNotNull(@NonNull LiveData<A> source) {
|
||||||
|
//noinspection Convert2MethodRef
|
||||||
|
return filter(source, a -> a != null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filters output of a given live data based off a predicate.
|
||||||
|
*/
|
||||||
|
public static @NonNull <A> LiveData<A> filter(@NonNull LiveData<A> source, @NonNull Predicate<A> predicate) {
|
||||||
|
MediatorLiveData<A> mediator = new MediatorLiveData<>();
|
||||||
|
|
||||||
|
mediator.addSource(source, newValue -> {
|
||||||
|
if (predicate.test(newValue)) {
|
||||||
|
mediator.setValue(newValue);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return mediator;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Runs the {@param backgroundFunction} on {@link SignalExecutors#BOUNDED}.
|
* Runs the {@param backgroundFunction} on {@link SignalExecutors#BOUNDED}.
|
||||||
* <p>
|
* <p>
|
||||||
|
@ -0,0 +1,95 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="match_parent">
|
||||||
|
|
||||||
|
<androidx.appcompat.widget.Toolbar
|
||||||
|
android:id="@+id/custom_notifications_toolbar"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="?attr/actionBarSize"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintEnd_toStartOf="@id/custom_notifications_enable_switch"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toTopOf="parent"
|
||||||
|
app:navigationIcon="@drawable/ic_arrow_left_24"
|
||||||
|
app:title="@string/CustomNotificationsDialogFragment__custom_notifications" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/custom_notifications_message_section_header"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginStart="12dp"
|
||||||
|
android:layout_marginTop="16dp"
|
||||||
|
android:text="@string/CustomNotificationsDialogFragment__messages"
|
||||||
|
android:textAppearance="@style/TextAppearance.Signal.Body2"
|
||||||
|
android:textColor="@color/ultramarine_text_button"
|
||||||
|
android:textStyle="bold"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/custom_notifications_toolbar" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/custom_notifications_enable_label"
|
||||||
|
style="@style/Widget.Signal.Button.TextButton"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:text="@string/CustomNotificationsDialogFragment__use_custom_notifications"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/custom_notifications_message_section_header" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/custom_notifications_sound_label"
|
||||||
|
style="@style/Widget.Signal.Button.TextButton"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:enabled="false"
|
||||||
|
android:text="@string/CustomNotificationsDialogFragment__notification_sound"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/custom_notifications_enable_label" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/custom_notifications_vibrate_label"
|
||||||
|
style="@style/Widget.Signal.Button.TextButton"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:enabled="false"
|
||||||
|
android:text="@string/CustomNotificationsDialogFragment__vibrate"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/custom_notifications_sound_label" />
|
||||||
|
|
||||||
|
<Switch
|
||||||
|
android:id="@+id/custom_notifications_enable_switch"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="0dp"
|
||||||
|
android:enabled="false"
|
||||||
|
app:layout_constraintBottom_toBottomOf="@id/custom_notifications_enable_label"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toEndOf="@id/custom_notifications_enable_label"
|
||||||
|
app:layout_constraintTop_toTopOf="@id/custom_notifications_enable_label" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/custom_notifications_sound_selection"
|
||||||
|
style="@style/Widget.Signal.Button.TextButton.Ultramarine"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:visibility="gone"
|
||||||
|
android:gravity="end"
|
||||||
|
app:layout_constraintBottom_toBottomOf="@id/custom_notifications_sound_label"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toEndOf="@id/custom_notifications_sound_label"
|
||||||
|
app:layout_constraintTop_toTopOf="@id/custom_notifications_sound_label"
|
||||||
|
tools:text="Default (Popcorn)" />
|
||||||
|
|
||||||
|
<Switch
|
||||||
|
android:id="@+id/custom_notifications_vibrate_switch"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="0dp"
|
||||||
|
android:visibility="gone"
|
||||||
|
app:layout_constraintBottom_toBottomOf="@id/custom_notifications_vibrate_label"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toEndOf="@id/custom_notifications_vibrate_label"
|
||||||
|
app:layout_constraintTop_toTopOf="@id/custom_notifications_vibrate_label" />
|
||||||
|
|
||||||
|
</androidx.constraintlayout.widget.ConstraintLayout>
|
@ -84,6 +84,88 @@
|
|||||||
|
|
||||||
</androidx.cardview.widget.CardView>
|
</androidx.cardview.widget.CardView>
|
||||||
|
|
||||||
|
<androidx.cardview.widget.CardView
|
||||||
|
android:id="@+id/group_notifications_card"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginTop="16dp"
|
||||||
|
app:cardBackgroundColor="?android:attr/windowBackground"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/group_disappearing_messages_card">
|
||||||
|
|
||||||
|
<androidx.constraintlayout.widget.ConstraintLayout
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content">
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/group_mute_notifications"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:gravity="start|center_vertical"
|
||||||
|
android:paddingStart="12dp"
|
||||||
|
android:paddingEnd="12dp"
|
||||||
|
android:text="@string/ManageGroupActivity_mute_notifications"
|
||||||
|
android:textAppearance="@style/TextAppearance.Signal.Body1"
|
||||||
|
app:layout_constraintBottom_toTopOf="@id/group_mute_notifications_until"
|
||||||
|
app:layout_constraintEnd_toStartOf="@id/group_mute_notifications_switch"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toTopOf="parent"
|
||||||
|
app:layout_goneMarginBottom="8dp" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/group_mute_notifications_until"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:paddingStart="12dp"
|
||||||
|
android:paddingEnd="12dp"
|
||||||
|
android:visibility="gone"
|
||||||
|
app:layout_constraintBottom_toTopOf="@id/group_custom_notifications"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/group_mute_notifications"
|
||||||
|
tools:text="Until 12:42 PM" />
|
||||||
|
|
||||||
|
<Switch
|
||||||
|
android:id="@+id/group_mute_notifications_switch"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="48dp"
|
||||||
|
android:enabled="false"
|
||||||
|
app:layout_constraintBottom_toTopOf="@id/group_custom_notifications_button"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toEndOf="@id/group_mute_notifications"
|
||||||
|
app:layout_constraintTop_toTopOf="parent" />
|
||||||
|
|
||||||
|
<Button
|
||||||
|
android:id="@+id/group_custom_notifications"
|
||||||
|
style="@style/Widget.Signal.Button.TextButton"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:gravity="start|center_vertical"
|
||||||
|
android:text="@string/ManageGroupActivity_custom_notifications"
|
||||||
|
app:layout_constraintBottom_toBottomOf="parent"
|
||||||
|
app:layout_constraintEnd_toStartOf="@id/group_mute_notifications_switch"
|
||||||
|
app:layout_constraintStart_toStartOf="parent" />
|
||||||
|
|
||||||
|
<Button
|
||||||
|
android:id="@+id/group_custom_notifications_button"
|
||||||
|
style="@style/Widget.Signal.Button.TextButton.Ultramarine"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="48dp"
|
||||||
|
android:textAlignment="textEnd"
|
||||||
|
app:layout_constraintBottom_toBottomOf="parent"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toEndOf="@id/group_custom_notifications"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/group_mute_notifications_switch"
|
||||||
|
tools:text="Off" />
|
||||||
|
|
||||||
|
<androidx.constraintlayout.widget.Group
|
||||||
|
android:id="@+id/group_custom_notifications_controls"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="0dp"
|
||||||
|
app:constraint_referenced_ids="group_custom_notifications,group_custom_notifications_button"/>
|
||||||
|
|
||||||
|
</androidx.constraintlayout.widget.ConstraintLayout>
|
||||||
|
|
||||||
|
</androidx.cardview.widget.CardView>
|
||||||
|
|
||||||
<androidx.cardview.widget.CardView
|
<androidx.cardview.widget.CardView
|
||||||
android:id="@+id/group_media_card"
|
android:id="@+id/group_media_card"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
@ -91,7 +173,7 @@
|
|||||||
android:layout_marginTop="16dp"
|
android:layout_marginTop="16dp"
|
||||||
android:visibility="gone"
|
android:visibility="gone"
|
||||||
app:cardBackgroundColor="?android:attr/windowBackground"
|
app:cardBackgroundColor="?android:attr/windowBackground"
|
||||||
app:layout_constraintTop_toBottomOf="@id/group_disappearing_messages_card"
|
app:layout_constraintTop_toBottomOf="@id/group_notifications_card"
|
||||||
tools:visibility="visible">
|
tools:visibility="visible">
|
||||||
|
|
||||||
<LinearLayout
|
<LinearLayout
|
||||||
|
@ -494,6 +494,11 @@
|
|||||||
<string name="ManageGroupActivity_who_can_edit_group_info">Who can edit group info</string>
|
<string name="ManageGroupActivity_who_can_edit_group_info">Who can edit group info</string>
|
||||||
<string name="ManageGroupActivity_block_group">Block group</string>
|
<string name="ManageGroupActivity_block_group">Block group</string>
|
||||||
<string name="ManageGroupActivity_leave_group">Leave group</string>
|
<string name="ManageGroupActivity_leave_group">Leave group</string>
|
||||||
|
<string name="ManageGroupActivity_mute_notifications">Mute notifications</string>
|
||||||
|
<string name="ManageGroupActivity_custom_notifications">Custom notifications</string>
|
||||||
|
<string name="ManageGroupActivity_until_s">Until %1$s</string>
|
||||||
|
<string name="ManageGroupActivity_off">Off</string>
|
||||||
|
<string name="ManageGroupActivity_on">On</string>
|
||||||
|
|
||||||
<string name="ManageGroupActivity_you_dont_have_the_rights_to_do_this">You don\'t have the rights to do this</string>
|
<string name="ManageGroupActivity_you_dont_have_the_rights_to_do_this">You don\'t have the rights to do this</string>
|
||||||
<string name="ManageGroupActivity_failed_to_update_the_group">Failed to update the group</string>
|
<string name="ManageGroupActivity_failed_to_update_the_group">Failed to update the group</string>
|
||||||
@ -508,6 +513,13 @@
|
|||||||
<item quantity="other">%1$s invited %2$d people</item>
|
<item quantity="other">%1$s invited %2$d people</item>
|
||||||
</plurals>
|
</plurals>
|
||||||
|
|
||||||
|
<!-- CustomNotificationsDialogFragment -->
|
||||||
|
<string name="CustomNotificationsDialogFragment__custom_notifications">Custom notifications</string>
|
||||||
|
<string name="CustomNotificationsDialogFragment__messages">Messages</string>
|
||||||
|
<string name="CustomNotificationsDialogFragment__use_custom_notifications">Use custom notifications</string>
|
||||||
|
<string name="CustomNotificationsDialogFragment__notification_sound">Notification sound</string>
|
||||||
|
<string name="CustomNotificationsDialogFragment__vibrate">Vibrate</string>
|
||||||
|
|
||||||
<!-- GV2 Invite cancellation confirmation -->
|
<!-- GV2 Invite cancellation confirmation -->
|
||||||
<string name="GroupManagement_cancel_own_single_invite">Do you want to cancel the invite you sent to %1$s?</string>
|
<string name="GroupManagement_cancel_own_single_invite">Do you want to cancel the invite you sent to %1$s?</string>
|
||||||
<plurals name="GroupManagement_cancel_others_invites">
|
<plurals name="GroupManagement_cancel_others_invites">
|
||||||
|
Loading…
x
Reference in New Issue
Block a user