forked from mirror/monocles_chat
add option to restrict avatar access model to contacts
This commit is contained in:
parent
4b10ecf079
commit
4497149950
5 changed files with 185 additions and 127 deletions
src/main
java/eu/siacs/conversations
res
|
@ -4577,23 +4577,32 @@ public class XmppConnectionService extends Service {
|
||||||
}).start();
|
}).start();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void publishAvatar(final Account account, final Uri image, final OnAvatarPublication callback) {
|
public void publishAvatarAsync(
|
||||||
new Thread(() -> {
|
final Account account,
|
||||||
final Bitmap.CompressFormat format = Config.AVATAR_FORMAT;
|
final Uri image,
|
||||||
final int size = Config.AVATAR_SIZE;
|
final boolean open,
|
||||||
final Avatar avatar = getFileBackend().getPepAvatar(image, size, format);
|
final OnAvatarPublication callback) {
|
||||||
if (avatar != null) {
|
new Thread(() -> publishAvatar(account, image, open, callback)).start();
|
||||||
if (!getFileBackend().save(avatar)) {
|
}
|
||||||
Log.d(Config.LOGTAG, "unable to save vcard");
|
|
||||||
callback.onAvatarPublicationFailed(R.string.error_saving_avatar);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
publishAvatar(account, avatar, callback);
|
|
||||||
} else {
|
|
||||||
callback.onAvatarPublicationFailed(R.string.error_publish_avatar_converting);
|
|
||||||
}
|
|
||||||
}).start();
|
|
||||||
|
|
||||||
|
private void publishAvatar(
|
||||||
|
final Account account,
|
||||||
|
final Uri image,
|
||||||
|
final boolean open,
|
||||||
|
final OnAvatarPublication callback) {
|
||||||
|
final Bitmap.CompressFormat format = Config.AVATAR_FORMAT;
|
||||||
|
final int size = Config.AVATAR_SIZE;
|
||||||
|
final Avatar avatar = getFileBackend().getPepAvatar(image, size, format);
|
||||||
|
if (avatar != null) {
|
||||||
|
if (!getFileBackend().save(avatar)) {
|
||||||
|
Log.d(Config.LOGTAG, "unable to save vcard");
|
||||||
|
callback.onAvatarPublicationFailed(R.string.error_saving_avatar);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
publishAvatar(account, avatar, open, callback);
|
||||||
|
} else {
|
||||||
|
callback.onAvatarPublicationFailed(R.string.error_publish_avatar_converting);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void publishMucAvatar(Conversation conversation, Avatar avatar, OnAvatarPublication callback) {
|
private void publishMucAvatar(Conversation conversation, Avatar avatar, OnAvatarPublication callback) {
|
||||||
|
@ -4631,10 +4640,14 @@ public class XmppConnectionService extends Service {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public void publishAvatar(Account account, final Avatar avatar, final OnAvatarPublication callback) {
|
public void publishAvatar(
|
||||||
|
final Account account,
|
||||||
|
final Avatar avatar,
|
||||||
|
final boolean open,
|
||||||
|
final OnAvatarPublication callback) {
|
||||||
final Bundle options;
|
final Bundle options;
|
||||||
if (account.getXmppConnection().getFeatures().pepPublishOptions()) {
|
if (account.getXmppConnection().getFeatures().pepPublishOptions()) {
|
||||||
options = PublishOptions.openAccess();
|
options = open ? PublishOptions.openAccess() : PublishOptions.presenceAccess();
|
||||||
} else {
|
} else {
|
||||||
options = null;
|
options = null;
|
||||||
}
|
}
|
||||||
|
@ -4706,7 +4719,7 @@ public class XmppConnectionService extends Service {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public void republishAvatarIfNeeded(Account account) {
|
public void republishAvatarIfNeeded(final Account account) {
|
||||||
if (account.getAxolotlService().isPepBroken()) {
|
if (account.getAxolotlService().isPepBroken()) {
|
||||||
Log.d(Config.LOGTAG, account.getJid().asBareJid() + ": skipping republication of avatar because pep is broken");
|
Log.d(Config.LOGTAG, account.getJid().asBareJid() + ": skipping republication of avatar because pep is broken");
|
||||||
return;
|
return;
|
||||||
|
@ -4735,12 +4748,19 @@ public class XmppConnectionService extends Service {
|
||||||
@Override
|
@Override
|
||||||
public void accept(final Iq packet) {
|
public void accept(final Iq packet) {
|
||||||
if (packet.getType() == Iq.Type.RESULT || errorIsItemNotFound(packet)) {
|
if (packet.getType() == Iq.Type.RESULT || errorIsItemNotFound(packet)) {
|
||||||
Avatar serverAvatar = parseAvatar(packet);
|
final Avatar serverAvatar = parseAvatar(packet);
|
||||||
if (serverAvatar == null && account.getAvatar() != null) {
|
if (serverAvatar == null && account.getAvatar() != null) {
|
||||||
Avatar avatar = fileBackend.getStoredPepAvatar(account.getAvatar());
|
final Avatar avatar =
|
||||||
|
fileBackend.getStoredPepAvatar(account.getAvatar());
|
||||||
if (avatar != null) {
|
if (avatar != null) {
|
||||||
Log.d(Config.LOGTAG, account.getJid().asBareJid() + ": avatar on server was null. republishing");
|
Log.d(Config.LOGTAG, account.getJid().asBareJid() + ": avatar on server was null. republishing");
|
||||||
publishAvatar(account, fileBackend.getStoredPepAvatar(account.getAvatar()), null);
|
// publishing as 'open' - old server (that requires
|
||||||
|
// republication) likely doesn't support access models anyway
|
||||||
|
publishAvatar(
|
||||||
|
account,
|
||||||
|
fileBackend.getStoredPepAvatar(account.getAvatar()),
|
||||||
|
true,
|
||||||
|
null);
|
||||||
} else {
|
} else {
|
||||||
Log.e(Config.LOGTAG, account.getJid().asBareJid() + ": error rereading avatar");
|
Log.e(Config.LOGTAG, account.getJid().asBareJid() + ": error rereading avatar");
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,19 +14,12 @@ import android.view.Menu;
|
||||||
import android.view.MenuItem;
|
import android.view.MenuItem;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.view.View.OnLongClickListener;
|
import android.view.View.OnLongClickListener;
|
||||||
import android.widget.Button;
|
|
||||||
import android.widget.ImageView;
|
|
||||||
import android.widget.TextView;
|
|
||||||
import android.widget.Toast;
|
import android.widget.Toast;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
import androidx.annotation.StringRes;
|
import androidx.annotation.StringRes;
|
||||||
import androidx.databinding.DataBindingUtil;
|
import androidx.databinding.DataBindingUtil;
|
||||||
|
|
||||||
import com.canhub.cropper.CropImage;
|
import com.canhub.cropper.CropImage;
|
||||||
|
import com.google.android.material.dialog.MaterialAlertDialogBuilder;
|
||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
|
||||||
|
|
||||||
import eu.siacs.conversations.Config;
|
import eu.siacs.conversations.Config;
|
||||||
import eu.siacs.conversations.R;
|
import eu.siacs.conversations.R;
|
||||||
import eu.siacs.conversations.databinding.ActivityPublishProfilePictureBinding;
|
import eu.siacs.conversations.databinding.ActivityPublishProfilePictureBinding;
|
||||||
|
@ -35,89 +28,97 @@ import eu.siacs.conversations.persistance.FileBackend;
|
||||||
import eu.siacs.conversations.services.XmppConnectionService;
|
import eu.siacs.conversations.services.XmppConnectionService;
|
||||||
import eu.siacs.conversations.ui.interfaces.OnAvatarPublication;
|
import eu.siacs.conversations.ui.interfaces.OnAvatarPublication;
|
||||||
import eu.siacs.conversations.utils.PhoneHelper;
|
import eu.siacs.conversations.utils.PhoneHelper;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
public class PublishProfilePictureActivity extends XmppActivity implements XmppConnectionService.OnAccountUpdate, OnAvatarPublication {
|
public class PublishProfilePictureActivity extends XmppActivity
|
||||||
|
implements XmppConnectionService.OnAccountUpdate, OnAvatarPublication {
|
||||||
public static final int REQUEST_CHOOSE_PICTURE = 0x1337;
|
public static final int REQUEST_CHOOSE_PICTURE = 0x1337;
|
||||||
|
|
||||||
private ImageView avatar;
|
private ActivityPublishProfilePictureBinding binding;
|
||||||
private TextView hintOrWarning;
|
|
||||||
private TextView secondaryHint;
|
|
||||||
private Button cancelButton;
|
|
||||||
private Button publishButton;
|
|
||||||
private Uri avatarUri;
|
private Uri avatarUri;
|
||||||
private Uri defaultUri;
|
private Uri defaultUri;
|
||||||
private Account account;
|
private Account account;
|
||||||
private boolean support = false;
|
private boolean support = false;
|
||||||
private boolean publishing = false;
|
private boolean publishing = false;
|
||||||
private final AtomicBoolean handledExternalUri = new AtomicBoolean(false);
|
private final AtomicBoolean handledExternalUri = new AtomicBoolean(false);
|
||||||
private final OnLongClickListener backToDefaultListener = new OnLongClickListener() {
|
private final OnLongClickListener backToDefaultListener =
|
||||||
|
new OnLongClickListener() {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onLongClick(View v) {
|
public boolean onLongClick(View v) {
|
||||||
avatarUri = defaultUri;
|
avatarUri = defaultUri;
|
||||||
loadImageIntoPreview(defaultUri);
|
loadImageIntoPreview(defaultUri);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
private boolean mInitialAccountSetup;
|
private boolean mInitialAccountSetup;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onAvatarPublicationSucceeded() {
|
public void onAvatarPublicationSucceeded() {
|
||||||
runOnUiThread(() -> {
|
runOnUiThread(
|
||||||
if (mInitialAccountSetup) {
|
() -> {
|
||||||
Intent intent = new Intent(getApplicationContext(), StartConversationActivity.class);
|
if (mInitialAccountSetup) {
|
||||||
StartConversationActivity.addInviteUri(intent, getIntent());
|
Intent intent =
|
||||||
intent.putExtra("init", true);
|
new Intent(
|
||||||
intent.putExtra(EXTRA_ACCOUNT, account.getJid().asBareJid().toEscapedString());
|
getApplicationContext(), StartConversationActivity.class);
|
||||||
startActivity(intent);
|
StartConversationActivity.addInviteUri(intent, getIntent());
|
||||||
}
|
intent.putExtra("init", true);
|
||||||
Toast.makeText(PublishProfilePictureActivity.this,
|
intent.putExtra(
|
||||||
R.string.avatar_has_been_published,
|
EXTRA_ACCOUNT, account.getJid().asBareJid().toEscapedString());
|
||||||
Toast.LENGTH_SHORT).show();
|
startActivity(intent);
|
||||||
finish();
|
}
|
||||||
});
|
Toast.makeText(
|
||||||
|
PublishProfilePictureActivity.this,
|
||||||
|
R.string.avatar_has_been_published,
|
||||||
|
Toast.LENGTH_SHORT)
|
||||||
|
.show();
|
||||||
|
finish();
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onAvatarPublicationFailed(int res) {
|
public void onAvatarPublicationFailed(final int res) {
|
||||||
runOnUiThread(() -> {
|
runOnUiThread(
|
||||||
hintOrWarning.setText(res);
|
() -> {
|
||||||
hintOrWarning.setVisibility(View.VISIBLE);
|
this.binding.hintOrWarning.setText(res);
|
||||||
publishing = false;
|
this.binding.hintOrWarning.setVisibility(View.VISIBLE);
|
||||||
togglePublishButton(true, R.string.publish);
|
this.publishing = false;
|
||||||
});
|
togglePublishButton(true, R.string.publish);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onCreate(Bundle savedInstanceState) {
|
public void onCreate(final Bundle savedInstanceState) {
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
|
|
||||||
ActivityPublishProfilePictureBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_publish_profile_picture);
|
this.binding =
|
||||||
|
DataBindingUtil.setContentView(this, R.layout.activity_publish_profile_picture);
|
||||||
|
|
||||||
setSupportActionBar(binding.toolbar);
|
setSupportActionBar(binding.toolbar);
|
||||||
|
|
||||||
Activities.setStatusAndNavigationBarColors(this, binding.getRoot());
|
Activities.setStatusAndNavigationBarColors(this, binding.getRoot());
|
||||||
|
|
||||||
this.avatar = findViewById(R.id.account_image);
|
this.binding.publishButton.setOnClickListener(
|
||||||
this.cancelButton = findViewById(R.id.cancel_button);
|
v -> {
|
||||||
this.publishButton = findViewById(R.id.publish_button);
|
final boolean open = !this.binding.contactOnly.isChecked();
|
||||||
this.hintOrWarning = findViewById(R.id.hint_or_warning);
|
final var uri = this.avatarUri;
|
||||||
this.secondaryHint = findViewById(R.id.secondary_hint);
|
if (uri == null) {
|
||||||
this.publishButton.setOnClickListener(v -> {
|
return;
|
||||||
if (avatarUri != null) {
|
}
|
||||||
publishing = true;
|
publishing = true;
|
||||||
togglePublishButton(false, R.string.publishing);
|
togglePublishButton(false, R.string.publishing);
|
||||||
xmppConnectionService.publishAvatar(account, avatarUri, this);
|
xmppConnectionService.publishAvatarAsync(account, uri, open, this);
|
||||||
}
|
});
|
||||||
});
|
this.binding.cancelButton.setOnClickListener(
|
||||||
this.cancelButton.setOnClickListener(
|
|
||||||
v -> {
|
v -> {
|
||||||
if (mInitialAccountSetup) {
|
if (mInitialAccountSetup) {
|
||||||
final Intent intent =
|
final Intent intent =
|
||||||
new Intent(
|
new Intent(
|
||||||
getApplicationContext(), StartConversationActivity.class);
|
getApplicationContext(), StartConversationActivity.class);
|
||||||
intent.putExtra("init", true);
|
if (xmppConnectionService != null
|
||||||
|
&& xmppConnectionService.getAccounts().size() == 1) {
|
||||||
|
intent.putExtra("init", true);
|
||||||
|
}
|
||||||
StartConversationActivity.addInviteUri(intent, getIntent());
|
StartConversationActivity.addInviteUri(intent, getIntent());
|
||||||
if (account != null) {
|
if (account != null) {
|
||||||
intent.putExtra(
|
intent.putExtra(
|
||||||
|
@ -127,11 +128,12 @@ public class PublishProfilePictureActivity extends XmppActivity implements XmppC
|
||||||
}
|
}
|
||||||
finish();
|
finish();
|
||||||
});
|
});
|
||||||
this.avatar.setOnClickListener(v -> chooseAvatar(this));
|
this.binding.accountImage.setOnClickListener(v -> chooseAvatar(this));
|
||||||
this.defaultUri = PhoneHelper.getProfilePictureUri(getApplicationContext());
|
this.defaultUri = PhoneHelper.getProfilePictureUri(getApplicationContext());
|
||||||
if (savedInstanceState != null) {
|
if (savedInstanceState != null) {
|
||||||
this.avatarUri = savedInstanceState.getParcelable("uri");
|
this.avatarUri = savedInstanceState.getParcelable("uri");
|
||||||
this.handledExternalUri.set(savedInstanceState.getBoolean("handle_external_uri",false));
|
this.handledExternalUri.set(
|
||||||
|
savedInstanceState.getBoolean("handle_external_uri", false));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -144,8 +146,8 @@ public class PublishProfilePictureActivity extends XmppActivity implements XmppC
|
||||||
@Override
|
@Override
|
||||||
public boolean onOptionsItemSelected(final MenuItem item) {
|
public boolean onOptionsItemSelected(final MenuItem item) {
|
||||||
if (item.getItemId() == R.id.action_delete_avatar) {
|
if (item.getItemId() == R.id.action_delete_avatar) {
|
||||||
if (xmppConnectionService != null && account != null) {
|
if (account != null) {
|
||||||
xmppConnectionService.deleteAvatar(account);
|
deleteAvatar(account);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
|
@ -153,6 +155,22 @@ public class PublishProfilePictureActivity extends XmppActivity implements XmppC
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void deleteAvatar(final Account account) {
|
||||||
|
new MaterialAlertDialogBuilder(this)
|
||||||
|
.setTitle(R.string.delete_avatar)
|
||||||
|
.setMessage(R.string.delete_avatar_message)
|
||||||
|
.setNegativeButton(R.string.cancel, null)
|
||||||
|
.setPositiveButton(
|
||||||
|
R.string.confirm,
|
||||||
|
(d, v) -> {
|
||||||
|
if (xmppConnectionService != null) {
|
||||||
|
xmppConnectionService.deleteAvatar(account);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.create()
|
||||||
|
.show();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onSaveInstanceState(@NonNull Bundle outState) {
|
public void onSaveInstanceState(@NonNull Bundle outState) {
|
||||||
if (this.avatarUri != null) {
|
if (this.avatarUri != null) {
|
||||||
|
@ -191,9 +209,9 @@ public class PublishProfilePictureActivity extends XmppActivity implements XmppC
|
||||||
final Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
|
final Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
|
||||||
intent.setType("image/*");
|
intent.setType("image/*");
|
||||||
activity.startActivityForResult(
|
activity.startActivityForResult(
|
||||||
Intent.createChooser(intent, activity.getString(R.string.attach_choose_picture)),
|
Intent.createChooser(
|
||||||
REQUEST_CHOOSE_PICTURE
|
intent, activity.getString(R.string.attach_choose_picture)),
|
||||||
);
|
REQUEST_CHOOSE_PICTURE);
|
||||||
} else {
|
} else {
|
||||||
CropImage.activity()
|
CropImage.activity()
|
||||||
.setOutputCompressFormat(Bitmap.CompressFormat.PNG)
|
.setOutputCompressFormat(Bitmap.CompressFormat.PNG)
|
||||||
|
@ -212,7 +230,9 @@ public class PublishProfilePictureActivity extends XmppActivity implements XmppC
|
||||||
}
|
}
|
||||||
|
|
||||||
private void reloadAvatar() {
|
private void reloadAvatar() {
|
||||||
this.support = this.account.getXmppConnection() != null && this.account.getXmppConnection().getFeatures().pep();
|
this.support =
|
||||||
|
this.account.getXmppConnection() != null
|
||||||
|
&& this.account.getXmppConnection().getFeatures().pep();
|
||||||
if (this.avatarUri == null) {
|
if (this.avatarUri == null) {
|
||||||
if (this.account.getAvatar() != null || this.defaultUri == null) {
|
if (this.account.getAvatar() != null || this.defaultUri == null) {
|
||||||
loadImageIntoPreview(null);
|
loadImageIntoPreview(null);
|
||||||
|
@ -233,21 +253,22 @@ public class PublishProfilePictureActivity extends XmppActivity implements XmppC
|
||||||
|
|
||||||
final Uri uri = intent != null ? intent.getData() : null;
|
final Uri uri = intent != null ? intent.getData() : null;
|
||||||
|
|
||||||
if (uri != null && handledExternalUri.compareAndSet(false,true)) {
|
if (uri != null && handledExternalUri.compareAndSet(false, true)) {
|
||||||
cropUri(uri);
|
cropUri(uri);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.mInitialAccountSetup) {
|
if (this.mInitialAccountSetup) {
|
||||||
this.cancelButton.setText(R.string.skip);
|
this.binding.cancelButton.setText(R.string.skip);
|
||||||
}
|
}
|
||||||
configureActionBar(getSupportActionBar(), !this.mInitialAccountSetup && !handledExternalUri.get());
|
configureActionBar(
|
||||||
|
getSupportActionBar(), !this.mInitialAccountSetup && !handledExternalUri.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
public void cropUri(final Uri uri) {
|
public void cropUri(final Uri uri) {
|
||||||
if (Build.VERSION.SDK_INT >= 28) {
|
if (Build.VERSION.SDK_INT >= 28) {
|
||||||
loadImageIntoPreview(uri);
|
loadImageIntoPreview(uri);
|
||||||
if (this.avatar.getDrawable() instanceof AnimatedImageDrawable || this.avatar.getDrawable() instanceof FileBackend.SVGDrawable) {
|
if (this.binding.accountImage.getDrawable() instanceof AnimatedImageDrawable || this.binding.accountImage.getDrawable() instanceof FileBackend.SVGDrawable) {
|
||||||
this.avatarUri = uri;
|
this.avatarUri = uri;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -259,7 +280,7 @@ public class PublishProfilePictureActivity extends XmppActivity implements XmppC
|
||||||
.start(this);
|
.start(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void loadImageIntoPreview(Uri uri) {
|
protected void loadImageIntoPreview(final Uri uri) {
|
||||||
|
|
||||||
Drawable bm = null;
|
Drawable bm = null;
|
||||||
if (uri == null) {
|
if (uri == null) {
|
||||||
|
@ -267,46 +288,46 @@ public class PublishProfilePictureActivity extends XmppActivity implements XmppC
|
||||||
} else {
|
} else {
|
||||||
try {
|
try {
|
||||||
bm = xmppConnectionService.getFileBackend().cropCenterSquareDrawable(uri, (int) getResources().getDimension(R.dimen.publish_avatar_size));
|
bm = xmppConnectionService.getFileBackend().cropCenterSquareDrawable(uri, (int) getResources().getDimension(R.dimen.publish_avatar_size));
|
||||||
} catch (Exception e) {
|
} catch (final Exception e) {
|
||||||
Log.d(Config.LOGTAG, "unable to load bitmap into image view", e);
|
Log.d(Config.LOGTAG, "unable to load bitmap into image view", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bm == null) {
|
if (bm == null) {
|
||||||
togglePublishButton(false, R.string.publish);
|
togglePublishButton(false, R.string.publish);
|
||||||
this.hintOrWarning.setVisibility(View.VISIBLE);
|
this.binding.hintOrWarning.setVisibility(View.VISIBLE);
|
||||||
this.hintOrWarning.setText(R.string.error_publish_avatar_converting);
|
this.binding.hintOrWarning.setText(R.string.error_publish_avatar_converting);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this.avatar.setImageDrawable(bm);
|
this.binding.accountImage.setImageDrawable(bm);
|
||||||
if (Build.VERSION.SDK_INT >= 28 && bm instanceof AnimatedImageDrawable) {
|
if (Build.VERSION.SDK_INT >= 28 && bm instanceof AnimatedImageDrawable) {
|
||||||
((AnimatedImageDrawable) bm).start();
|
((AnimatedImageDrawable) bm).start();
|
||||||
}
|
}
|
||||||
if (support) {
|
if (support) {
|
||||||
togglePublishButton(uri != null, R.string.publish);
|
togglePublishButton(uri != null, R.string.publish);
|
||||||
this.hintOrWarning.setVisibility(View.INVISIBLE);
|
this.binding.hintOrWarning.setVisibility(View.INVISIBLE);
|
||||||
} else {
|
} else {
|
||||||
togglePublishButton(false, R.string.publish);
|
togglePublishButton(false, R.string.publish);
|
||||||
this.hintOrWarning.setVisibility(View.VISIBLE);
|
this.binding.hintOrWarning.setVisibility(View.VISIBLE);
|
||||||
if (account.getStatus() == Account.State.ONLINE) {
|
if (account.getStatus() == Account.State.ONLINE) {
|
||||||
this.hintOrWarning.setText(R.string.error_publish_avatar_no_server_support);
|
this.binding.hintOrWarning.setText(R.string.error_publish_avatar_no_server_support);
|
||||||
} else {
|
} else {
|
||||||
this.hintOrWarning.setText(R.string.error_publish_avatar_offline);
|
this.binding.hintOrWarning.setText(R.string.error_publish_avatar_offline);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (this.defaultUri == null || this.defaultUri.equals(uri)) {
|
if (this.defaultUri == null || this.defaultUri.equals(uri)) {
|
||||||
this.secondaryHint.setVisibility(View.INVISIBLE);
|
this.binding.secondaryHint.setVisibility(View.INVISIBLE);
|
||||||
this.avatar.setOnLongClickListener(null);
|
this.binding.accountImage.setOnLongClickListener(null);
|
||||||
} else if (this.defaultUri != null) {
|
} else if (this.defaultUri != null) {
|
||||||
this.secondaryHint.setVisibility(View.VISIBLE);
|
this.binding.secondaryHint.setVisibility(View.VISIBLE);
|
||||||
this.avatar.setOnLongClickListener(this.backToDefaultListener);
|
this.binding.accountImage.setOnLongClickListener(this.backToDefaultListener);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void togglePublishButton(boolean enabled, @StringRes int res) {
|
protected void togglePublishButton(boolean enabled, @StringRes int res) {
|
||||||
final boolean status = enabled && !publishing;
|
final boolean status = enabled && !publishing;
|
||||||
this.publishButton.setText(publishing ? R.string.publishing : res);
|
this.binding.publishButton.setText(publishing ? R.string.publishing : res);
|
||||||
this.publishButton.setEnabled(status);
|
this.binding.publishButton.setEnabled(status);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void refreshUiReal() {
|
public void refreshUiReal() {
|
||||||
|
|
|
@ -1,16 +1,13 @@
|
||||||
package eu.siacs.conversations.xmpp.pep;
|
package eu.siacs.conversations.xmpp.pep;
|
||||||
|
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
|
|
||||||
import eu.siacs.conversations.xml.Element;
|
import eu.siacs.conversations.xml.Element;
|
||||||
import eu.siacs.conversations.xml.Namespace;
|
import eu.siacs.conversations.xml.Namespace;
|
||||||
import im.conversations.android.xmpp.model.stanza.Iq;
|
import im.conversations.android.xmpp.model.stanza.Iq;
|
||||||
|
|
||||||
public class PublishOptions {
|
public class PublishOptions {
|
||||||
|
|
||||||
private PublishOptions() {
|
private PublishOptions() {}
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public static Bundle openAccess() {
|
public static Bundle openAccess() {
|
||||||
final Bundle options = new Bundle();
|
final Bundle options = new Bundle();
|
||||||
|
@ -18,6 +15,12 @@ public class PublishOptions {
|
||||||
return options;
|
return options;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static Bundle presenceAccess() {
|
||||||
|
final Bundle options = new Bundle();
|
||||||
|
options.putString("pubsub#access_model", "presence");
|
||||||
|
return options;
|
||||||
|
}
|
||||||
|
|
||||||
public static Bundle persistentWhitelistAccess() {
|
public static Bundle persistentWhitelistAccess() {
|
||||||
final Bundle options = new Bundle();
|
final Bundle options = new Bundle();
|
||||||
options.putString("pubsub#persist_items", "true");
|
options.putString("pubsub#persist_items", "true");
|
||||||
|
@ -32,14 +35,15 @@ public class PublishOptions {
|
||||||
options.putString("pubsub#send_last_published_item", "never");
|
options.putString("pubsub#send_last_published_item", "never");
|
||||||
options.putString("pubsub#max_items", "max");
|
options.putString("pubsub#max_items", "max");
|
||||||
options.putString("pubsub#notify_delete", "true");
|
options.putString("pubsub#notify_delete", "true");
|
||||||
options.putString("pubsub#notify_retract", "true"); //one could also set notify=true on the retract
|
options.putString(
|
||||||
|
"pubsub#notify_retract", "true"); // one could also set notify=true on the retract
|
||||||
|
|
||||||
return options;
|
return options;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean preconditionNotMet(Iq response) {
|
public static boolean preconditionNotMet(Iq response) {
|
||||||
final Element error = response.getType() == Iq.Type.ERROR ? response.findChild("error") : null;
|
final Element error =
|
||||||
|
response.getType() == Iq.Type.ERROR ? response.findChild("error") : null;
|
||||||
return error != null && error.hasChild("precondition-not-met", Namespace.PUBSUB_ERROR);
|
return error != null && error.hasChild("precondition-not-met", Namespace.PUBSUB_ERROR);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<layout xmlns:android="http://schemas.android.com/apk/res/android">
|
<layout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:tools="http://schemas.android.com/tools">
|
||||||
|
|
||||||
<RelativeLayout
|
<RelativeLayout
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
|
@ -19,18 +20,19 @@
|
||||||
</com.google.android.material.appbar.AppBarLayout>
|
</com.google.android.material.appbar.AppBarLayout>
|
||||||
|
|
||||||
|
|
||||||
<LinearLayout
|
<ScrollView
|
||||||
android:layout_width="fill_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_below="@id/app_bar_layout"
|
android:layout_above="@+id/button_bar"
|
||||||
android:layout_marginLeft="@dimen/activity_horizontal_margin"
|
android:layout_below="@id/app_bar_layout">
|
||||||
android:layout_marginTop="@dimen/activity_vertical_margin"
|
|
||||||
android:layout_marginRight="@dimen/activity_horizontal_margin"
|
|
||||||
android:layout_marginBottom="@dimen/activity_vertical_margin">
|
|
||||||
|
|
||||||
<LinearLayout
|
<LinearLayout
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginLeft="@dimen/activity_horizontal_margin"
|
||||||
|
android:layout_marginTop="@dimen/activity_vertical_margin"
|
||||||
|
android:layout_marginRight="@dimen/activity_horizontal_margin"
|
||||||
|
android:layout_marginBottom="@dimen/activity_vertical_margin"
|
||||||
android:gravity="center_horizontal"
|
android:gravity="center_horizontal"
|
||||||
android:orientation="vertical">
|
android:orientation="vertical">
|
||||||
|
|
||||||
|
@ -62,17 +64,26 @@
|
||||||
android:text="@string/or_long_press_for_default"
|
android:text="@string/or_long_press_for_default"
|
||||||
android:textAppearance="?textAppearanceBodyMedium" />
|
android:textAppearance="?textAppearanceBodyMedium" />
|
||||||
|
|
||||||
|
<com.google.android.material.materialswitch.MaterialSwitch
|
||||||
|
android:id="@+id/contact_only"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_marginVertical="12dp"
|
||||||
|
android:text="@string/show_to_contacts_only" />
|
||||||
|
|
||||||
<TextView
|
<TextView
|
||||||
android:id="@+id/hint_or_warning"
|
android:id="@+id/hint_or_warning"
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_marginTop="8dp"
|
android:layout_marginVertical="12dp"
|
||||||
android:layout_marginBottom="8dp"
|
|
||||||
android:textAppearance="?textAppearanceBodyMedium"
|
android:textAppearance="?textAppearanceBodyMedium"
|
||||||
android:textColor="?colorError" />
|
android:textColor="?colorError"
|
||||||
|
tools:text="@string/error_saving_avatar" />
|
||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
</LinearLayout>
|
|
||||||
|
|
||||||
|
</ScrollView>
|
||||||
|
|
||||||
<RelativeLayout
|
<RelativeLayout
|
||||||
android:id="@+id/button_bar"
|
android:id="@+id/button_bar"
|
||||||
|
|
|
@ -1408,4 +1408,6 @@
|
||||||
<string name="pref_prefer_ipv6">Prefer IPv6</string>
|
<string name="pref_prefer_ipv6">Prefer IPv6</string>
|
||||||
<string name="pref_use_colored_muc_names">Use colored user names</string>
|
<string name="pref_use_colored_muc_names">Use colored user names</string>
|
||||||
<string name="pref_use_colored_muc_names_summary">Use colored user names in group chats</string>
|
<string name="pref_use_colored_muc_names_summary">Use colored user names in group chats</string>
|
||||||
|
<string name="delete_avatar_message">Would you like to delete your avatar? Some clients might continue to display a cached copy of your avatar.</string>
|
||||||
|
<string name="show_to_contacts_only">Show to contacts only</string>
|
||||||
</resources>
|
</resources>
|
Loading…
Add table
Reference in a new issue