mirror of https://github.com/M66B/FairEmail.git
Progress bar for attachment downloads
This commit is contained in:
parent
f4970a0aa5
commit
b460c06d1b
|
@ -35,6 +35,7 @@ import android.view.LayoutInflater;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.view.ViewGroup;
|
import android.view.ViewGroup;
|
||||||
import android.widget.ImageView;
|
import android.widget.ImageView;
|
||||||
|
import android.widget.ProgressBar;
|
||||||
import android.widget.TextView;
|
import android.widget.TextView;
|
||||||
import android.widget.Toast;
|
import android.widget.Toast;
|
||||||
|
|
||||||
|
@ -51,15 +52,15 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
private Context context;
|
private Context context;
|
||||||
private ExecutorService executor = Executors.newCachedThreadPool();
|
private ExecutorService executor = Executors.newCachedThreadPool();
|
||||||
|
|
||||||
private List<EntityAttachment> all = new ArrayList<>();
|
private List<TupleAttachment> all = new ArrayList<>();
|
||||||
private List<EntityAttachment> filtered = new ArrayList<>();
|
private List<TupleAttachment> filtered = new ArrayList<>();
|
||||||
|
|
||||||
public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
|
public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
|
||||||
View itemView;
|
View itemView;
|
||||||
TextView tvName;
|
TextView tvName;
|
||||||
TextView tvSize;
|
TextView tvSize;
|
||||||
TextView tvProgress;
|
|
||||||
ImageView ivStatus;
|
ImageView ivStatus;
|
||||||
|
ProgressBar progressbar;
|
||||||
|
|
||||||
ViewHolder(View itemView) {
|
ViewHolder(View itemView) {
|
||||||
super(itemView);
|
super(itemView);
|
||||||
|
@ -67,8 +68,8 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
this.itemView = itemView;
|
this.itemView = itemView;
|
||||||
tvName = itemView.findViewById(R.id.tvName);
|
tvName = itemView.findViewById(R.id.tvName);
|
||||||
tvSize = itemView.findViewById(R.id.tvSize);
|
tvSize = itemView.findViewById(R.id.tvSize);
|
||||||
tvProgress = itemView.findViewById(R.id.tvProgress);
|
|
||||||
ivStatus = itemView.findViewById(R.id.ivStatus);
|
ivStatus = itemView.findViewById(R.id.ivStatus);
|
||||||
|
progressbar = itemView.findViewById(R.id.progressbar);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void wire() {
|
private void wire() {
|
||||||
|
@ -81,26 +82,14 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onClick(View view) {
|
public void onClick(View view) {
|
||||||
final EntityAttachment attachment = filtered.get(getLayoutPosition());
|
final TupleAttachment attachment = filtered.get(getLayoutPosition());
|
||||||
if (attachment != null)
|
if (attachment != null)
|
||||||
if (attachment.content == null) {
|
if (attachment.content) {
|
||||||
if (attachment.progress == null)
|
|
||||||
// Download
|
|
||||||
executor.submit(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
DB db = DB.getInstance(context);
|
|
||||||
attachment.progress = 0;
|
|
||||||
db.attachment().updateAttachment(attachment);
|
|
||||||
EntityMessage message = db.message().getMessage(attachment.message);
|
|
||||||
EntityOperation.queue(context, message, EntityOperation.ATTACHMENT, attachment.sequence);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
// Build file name
|
// Build file name
|
||||||
final File dir = new File(context.getCacheDir(), "attachments");
|
final File dir = new File(context.getCacheDir(), "attachments");
|
||||||
final File file = new File(dir, TextUtils.isEmpty(attachment.name)
|
final File file = new File(dir, TextUtils.isEmpty(attachment.name)
|
||||||
? "attachment_" + attachment.id : attachment.name);
|
? "attachment_" + attachment.id
|
||||||
|
: attachment.name.toLowerCase().replaceAll("[^a-zA-Z0-9-.]", "_"));
|
||||||
|
|
||||||
// https://developer.android.com/reference/android/support/v4/content/FileProvider
|
// https://developer.android.com/reference/android/support/v4/content/FileProvider
|
||||||
Uri uri = FileProvider.getUriForFile(context, "eu.faircode.email", file);
|
Uri uri = FileProvider.getUriForFile(context, "eu.faircode.email", file);
|
||||||
|
@ -133,10 +122,14 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
dir.mkdir();
|
dir.mkdir();
|
||||||
file.createNewFile();
|
file.createNewFile();
|
||||||
|
|
||||||
|
// Get attachment content
|
||||||
|
byte[] content = DB.getInstance(context).attachment().getContent(attachment.id);
|
||||||
|
|
||||||
|
// Write attachment content to file
|
||||||
FileOutputStream fos = null;
|
FileOutputStream fos = null;
|
||||||
try {
|
try {
|
||||||
fos = new FileOutputStream(file);
|
fos = new FileOutputStream(file);
|
||||||
fos.write(attachment.content);
|
fos.write(content);
|
||||||
} finally {
|
} finally {
|
||||||
if (fos != null)
|
if (fos != null)
|
||||||
fos.close();
|
fos.close();
|
||||||
|
@ -150,6 +143,18 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
} else {
|
||||||
|
if (attachment.progress == null)
|
||||||
|
// Download
|
||||||
|
executor.submit(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
DB db = DB.getInstance(context);
|
||||||
|
db.attachment().setProgress(attachment.id, 0);
|
||||||
|
EntityMessage message = db.message().getMessage(attachment.message);
|
||||||
|
EntityOperation.queue(context, message, EntityOperation.ATTACHMENT, attachment.sequence);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -159,12 +164,12 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
setHasStableIds(true);
|
setHasStableIds(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void set(List<EntityAttachment> attachments) {
|
public void set(List<TupleAttachment> attachments) {
|
||||||
Log.i(Helper.TAG, "Set attachments=" + attachments.size());
|
Log.i(Helper.TAG, "Set attachments=" + attachments.size());
|
||||||
|
|
||||||
Collections.sort(attachments, new Comparator<EntityAttachment>() {
|
Collections.sort(attachments, new Comparator<TupleAttachment>() {
|
||||||
@Override
|
@Override
|
||||||
public int compare(EntityAttachment a1, EntityAttachment a2) {
|
public int compare(TupleAttachment a1, TupleAttachment a2) {
|
||||||
return a1.sequence.compareTo(a2.sequence);
|
return a1.sequence.compareTo(a2.sequence);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -202,10 +207,10 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
}
|
}
|
||||||
|
|
||||||
private class MessageDiffCallback extends DiffUtil.Callback {
|
private class MessageDiffCallback extends DiffUtil.Callback {
|
||||||
private List<EntityAttachment> prev;
|
private List<TupleAttachment> prev;
|
||||||
private List<EntityAttachment> next;
|
private List<TupleAttachment> next;
|
||||||
|
|
||||||
MessageDiffCallback(List<EntityAttachment> prev, List<EntityAttachment> next) {
|
MessageDiffCallback(List<TupleAttachment> prev, List<TupleAttachment> next) {
|
||||||
this.prev = prev;
|
this.prev = prev;
|
||||||
this.next = next;
|
this.next = next;
|
||||||
}
|
}
|
||||||
|
@ -222,15 +227,15 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
|
public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
|
||||||
EntityAttachment a1 = prev.get(oldItemPosition);
|
TupleAttachment a1 = prev.get(oldItemPosition);
|
||||||
EntityAttachment a2 = next.get(newItemPosition);
|
TupleAttachment a2 = next.get(newItemPosition);
|
||||||
return a1.id.equals(a2.id);
|
return a1.id.equals(a2.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) {
|
public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) {
|
||||||
EntityAttachment a1 = prev.get(oldItemPosition);
|
TupleAttachment a1 = prev.get(oldItemPosition);
|
||||||
EntityAttachment a2 = next.get(newItemPosition);
|
TupleAttachment a2 = next.get(newItemPosition);
|
||||||
return a1.equals(a2);
|
return a1.equals(a2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -255,7 +260,7 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
public void onBindViewHolder(@NonNull ViewHolder holder, int position) {
|
public void onBindViewHolder(@NonNull ViewHolder holder, int position) {
|
||||||
holder.unwire();
|
holder.unwire();
|
||||||
|
|
||||||
EntityAttachment attachment = filtered.get(position);
|
TupleAttachment attachment = filtered.get(position);
|
||||||
holder.tvName.setText(attachment.name);
|
holder.tvName.setText(attachment.name);
|
||||||
|
|
||||||
if (attachment.size != null)
|
if (attachment.size != null)
|
||||||
|
@ -263,19 +268,19 @@ public class AdapterAttachment extends RecyclerView.Adapter<AdapterAttachment.Vi
|
||||||
holder.tvSize.setVisibility(attachment.size == null ? View.GONE : View.VISIBLE);
|
holder.tvSize.setVisibility(attachment.size == null ? View.GONE : View.VISIBLE);
|
||||||
|
|
||||||
if (attachment.progress != null)
|
if (attachment.progress != null)
|
||||||
holder.tvProgress.setText(String.format("%d %%", attachment.progress));
|
holder.progressbar.setProgress(attachment.progress);
|
||||||
holder.tvProgress.setVisibility(
|
holder.progressbar.setVisibility(
|
||||||
attachment.progress == null || attachment.content != null ? View.GONE : View.VISIBLE);
|
attachment.progress == null || attachment.content ? View.GONE : View.VISIBLE);
|
||||||
|
|
||||||
if (attachment.content == null) {
|
if (attachment.content) {
|
||||||
|
holder.ivStatus.setImageResource(R.drawable.baseline_visibility_24);
|
||||||
|
holder.ivStatus.setVisibility(View.VISIBLE);
|
||||||
|
} else {
|
||||||
if (attachment.progress == null) {
|
if (attachment.progress == null) {
|
||||||
holder.ivStatus.setImageResource(R.drawable.baseline_get_app_24);
|
holder.ivStatus.setImageResource(R.drawable.baseline_get_app_24);
|
||||||
holder.ivStatus.setVisibility(View.VISIBLE);
|
holder.ivStatus.setVisibility(View.VISIBLE);
|
||||||
} else
|
} else
|
||||||
holder.ivStatus.setVisibility(View.GONE);
|
holder.ivStatus.setVisibility(View.GONE);
|
||||||
} else {
|
|
||||||
holder.ivStatus.setImageResource(R.drawable.baseline_visibility_24);
|
|
||||||
holder.ivStatus.setVisibility(View.VISIBLE);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
holder.wire();
|
holder.wire();
|
||||||
|
|
|
@ -30,12 +30,20 @@ import java.util.List;
|
||||||
|
|
||||||
@Dao
|
@Dao
|
||||||
public interface DaoAttachment {
|
public interface DaoAttachment {
|
||||||
@Query("SELECT * FROM attachment WHERE message = :message")
|
@Query("SELECT id,message,sequence,name,type,size,progress" +
|
||||||
LiveData<List<EntityAttachment>> liveAttachments(long message);
|
", (NOT content IS NULL) as content" +
|
||||||
|
" FROM attachment WHERE message = :message")
|
||||||
|
LiveData<List<TupleAttachment>> liveAttachments(long message);
|
||||||
|
|
||||||
@Query("SELECT * FROM attachment WHERE message = :message AND sequence = :sequence")
|
@Query("SELECT * FROM attachment WHERE message = :message AND sequence = :sequence")
|
||||||
EntityAttachment getAttachment(long message, int sequence);
|
EntityAttachment getAttachment(long message, int sequence);
|
||||||
|
|
||||||
|
@Query("UPDATE attachment SET progress = :progress WHERE id = :id")
|
||||||
|
void setProgress(long id, int progress);
|
||||||
|
|
||||||
|
@Query("SELECT content FROM attachment WHERE id = :id")
|
||||||
|
byte[] getContent(long id);
|
||||||
|
|
||||||
@Insert(onConflict = OnConflictStrategy.REPLACE)
|
@Insert(onConflict = OnConflictStrategy.REPLACE)
|
||||||
long insertAttachment(EntityAttachment attachment);
|
long insertAttachment(EntityAttachment attachment);
|
||||||
|
|
||||||
|
|
|
@ -30,6 +30,9 @@ import java.util.List;
|
||||||
|
|
||||||
@Dao
|
@Dao
|
||||||
public interface DaoFolder {
|
public interface DaoFolder {
|
||||||
|
@Query("SELECT * FROM folder WHERE account = :account")
|
||||||
|
List<EntityFolder> getFolders(long account);
|
||||||
|
|
||||||
@Query("SELECT * FROM folder WHERE account = :account AND synchronize = :synchronize")
|
@Query("SELECT * FROM folder WHERE account = :account AND synchronize = :synchronize")
|
||||||
List<EntityFolder> getFolders(long account, boolean synchronize);
|
List<EntityFolder> getFolders(long account, boolean synchronize);
|
||||||
|
|
||||||
|
|
|
@ -237,9 +237,9 @@ public class FragmentMessage extends Fragment {
|
||||||
|
|
||||||
DB.getInstance(getContext()).attachment().liveAttachments(id).removeObservers(FragmentMessage.this);
|
DB.getInstance(getContext()).attachment().liveAttachments(id).removeObservers(FragmentMessage.this);
|
||||||
DB.getInstance(getContext()).attachment().liveAttachments(id).observe(FragmentMessage.this,
|
DB.getInstance(getContext()).attachment().liveAttachments(id).observe(FragmentMessage.this,
|
||||||
new Observer<List<EntityAttachment>>() {
|
new Observer<List<TupleAttachment>>() {
|
||||||
@Override
|
@Override
|
||||||
public void onChanged(@Nullable List<EntityAttachment> attachments) {
|
public void onChanged(@Nullable List<TupleAttachment> attachments) {
|
||||||
adapter.set(attachments);
|
adapter.set(attachments);
|
||||||
grpAttachments.setVisibility(attachments.size() > 0 ? View.VISIBLE : View.GONE);
|
grpAttachments.setVisibility(attachments.size() > 0 ? View.VISIBLE : View.GONE);
|
||||||
}
|
}
|
||||||
|
|
|
@ -310,7 +310,6 @@ public class MessageHelper {
|
||||||
if (Part.ATTACHMENT.equalsIgnoreCase(part.getDisposition()) || !TextUtils.isEmpty(part.getFileName())) {
|
if (Part.ATTACHMENT.equalsIgnoreCase(part.getDisposition()) || !TextUtils.isEmpty(part.getFileName())) {
|
||||||
ContentType ct = new ContentType(part.getContentType());
|
ContentType ct = new ContentType(part.getContentType());
|
||||||
EntityAttachment attachment = new EntityAttachment();
|
EntityAttachment attachment = new EntityAttachment();
|
||||||
attachment.sequence = result.size() + 1;
|
|
||||||
attachment.name = part.getFileName();
|
attachment.name = part.getFileName();
|
||||||
attachment.type = ct.getBaseType();
|
attachment.type = ct.getBaseType();
|
||||||
attachment.size = part.getSize();
|
attachment.size = part.getSize();
|
||||||
|
|
|
@ -338,7 +338,7 @@ public class ServiceSynchronize extends LifecycleService {
|
||||||
LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(ServiceSynchronize.this);
|
LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(ServiceSynchronize.this);
|
||||||
lbm.registerReceiver(processReceiver, new IntentFilter(ACTION_PROCESS_FOLDER));
|
lbm.registerReceiver(processReceiver, new IntentFilter(ACTION_PROCESS_FOLDER));
|
||||||
Log.i(Helper.TAG, "listen process folder");
|
Log.i(Helper.TAG, "listen process folder");
|
||||||
for (final EntityFolder folder : db.folder().getFolders(account.id, false))
|
for (final EntityFolder folder : db.folder().getFolders(account.id))
|
||||||
if (!EntityFolder.TYPE_OUTBOX.equals(folder.type))
|
if (!EntityFolder.TYPE_OUTBOX.equals(folder.type))
|
||||||
lbm.sendBroadcast(new Intent(ACTION_PROCESS_FOLDER).putExtra("folder", folder.id));
|
lbm.sendBroadcast(new Intent(ACTION_PROCESS_FOLDER).putExtra("folder", folder.id));
|
||||||
|
|
||||||
|
@ -1002,9 +1002,13 @@ public class ServiceSynchronize extends LifecycleService {
|
||||||
message.id = db.message().insertMessage(message);
|
message.id = db.message().insertMessage(message);
|
||||||
Log.i(Helper.TAG, folder.name + " added id=" + message.id);
|
Log.i(Helper.TAG, folder.name + " added id=" + message.id);
|
||||||
|
|
||||||
|
int sequence = 0;
|
||||||
for (EntityAttachment attachment : helper.getAttachments()) {
|
for (EntityAttachment attachment : helper.getAttachments()) {
|
||||||
Log.i(Helper.TAG, "attachment name=" + attachment.name + " type=" + attachment.type);
|
sequence++;
|
||||||
|
Log.i(Helper.TAG, "attachment seq=" + sequence +
|
||||||
|
" name=" + attachment.name + " type=" + attachment.type);
|
||||||
attachment.message = message.id;
|
attachment.message = message.id;
|
||||||
|
attachment.sequence = sequence;
|
||||||
attachment.id = db.attachment().insertAttachment(attachment);
|
attachment.id = db.attachment().insertAttachment(attachment);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,19 @@
|
||||||
|
package eu.faircode.email;
|
||||||
|
|
||||||
|
import android.support.annotation.NonNull;
|
||||||
|
|
||||||
|
public class TupleAttachment {
|
||||||
|
@NonNull
|
||||||
|
public Long id;
|
||||||
|
@NonNull
|
||||||
|
public Long message;
|
||||||
|
@NonNull
|
||||||
|
public Integer sequence;
|
||||||
|
public String name;
|
||||||
|
@NonNull
|
||||||
|
public String type;
|
||||||
|
public Integer size;
|
||||||
|
public Integer progress;
|
||||||
|
@NonNull
|
||||||
|
public boolean content;
|
||||||
|
}
|
|
@ -153,10 +153,11 @@
|
||||||
android:layout_marginEnd="6dp"
|
android:layout_marginEnd="6dp"
|
||||||
android:layout_marginStart="6dp"
|
android:layout_marginStart="6dp"
|
||||||
android:layout_marginTop="3dp"
|
android:layout_marginTop="3dp"
|
||||||
android:maxHeight="90dp"
|
|
||||||
android:scrollbarStyle="outsideOverlay"
|
android:scrollbarStyle="outsideOverlay"
|
||||||
android:scrollbars="vertical"
|
android:scrollbars="vertical"
|
||||||
|
app:layout_constrainedHeight="true"
|
||||||
app:layout_constraintEnd_toEndOf="parent"
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintHeight_max="150dp"
|
||||||
app:layout_constraintStart_toStartOf="parent"
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
app:layout_constraintTop_toBottomOf="@+id/vSeparatorAttachments" />
|
app:layout_constraintTop_toBottomOf="@+id/vSeparatorAttachments" />
|
||||||
|
|
||||||
|
|
|
@ -36,20 +36,8 @@
|
||||||
android:text="10 kB"
|
android:text="10 kB"
|
||||||
android:textAppearance="@style/TextAppearance.AppCompat.Small"
|
android:textAppearance="@style/TextAppearance.AppCompat.Small"
|
||||||
app:layout_constraintBottom_toBottomOf="@id/ivAttachments"
|
app:layout_constraintBottom_toBottomOf="@id/ivAttachments"
|
||||||
app:layout_constraintEnd_toStartOf="@+id/tvProgress"
|
|
||||||
app:layout_constraintStart_toEndOf="@id/tvName"
|
|
||||||
app:layout_constraintTop_toTopOf="@id/ivAttachments" />
|
|
||||||
|
|
||||||
<TextView
|
|
||||||
android:id="@+id/tvProgress"
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:layout_marginStart="6dp"
|
|
||||||
android:text="50 %"
|
|
||||||
android:textAppearance="@style/TextAppearance.AppCompat.Small"
|
|
||||||
app:layout_constraintBottom_toBottomOf="@id/ivAttachments"
|
|
||||||
app:layout_constraintEnd_toStartOf="@+id/ivStatus"
|
app:layout_constraintEnd_toStartOf="@+id/ivStatus"
|
||||||
app:layout_constraintStart_toEndOf="@id/tvSize"
|
app:layout_constraintStart_toEndOf="@id/tvName"
|
||||||
app:layout_constraintTop_toTopOf="@id/ivAttachments" />
|
app:layout_constraintTop_toTopOf="@id/ivAttachments" />
|
||||||
|
|
||||||
<ImageView
|
<ImageView
|
||||||
|
@ -59,6 +47,16 @@
|
||||||
android:layout_marginStart="6dp"
|
android:layout_marginStart="6dp"
|
||||||
android:src="@drawable/baseline_get_app_24"
|
android:src="@drawable/baseline_get_app_24"
|
||||||
app:layout_constraintEnd_toEndOf="parent"
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
app:layout_constraintStart_toEndOf="@id/tvProgress"
|
app:layout_constraintStart_toEndOf="@id/tvSize"
|
||||||
app:layout_constraintTop_toTopOf="parent" />
|
app:layout_constraintTop_toTopOf="parent" />
|
||||||
|
|
||||||
|
<ProgressBar
|
||||||
|
android:id="@+id/progressbar"
|
||||||
|
style="@style/Widget.AppCompat.ProgressBar.Horizontal"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:progress="50"
|
||||||
|
app:layout_constraintEnd_toEndOf="parent"
|
||||||
|
app:layout_constraintStart_toStartOf="parent"
|
||||||
|
app:layout_constraintTop_toBottomOf="@id/ivAttachments" />
|
||||||
</android.support.constraint.ConstraintLayout>
|
</android.support.constraint.ConstraintLayout>
|
Loading…
Reference in New Issue