mirror of
https://github.com/oxen-io/session-android.git
synced 2024-11-28 12:35:17 +00:00
d05097a6fd
Closes #4480 // FREEBIE
546 lines
21 KiB
Java
546 lines
21 KiB
Java
/**
|
|
* Copyright (C) 2014 Open Whisper Systems
|
|
*
|
|
* 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.
|
|
*
|
|
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
package org.thoughtcrime.securesms;
|
|
|
|
import android.app.Activity;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.graphics.Bitmap;
|
|
import android.net.Uri;
|
|
import android.os.AsyncTask;
|
|
import android.os.Bundle;
|
|
import android.support.annotation.NonNull;
|
|
import android.support.annotation.Nullable;
|
|
import android.util.Log;
|
|
import android.view.Menu;
|
|
import android.view.MenuInflater;
|
|
import android.view.MenuItem;
|
|
import android.view.View;
|
|
import android.widget.EditText;
|
|
import android.widget.ImageView;
|
|
import android.widget.ListView;
|
|
import android.widget.TextView;
|
|
import android.widget.Toast;
|
|
|
|
import com.bumptech.glide.Glide;
|
|
import com.bumptech.glide.request.animation.GlideAnimation;
|
|
import com.bumptech.glide.request.target.SimpleTarget;
|
|
import com.soundcloud.android.crop.Crop;
|
|
|
|
import org.thoughtcrime.securesms.components.PushRecipientsPanel;
|
|
import org.thoughtcrime.securesms.components.PushRecipientsPanel.RecipientsPanelChangedListener;
|
|
import org.thoughtcrime.securesms.contacts.RecipientsEditor;
|
|
import org.thoughtcrime.securesms.crypto.MasterSecret;
|
|
import org.thoughtcrime.securesms.database.DatabaseFactory;
|
|
import org.thoughtcrime.securesms.database.GroupDatabase;
|
|
import org.thoughtcrime.securesms.database.GroupDatabase.GroupRecord;
|
|
import org.thoughtcrime.securesms.database.NotInDirectoryException;
|
|
import org.thoughtcrime.securesms.database.TextSecureDirectory;
|
|
import org.thoughtcrime.securesms.database.ThreadDatabase;
|
|
import org.thoughtcrime.securesms.groups.GroupManager;
|
|
import org.thoughtcrime.securesms.groups.GroupManager.GroupActionResult;
|
|
import org.thoughtcrime.securesms.mms.RoundedCorners;
|
|
import org.thoughtcrime.securesms.recipients.Recipient;
|
|
import org.thoughtcrime.securesms.recipients.RecipientFactory;
|
|
import org.thoughtcrime.securesms.recipients.Recipients;
|
|
import org.thoughtcrime.securesms.util.BitmapUtil;
|
|
import org.thoughtcrime.securesms.util.DynamicLanguage;
|
|
import org.thoughtcrime.securesms.util.DynamicTheme;
|
|
import org.thoughtcrime.securesms.util.GroupUtil;
|
|
import org.thoughtcrime.securesms.util.ProgressDialogAsyncTask;
|
|
import org.thoughtcrime.securesms.util.SelectedRecipientsAdapter;
|
|
import org.thoughtcrime.securesms.util.SelectedRecipientsAdapter.OnRecipientDeletedListener;
|
|
import org.thoughtcrime.securesms.util.TextSecurePreferences;
|
|
import org.thoughtcrime.securesms.util.Util;
|
|
import org.thoughtcrime.securesms.util.ViewUtil;
|
|
import org.whispersystems.libaxolotl.util.guava.Optional;
|
|
import org.whispersystems.textsecure.api.util.InvalidNumberException;
|
|
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.util.Collection;
|
|
import java.util.HashSet;
|
|
import java.util.List;
|
|
import java.util.Set;
|
|
|
|
/**
|
|
* Activity to create and update groups
|
|
*
|
|
* @author Jake McGinty
|
|
*/
|
|
public class GroupCreateActivity extends PassphraseRequiredActionBarActivity
|
|
implements OnRecipientDeletedListener,
|
|
RecipientsPanelChangedListener
|
|
{
|
|
|
|
private final static String TAG = GroupCreateActivity.class.getSimpleName();
|
|
|
|
public static final String GROUP_RECIPIENT_EXTRA = "group_recipient";
|
|
public static final String GROUP_THREAD_EXTRA = "group_thread";
|
|
|
|
private final DynamicTheme dynamicTheme = new DynamicTheme();
|
|
private final DynamicLanguage dynamicLanguage = new DynamicLanguage();
|
|
|
|
private static final int PICK_CONTACT = 1;
|
|
public static final int AVATAR_SIZE = 210;
|
|
|
|
private EditText groupName;
|
|
private ListView lv;
|
|
private ImageView avatar;
|
|
private TextView creatingText;
|
|
private MasterSecret masterSecret;
|
|
private Bitmap avatarBmp;
|
|
|
|
@NonNull private Optional<GroupData> groupToUpdate = Optional.absent();
|
|
|
|
@Override
|
|
protected void onPreCreate() {
|
|
dynamicTheme.onCreate(this);
|
|
dynamicLanguage.onCreate(this);
|
|
}
|
|
|
|
@Override
|
|
protected void onCreate(Bundle state, @NonNull MasterSecret masterSecret) {
|
|
this.masterSecret = masterSecret;
|
|
|
|
setContentView(R.layout.group_create_activity);
|
|
//noinspection ConstantConditions
|
|
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
|
|
initializeResources();
|
|
initializeExistingGroup();
|
|
}
|
|
|
|
@Override
|
|
public void onResume() {
|
|
super.onResume();
|
|
dynamicTheme.onResume(this);
|
|
dynamicLanguage.onResume(this);
|
|
updateViewState();
|
|
}
|
|
|
|
private boolean isSignalGroup() {
|
|
return TextSecurePreferences.isPushRegistered(this) && !getAdapter().hasNonPushMembers();
|
|
}
|
|
|
|
private void disableSignalGroupViews(int reasonResId) {
|
|
View pushDisabled = findViewById(R.id.push_disabled);
|
|
pushDisabled.setVisibility(View.VISIBLE);
|
|
((TextView) findViewById(R.id.push_disabled_reason)).setText(reasonResId);
|
|
avatar.setEnabled(false);
|
|
groupName.setEnabled(false);
|
|
}
|
|
|
|
private void enableSignalGroupViews() {
|
|
findViewById(R.id.push_disabled).setVisibility(View.GONE);
|
|
avatar.setEnabled(true);
|
|
groupName.setEnabled(true);
|
|
}
|
|
|
|
@SuppressWarnings("ConstantConditions")
|
|
private void updateViewState() {
|
|
if (!TextSecurePreferences.isPushRegistered(this)) {
|
|
disableSignalGroupViews(R.string.GroupCreateActivity_you_dont_support_push);
|
|
getSupportActionBar().setTitle(R.string.GroupCreateActivity_actionbar_mms_title);
|
|
} else if (getAdapter().hasNonPushMembers()) {
|
|
disableSignalGroupViews(R.string.GroupCreateActivity_contacts_dont_support_push);
|
|
getSupportActionBar().setTitle(R.string.GroupCreateActivity_actionbar_mms_title);
|
|
} else {
|
|
enableSignalGroupViews();
|
|
getSupportActionBar().setTitle(R.string.GroupCreateActivity_actionbar_title);
|
|
}
|
|
}
|
|
|
|
private static boolean isActiveInDirectory(Context context, Recipient recipient) {
|
|
try {
|
|
return TextSecureDirectory.getInstance(context)
|
|
.isSecureTextSupported(Util.canonicalizeNumber(context, recipient.getNumber()));
|
|
} catch (NotInDirectoryException | InvalidNumberException e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
private void addSelectedContact(@NonNull Recipient contact) {
|
|
final boolean isPushUser = isActiveInDirectory(this, contact);
|
|
if (groupToUpdate.isPresent() && !isPushUser) {
|
|
Toast.makeText(this, R.string.GroupCreateActivity_cannot_add_non_push_to_existing_group, Toast.LENGTH_LONG).show();
|
|
return;
|
|
}
|
|
|
|
getAdapter().add(contact, isPushUser);
|
|
updateViewState();
|
|
}
|
|
|
|
private void addAllSelectedContacts(Collection<Recipient> contacts) {
|
|
for (Recipient contact : contacts) {
|
|
addSelectedContact(contact);
|
|
}
|
|
}
|
|
|
|
private void initializeResources() {
|
|
RecipientsEditor recipientsEditor = ViewUtil.findById(this, R.id.recipients_text);
|
|
PushRecipientsPanel recipientsPanel = ViewUtil.findById(this, R.id.recipients);
|
|
lv = ViewUtil.findById(this, R.id.selected_contacts_list);
|
|
avatar = ViewUtil.findById(this, R.id.avatar);
|
|
groupName = ViewUtil.findById(this, R.id.group_name);
|
|
creatingText = ViewUtil.findById(this, R.id.creating_group_text);
|
|
SelectedRecipientsAdapter adapter = new SelectedRecipientsAdapter(this);
|
|
adapter.setOnRecipientDeletedListener(this);
|
|
lv.setAdapter(adapter);
|
|
recipientsEditor.setHint(R.string.recipients_panel__add_member);
|
|
recipientsPanel.setPanelChangeListener(this);
|
|
findViewById(R.id.contacts_button).setOnClickListener(new AddRecipientButtonListener());
|
|
avatar.setOnClickListener(new View.OnClickListener() {
|
|
@Override
|
|
public void onClick(View view) {
|
|
Crop.pickImage(GroupCreateActivity.this);
|
|
}
|
|
});
|
|
}
|
|
|
|
private void initializeExistingGroup() {
|
|
final String encodedGroupId = RecipientFactory.getRecipientForId(this, getIntent().getLongExtra(GROUP_RECIPIENT_EXTRA, -1), true)
|
|
.getNumber();
|
|
byte[] groupId;
|
|
try {
|
|
groupId = GroupUtil.getDecodedId(encodedGroupId);
|
|
} catch (IOException ioe) {
|
|
Log.w(TAG, "Couldn't decode the encoded groupId passed in via intent", ioe);
|
|
groupId = null;
|
|
}
|
|
if (groupId != null) {
|
|
new FillExistingGroupInfoAsyncTask(this).execute(groupId);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean onPrepareOptionsMenu(Menu menu) {
|
|
MenuInflater inflater = this.getMenuInflater();
|
|
menu.clear();
|
|
|
|
inflater.inflate(R.menu.group_create, menu);
|
|
super.onPrepareOptionsMenu(menu);
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public boolean onOptionsItemSelected(MenuItem item) {
|
|
super.onOptionsItemSelected(item);
|
|
switch (item.getItemId()) {
|
|
case android.R.id.home:
|
|
finish();
|
|
return true;
|
|
case R.id.menu_create_group:
|
|
if (groupToUpdate.isPresent()) handleGroupUpdate();
|
|
else handleGroupCreate();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public void onRecipientDeleted(Recipient recipient) {
|
|
getAdapter().remove(recipient);
|
|
updateViewState();
|
|
}
|
|
|
|
@Override
|
|
public void onRecipientsPanelUpdate(Recipients recipients) {
|
|
if (recipients != null) addAllSelectedContacts(recipients.getRecipientsList());
|
|
}
|
|
|
|
private void handleGroupCreate() {
|
|
if (getAdapter().getCount() < 1) {
|
|
Log.i(TAG, getString(R.string.GroupCreateActivity_contacts_no_members));
|
|
Toast.makeText(getApplicationContext(), R.string.GroupCreateActivity_contacts_no_members, Toast.LENGTH_SHORT).show();
|
|
return;
|
|
}
|
|
if (isSignalGroup()) {
|
|
new CreateSignalGroupTask(this, masterSecret, avatarBmp, getGroupName(), getAdapter().getRecipients()).execute();
|
|
} else {
|
|
new CreateMmsGroupTask(this, getAdapter().getRecipients()).execute();
|
|
}
|
|
}
|
|
|
|
private void handleGroupUpdate() {
|
|
new UpdateSignalGroupTask(this, masterSecret, groupToUpdate.get().id, avatarBmp,
|
|
getGroupName(), getAdapter().getRecipients()).execute();
|
|
}
|
|
|
|
private void handleOpenConversation(long threadId, Recipients recipients) {
|
|
Intent intent = new Intent(this, ConversationActivity.class);
|
|
intent.putExtra(ConversationActivity.THREAD_ID_EXTRA, threadId);
|
|
intent.putExtra(ConversationActivity.DISTRIBUTION_TYPE_EXTRA, ThreadDatabase.DistributionTypes.DEFAULT);
|
|
intent.putExtra(ConversationActivity.RECIPIENTS_EXTRA, recipients.getIds());
|
|
startActivity(intent);
|
|
finish();
|
|
}
|
|
|
|
private SelectedRecipientsAdapter getAdapter() {
|
|
return (SelectedRecipientsAdapter)lv.getAdapter();
|
|
}
|
|
|
|
private @Nullable String getGroupName() {
|
|
return groupName.getText() != null ? groupName.getText().toString() : null;
|
|
}
|
|
|
|
@Override
|
|
public void onActivityResult(int reqCode, int resultCode, final Intent data) {
|
|
super.onActivityResult(reqCode, resultCode, data);
|
|
Uri outputFile = Uri.fromFile(new File(getCacheDir(), "cropped"));
|
|
|
|
if (data == null || resultCode != Activity.RESULT_OK)
|
|
return;
|
|
|
|
switch (reqCode) {
|
|
case PICK_CONTACT:
|
|
List<String> selected = data.getStringArrayListExtra("contacts");
|
|
for (String contact : selected) {
|
|
final Recipient recipient = RecipientFactory.getRecipientsFromString(this, contact, false).getPrimaryRecipient();
|
|
if (recipient != null) addSelectedContact(recipient);
|
|
}
|
|
break;
|
|
|
|
case Crop.REQUEST_PICK:
|
|
new Crop(data.getData()).output(outputFile).asSquare().start(this);
|
|
break;
|
|
case Crop.REQUEST_CROP:
|
|
Glide.with(this).load(Crop.getOutput(data)).asBitmap().skipMemoryCache(true)
|
|
.centerCrop().override(AVATAR_SIZE, AVATAR_SIZE)
|
|
.into(new SimpleTarget<Bitmap>() {
|
|
@Override
|
|
public void onResourceReady(Bitmap resource,
|
|
GlideAnimation<? super Bitmap> glideAnimation)
|
|
{
|
|
avatarBmp = resource;
|
|
Glide.with(GroupCreateActivity.this).load(Crop.getOutput(data)).skipMemoryCache(true)
|
|
.transform(new RoundedCorners(GroupCreateActivity.this, avatar.getWidth() / 2))
|
|
.into(avatar);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
private class AddRecipientButtonListener implements View.OnClickListener {
|
|
@Override
|
|
public void onClick(View v) {
|
|
Intent intent = new Intent(GroupCreateActivity.this, PushContactSelectionActivity.class);
|
|
if (groupToUpdate.isPresent()) intent.putExtra(ContactSelectionListFragment.DISPLAY_MODE,
|
|
ContactSelectionListFragment.DISPLAY_MODE_PUSH_ONLY);
|
|
startActivityForResult(intent, PICK_CONTACT);
|
|
}
|
|
}
|
|
|
|
private static class CreateMmsGroupTask extends AsyncTask<Void,Void,Long> {
|
|
private GroupCreateActivity activity;
|
|
private Set<Recipient> members;
|
|
|
|
public CreateMmsGroupTask(GroupCreateActivity activity, Set<Recipient> members) {
|
|
this.activity = activity;
|
|
this.members = members;
|
|
}
|
|
|
|
@Override
|
|
protected Long doInBackground(Void... avoid) {
|
|
Recipients recipients = RecipientFactory.getRecipientsFor(activity, members, false);
|
|
return DatabaseFactory.getThreadDatabase(activity)
|
|
.getThreadIdFor(recipients, ThreadDatabase.DistributionTypes.CONVERSATION);
|
|
}
|
|
|
|
@Override
|
|
protected void onPostExecute(Long resultThread) {
|
|
if (resultThread > -1) {
|
|
activity.handleOpenConversation(resultThread,
|
|
RecipientFactory.getRecipientsFor(activity, members, true));
|
|
} else {
|
|
Toast.makeText(activity, R.string.GroupCreateActivity_contacts_mms_exception, Toast.LENGTH_LONG).show();
|
|
activity.finish();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onProgressUpdate(Void... values) {
|
|
super.onProgressUpdate(values);
|
|
}
|
|
}
|
|
|
|
private abstract static class SignalGroupTask extends AsyncTask<Void,Void,Optional<GroupActionResult>> {
|
|
protected GroupCreateActivity activity;
|
|
protected MasterSecret masterSecret;
|
|
protected Bitmap avatar;
|
|
protected Set<Recipient> members;
|
|
protected String name;
|
|
|
|
public SignalGroupTask(GroupCreateActivity activity,
|
|
MasterSecret masterSecret,
|
|
Bitmap avatar,
|
|
String name,
|
|
Set<Recipient> members)
|
|
{
|
|
this.activity = activity;
|
|
this.masterSecret = masterSecret;
|
|
this.avatar = avatar;
|
|
this.name = name;
|
|
this.members = members;
|
|
}
|
|
|
|
@Override
|
|
protected void onPreExecute() {
|
|
activity.findViewById(R.id.group_details_layout).setVisibility(View.GONE);
|
|
activity.findViewById(R.id.creating_group_layout).setVisibility(View.VISIBLE);
|
|
activity.findViewById(R.id.menu_create_group).setVisibility(View.GONE);
|
|
final int titleResId = activity.groupToUpdate.isPresent()
|
|
? R.string.GroupCreateActivity_updating_group
|
|
: R.string.GroupCreateActivity_creating_group;
|
|
activity.creatingText.setText(activity.getString(titleResId, activity.getGroupName()));
|
|
}
|
|
|
|
@Override
|
|
protected void onPostExecute(Optional<GroupActionResult> groupActionResultOptional) {
|
|
if (activity.isFinishing()) return;
|
|
activity.findViewById(R.id.group_details_layout).setVisibility(View.VISIBLE);
|
|
activity.findViewById(R.id.creating_group_layout).setVisibility(View.GONE);
|
|
activity.findViewById(R.id.menu_create_group).setVisibility(View.VISIBLE);
|
|
}
|
|
}
|
|
|
|
private static class CreateSignalGroupTask extends SignalGroupTask {
|
|
public CreateSignalGroupTask(GroupCreateActivity activity, MasterSecret masterSecret, Bitmap avatar, String name, Set<Recipient> members) {
|
|
super(activity, masterSecret, avatar, name, members);
|
|
}
|
|
|
|
@Override
|
|
protected Optional<GroupActionResult> doInBackground(Void... aVoid) {
|
|
try {
|
|
return Optional.of(GroupManager.createGroup(activity, masterSecret, members, avatar, name));
|
|
} catch (InvalidNumberException e) {
|
|
return Optional.absent();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onPostExecute(Optional<GroupActionResult> result) {
|
|
if (result.isPresent() && result.get().getThreadId() > -1) {
|
|
if (!activity.isFinishing()) {
|
|
activity.handleOpenConversation(result.get().getThreadId(), result.get().getGroupRecipient());
|
|
}
|
|
} else {
|
|
super.onPostExecute(result);
|
|
Toast.makeText(activity.getApplicationContext(),
|
|
R.string.GroupCreateActivity_contacts_invalid_number, Toast.LENGTH_LONG).show();
|
|
}
|
|
}
|
|
}
|
|
|
|
private static class UpdateSignalGroupTask extends SignalGroupTask {
|
|
private byte[] groupId;
|
|
|
|
public UpdateSignalGroupTask(GroupCreateActivity activity,
|
|
MasterSecret masterSecret, byte[] groupId, Bitmap avatar, String name,
|
|
Set<Recipient> members)
|
|
{
|
|
super(activity, masterSecret, avatar, name, members);
|
|
this.groupId = groupId;
|
|
}
|
|
|
|
@Override
|
|
protected Optional<GroupActionResult> doInBackground(Void... aVoid) {
|
|
try {
|
|
return Optional.of(GroupManager.updateGroup(activity, masterSecret, groupId, members, avatar, name));
|
|
} catch (InvalidNumberException e) {
|
|
return Optional.absent();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onPostExecute(Optional<GroupActionResult> result) {
|
|
if (result.isPresent() && result.get().getThreadId() > -1) {
|
|
if (!activity.isFinishing()) {
|
|
Intent intent = activity.getIntent();
|
|
intent.putExtra(GROUP_THREAD_EXTRA, result.get().getThreadId());
|
|
intent.putExtra(GROUP_RECIPIENT_EXTRA, result.get().getGroupRecipient().getIds());
|
|
activity.setResult(RESULT_OK, intent);
|
|
activity.finish();
|
|
}
|
|
} else {
|
|
super.onPostExecute(result);
|
|
Toast.makeText(activity.getApplicationContext(),
|
|
R.string.GroupCreateActivity_contacts_invalid_number, Toast.LENGTH_LONG).show();
|
|
}
|
|
}
|
|
}
|
|
|
|
private static class FillExistingGroupInfoAsyncTask extends ProgressDialogAsyncTask<byte[],Void,Optional<GroupData>> {
|
|
private GroupCreateActivity activity;
|
|
|
|
public FillExistingGroupInfoAsyncTask(GroupCreateActivity activity) {
|
|
super(activity,
|
|
R.string.GroupCreateActivity_loading_group_details,
|
|
R.string.please_wait);
|
|
this.activity = activity;
|
|
}
|
|
|
|
@Override
|
|
protected Optional<GroupData> doInBackground(byte[]... groupIds) {
|
|
final GroupDatabase db = DatabaseFactory.getGroupDatabase(activity);
|
|
final Recipients recipients = db.getGroupMembers(groupIds[0], false);
|
|
final GroupRecord group = db.getGroup(groupIds[0]);
|
|
final Set<Recipient> existingContacts = new HashSet<>(recipients.getRecipientsList().size());
|
|
existingContacts.addAll(recipients.getRecipientsList());
|
|
|
|
if (group != null) {
|
|
return Optional.of(new GroupData(groupIds[0],
|
|
existingContacts,
|
|
BitmapUtil.fromByteArray(group.getAvatar()),
|
|
group.getTitle()));
|
|
} else {
|
|
return Optional.absent();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onPostExecute(Optional<GroupData> group) {
|
|
super.onPostExecute(group);
|
|
|
|
if (group.isPresent() && !activity.isFinishing()) {
|
|
activity.groupToUpdate = group;
|
|
|
|
activity.groupName.setText(group.get().name);
|
|
if (group.get().avatar != null) activity.avatar.setImageBitmap(group.get().avatar);
|
|
SelectedRecipientsAdapter adapter = new SelectedRecipientsAdapter(activity, group.get().recipients);
|
|
adapter.setOnRecipientDeletedListener(activity);
|
|
activity.lv.setAdapter(adapter);
|
|
}
|
|
}
|
|
}
|
|
|
|
private static class GroupData {
|
|
byte[] id;
|
|
Set<Recipient> recipients;
|
|
Bitmap avatar;
|
|
String name;
|
|
|
|
public GroupData(byte[] id, Set<Recipient> recipients, Bitmap avatar, String name) {
|
|
this.id = id;
|
|
this.recipients = recipients;
|
|
this.avatar = avatar;
|
|
this.name = name;
|
|
}
|
|
}
|
|
}
|