The block list is now its own functional piece, instead of just being a copy of the following/follows lists on account profiles.
This commit is contained in:
parent
9b6f5e63d3
commit
c4114b6be2
9 changed files with 399 additions and 121 deletions
|
@ -17,4 +17,5 @@ package com.keylesspalace.tusky;
|
||||||
|
|
||||||
public interface AccountActionListener {
|
public interface AccountActionListener {
|
||||||
void onViewAccount(String id);
|
void onViewAccount(String id);
|
||||||
|
void onBlock(final boolean block, final String id, final int position);
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,152 +15,59 @@
|
||||||
|
|
||||||
package com.keylesspalace.tusky;
|
package com.keylesspalace.tusky;
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.support.v7.widget.RecyclerView;
|
import android.support.v7.widget.RecyclerView;
|
||||||
import android.view.LayoutInflater;
|
|
||||||
import android.view.View;
|
|
||||||
import android.view.ViewGroup;
|
|
||||||
import android.widget.TextView;
|
|
||||||
|
|
||||||
import com.android.volley.toolbox.ImageLoader;
|
|
||||||
import com.android.volley.toolbox.NetworkImageView;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public class AccountAdapter extends RecyclerView.Adapter {
|
abstract class AccountAdapter extends RecyclerView.Adapter {
|
||||||
private static final int VIEW_TYPE_ACCOUNT = 0;
|
List<Account> accountList;
|
||||||
private static final int VIEW_TYPE_FOOTER = 1;
|
AccountActionListener accountActionListener;
|
||||||
|
FooterActionListener footerActionListener;
|
||||||
|
FooterViewHolder.State footerState;
|
||||||
|
|
||||||
private List<Account> accounts;
|
AccountAdapter(AccountActionListener accountActionListener,
|
||||||
private AccountActionListener accountActionListener;
|
|
||||||
private FooterActionListener footerActionListener;
|
|
||||||
private FooterViewHolder.State footerState;
|
|
||||||
|
|
||||||
public AccountAdapter(AccountActionListener accountActionListener,
|
|
||||||
FooterActionListener footerActionListener) {
|
FooterActionListener footerActionListener) {
|
||||||
super();
|
super();
|
||||||
accounts = new ArrayList<>();
|
accountList = new ArrayList<>();
|
||||||
this.accountActionListener = accountActionListener;
|
this.accountActionListener = accountActionListener;
|
||||||
this.footerActionListener = footerActionListener;
|
this.footerActionListener = footerActionListener;
|
||||||
footerState = FooterViewHolder.State.LOADING;
|
footerState = FooterViewHolder.State.LOADING;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
|
|
||||||
switch (viewType) {
|
|
||||||
default:
|
|
||||||
case VIEW_TYPE_ACCOUNT: {
|
|
||||||
View view = LayoutInflater.from(parent.getContext())
|
|
||||||
.inflate(R.layout.item_account, parent, false);
|
|
||||||
return new AccountViewHolder(view);
|
|
||||||
}
|
|
||||||
case VIEW_TYPE_FOOTER: {
|
|
||||||
View view = LayoutInflater.from(parent.getContext())
|
|
||||||
.inflate(R.layout.item_footer, parent, false);
|
|
||||||
return new FooterViewHolder(view);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onBindViewHolder(RecyclerView.ViewHolder viewHolder, int position) {
|
|
||||||
if (position < accounts.size()) {
|
|
||||||
AccountViewHolder holder = (AccountViewHolder) viewHolder;
|
|
||||||
holder.setupWithAccount(accounts.get(position));
|
|
||||||
holder.setupActionListener(accountActionListener);
|
|
||||||
} else {
|
|
||||||
FooterViewHolder holder = (FooterViewHolder) viewHolder;
|
|
||||||
holder.setState(footerState);
|
|
||||||
holder.setupButton(footerActionListener);
|
|
||||||
holder.setRetryMessage(R.string.footer_retry_accounts);
|
|
||||||
holder.setEndOfTimelineMessage(R.string.footer_end_of_accounts);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int getItemCount() {
|
public int getItemCount() {
|
||||||
return accounts.size() + 1;
|
return accountList.size() + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
void update(List<Account> newAccounts) {
|
||||||
public int getItemViewType(int position) {
|
if (accountList == null || accountList.isEmpty()) {
|
||||||
if (position == accounts.size()) {
|
accountList = newAccounts;
|
||||||
return VIEW_TYPE_FOOTER;
|
|
||||||
} else {
|
} else {
|
||||||
return VIEW_TYPE_ACCOUNT;
|
int index = newAccounts.indexOf(accountList.get(0));
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void update(List<Account> newAccounts) {
|
|
||||||
if (accounts == null || accounts.isEmpty()) {
|
|
||||||
accounts = newAccounts;
|
|
||||||
} else {
|
|
||||||
int index = newAccounts.indexOf(accounts.get(0));
|
|
||||||
if (index == -1) {
|
if (index == -1) {
|
||||||
accounts.addAll(0, newAccounts);
|
accountList.addAll(0, newAccounts);
|
||||||
} else {
|
} else {
|
||||||
accounts.addAll(0, newAccounts.subList(0, index));
|
accountList.addAll(0, newAccounts.subList(0, index));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
notifyDataSetChanged();
|
notifyDataSetChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void addItems(List<Account> newAccounts) {
|
void addItems(List<Account> newAccounts) {
|
||||||
int end = accounts.size();
|
int end = accountList.size();
|
||||||
accounts.addAll(newAccounts);
|
accountList.addAll(newAccounts);
|
||||||
notifyItemRangeInserted(end, newAccounts.size());
|
notifyItemRangeInserted(end, newAccounts.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
public Account getItem(int position) {
|
public Account getItem(int position) {
|
||||||
if (position >= 0 && position < accounts.size()) {
|
if (position >= 0 && position < accountList.size()) {
|
||||||
return accounts.get(position);
|
return accountList.get(position);
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setFooterState(FooterViewHolder.State state) {
|
void setFooterState(FooterViewHolder.State state) {
|
||||||
this.footerState = state;
|
this.footerState = state;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class AccountViewHolder extends RecyclerView.ViewHolder {
|
|
||||||
private View container;
|
|
||||||
private TextView username;
|
|
||||||
private TextView displayName;
|
|
||||||
private TextView note;
|
|
||||||
private NetworkImageView avatar;
|
|
||||||
private String id;
|
|
||||||
|
|
||||||
public AccountViewHolder(View itemView) {
|
|
||||||
super(itemView);
|
|
||||||
container = itemView.findViewById(R.id.account_container);
|
|
||||||
username = (TextView) itemView.findViewById(R.id.account_username);
|
|
||||||
displayName = (TextView) itemView.findViewById(R.id.account_display_name);
|
|
||||||
note = (TextView) itemView.findViewById(R.id.account_note);
|
|
||||||
avatar = (NetworkImageView) itemView.findViewById(R.id.account_avatar);
|
|
||||||
avatar.setDefaultImageResId(R.drawable.avatar_default);
|
|
||||||
avatar.setErrorImageResId(R.drawable.avatar_error);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setupWithAccount(Account account) {
|
|
||||||
id = account.id;
|
|
||||||
String format = username.getContext().getString(R.string.status_username_format);
|
|
||||||
String formattedUsername = String.format(format, account.username);
|
|
||||||
username.setText(formattedUsername);
|
|
||||||
displayName.setText(account.displayName);
|
|
||||||
note.setText(account.note);
|
|
||||||
Context context = avatar.getContext();
|
|
||||||
ImageLoader imageLoader = VolleySingleton.getInstance(context).getImageLoader();
|
|
||||||
avatar.setImageUrl(account.avatar, imageLoader);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setupActionListener(final AccountActionListener listener) {
|
|
||||||
container.setOnClickListener(new View.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(View v) {
|
|
||||||
listener.onViewAccount(id);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,9 +31,11 @@ import android.view.View;
|
||||||
import android.view.ViewGroup;
|
import android.view.ViewGroup;
|
||||||
|
|
||||||
import com.android.volley.AuthFailureError;
|
import com.android.volley.AuthFailureError;
|
||||||
|
import com.android.volley.Request;
|
||||||
import com.android.volley.Response;
|
import com.android.volley.Response;
|
||||||
import com.android.volley.VolleyError;
|
import com.android.volley.VolleyError;
|
||||||
import com.android.volley.toolbox.JsonArrayRequest;
|
import com.android.volley.toolbox.JsonArrayRequest;
|
||||||
|
import com.android.volley.toolbox.StringRequest;
|
||||||
|
|
||||||
import org.json.JSONArray;
|
import org.json.JSONArray;
|
||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
|
@ -123,7 +125,11 @@ public class AccountFragment extends Fragment implements AccountActionListener,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
recyclerView.addOnScrollListener(scrollListener);
|
recyclerView.addOnScrollListener(scrollListener);
|
||||||
adapter = new AccountAdapter(this, this);
|
if (type == Type.BLOCKS) {
|
||||||
|
adapter = new BlocksAdapter(this, this);
|
||||||
|
} else {
|
||||||
|
adapter = new FollowAdapter(this, this);
|
||||||
|
}
|
||||||
recyclerView.setAdapter(adapter);
|
recyclerView.setAdapter(adapter);
|
||||||
|
|
||||||
if (jumpToTopAllowed()) {
|
if (jumpToTopAllowed()) {
|
||||||
|
@ -266,6 +272,52 @@ public class AccountFragment extends Fragment implements AccountActionListener,
|
||||||
startActivity(intent);
|
startActivity(intent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void onBlock(final boolean block, final String id, final int position) {
|
||||||
|
String endpoint;
|
||||||
|
if (!block) {
|
||||||
|
endpoint = String.format(getString(R.string.endpoint_unblock), id);
|
||||||
|
} else {
|
||||||
|
endpoint = String.format(getString(R.string.endpoint_block), id);
|
||||||
|
}
|
||||||
|
String url = "https://" + domain + endpoint;
|
||||||
|
StringRequest request = new StringRequest(Request.Method.POST, url,
|
||||||
|
new Response.Listener<String>() {
|
||||||
|
@Override
|
||||||
|
public void onResponse(String response) {
|
||||||
|
onBlockSuccess(block, position);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
new Response.ErrorListener() {
|
||||||
|
@Override
|
||||||
|
public void onErrorResponse(VolleyError error) {
|
||||||
|
onBlockFailure(block, id);
|
||||||
|
}
|
||||||
|
}) {
|
||||||
|
@Override
|
||||||
|
public Map<String, String> getHeaders() throws AuthFailureError {
|
||||||
|
Map<String, String> headers = new HashMap<>();
|
||||||
|
headers.put("Authorization", "Bearer " + accessToken);
|
||||||
|
return headers;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
VolleySingleton.getInstance(getContext()).addToRequestQueue(request);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void onBlockSuccess(boolean blocked, int position) {
|
||||||
|
BlocksAdapter blocksAdapter = (BlocksAdapter) adapter;
|
||||||
|
blocksAdapter.setBlocked(blocked, position);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void onBlockFailure(boolean block, String id) {
|
||||||
|
String verb;
|
||||||
|
if (block) {
|
||||||
|
verb = "block";
|
||||||
|
} else {
|
||||||
|
verb = "unblock";
|
||||||
|
}
|
||||||
|
Log.e(TAG, String.format("Failed to %s account id %s", verb, id));
|
||||||
|
}
|
||||||
|
|
||||||
private boolean jumpToTopAllowed() {
|
private boolean jumpToTopAllowed() {
|
||||||
return type != Type.BLOCKS;
|
return type != Type.BLOCKS;
|
||||||
}
|
}
|
||||||
|
|
141
app/src/main/java/com/keylesspalace/tusky/BlocksAdapter.java
Normal file
141
app/src/main/java/com/keylesspalace/tusky/BlocksAdapter.java
Normal file
|
@ -0,0 +1,141 @@
|
||||||
|
/* Copyright 2017 Andrew Dawson
|
||||||
|
*
|
||||||
|
* This file is part of Tusky.
|
||||||
|
*
|
||||||
|
* Tusky is free software: you can redistribute it and/or modify it under the terms of the GNU
|
||||||
|
* General Public License as published by the Free Software Foundation, either version 3 of the
|
||||||
|
* License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
|
||||||
|
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
|
||||||
|
* Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along with Tusky. If not, see
|
||||||
|
* <http://www.gnu.org/licenses/>. */
|
||||||
|
|
||||||
|
package com.keylesspalace.tusky;
|
||||||
|
|
||||||
|
import android.support.v7.widget.RecyclerView;
|
||||||
|
import android.view.LayoutInflater;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.widget.Button;
|
||||||
|
import android.widget.TextView;
|
||||||
|
|
||||||
|
import com.android.volley.toolbox.NetworkImageView;
|
||||||
|
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
class BlocksAdapter extends AccountAdapter {
|
||||||
|
private static final int VIEW_TYPE_BLOCKED_USER = 0;
|
||||||
|
private static final int VIEW_TYPE_FOOTER = 1;
|
||||||
|
|
||||||
|
private Set<Integer> unblockedAccountPositions;
|
||||||
|
|
||||||
|
BlocksAdapter(AccountActionListener accountActionListener,
|
||||||
|
FooterActionListener footerActionListener) {
|
||||||
|
super(accountActionListener, footerActionListener);
|
||||||
|
unblockedAccountPositions = new HashSet<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
|
||||||
|
switch (viewType) {
|
||||||
|
default:
|
||||||
|
case VIEW_TYPE_BLOCKED_USER: {
|
||||||
|
View view = LayoutInflater.from(parent.getContext())
|
||||||
|
.inflate(R.layout.item_blocked_user, parent, false);
|
||||||
|
return new BlockedUserViewHolder(view);
|
||||||
|
}
|
||||||
|
case VIEW_TYPE_FOOTER: {
|
||||||
|
View view = LayoutInflater.from(parent.getContext())
|
||||||
|
.inflate(R.layout.item_footer, parent, false);
|
||||||
|
return new FooterViewHolder(view);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onBindViewHolder(RecyclerView.ViewHolder viewHolder, int position) {
|
||||||
|
if (position < accountList.size()) {
|
||||||
|
BlockedUserViewHolder holder = (BlockedUserViewHolder) viewHolder;
|
||||||
|
holder.setupWithAccount(accountList.get(position));
|
||||||
|
boolean blocked = !unblockedAccountPositions.contains(position);
|
||||||
|
holder.setupActionListener(accountActionListener, blocked, position);
|
||||||
|
} else {
|
||||||
|
FooterViewHolder holder = (FooterViewHolder) viewHolder;
|
||||||
|
holder.setState(footerState);
|
||||||
|
holder.setupButton(footerActionListener);
|
||||||
|
holder.setRetryMessage(R.string.footer_retry_accounts);
|
||||||
|
holder.setEndOfTimelineMessage(R.string.footer_end_of_accounts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getItemViewType(int position) {
|
||||||
|
if (position == accountList.size()) {
|
||||||
|
return VIEW_TYPE_FOOTER;
|
||||||
|
} else {
|
||||||
|
return VIEW_TYPE_BLOCKED_USER;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setBlocked(boolean blocked, int position) {
|
||||||
|
if (blocked) {
|
||||||
|
unblockedAccountPositions.remove(position);
|
||||||
|
} else {
|
||||||
|
unblockedAccountPositions.add(position);
|
||||||
|
}
|
||||||
|
notifyItemChanged(position);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class BlockedUserViewHolder extends RecyclerView.ViewHolder {
|
||||||
|
private NetworkImageView avatar;
|
||||||
|
private TextView username;
|
||||||
|
private TextView displayName;
|
||||||
|
private Button unblock;
|
||||||
|
private String id;
|
||||||
|
|
||||||
|
BlockedUserViewHolder(View itemView) {
|
||||||
|
super(itemView);
|
||||||
|
avatar = (NetworkImageView) itemView.findViewById(R.id.blocked_user_avatar);
|
||||||
|
displayName = (TextView) itemView.findViewById(R.id.blocked_user_display_name);
|
||||||
|
username = (TextView) itemView.findViewById(R.id.blocked_user_username);
|
||||||
|
unblock = (Button) itemView.findViewById(R.id.blocked_user_unblock);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setupWithAccount(Account account) {
|
||||||
|
id = account.id;
|
||||||
|
displayName.setText(account.displayName);
|
||||||
|
String format = username.getContext().getString(R.string.status_username_format);
|
||||||
|
String formattedUsername = String.format(format, account.username);
|
||||||
|
username.setText(formattedUsername);
|
||||||
|
avatar.setImageUrl(account.avatar,
|
||||||
|
VolleySingleton.getInstance(avatar.getContext()).getImageLoader());
|
||||||
|
}
|
||||||
|
|
||||||
|
void setupActionListener(final AccountActionListener listener, final boolean blocked,
|
||||||
|
final int position) {
|
||||||
|
int unblockTextId;
|
||||||
|
if (blocked) {
|
||||||
|
unblockTextId = R.string.action_unblock;
|
||||||
|
} else {
|
||||||
|
unblockTextId = R.string.action_block;
|
||||||
|
}
|
||||||
|
unblock.setText(unblock.getContext().getString(unblockTextId));
|
||||||
|
unblock.setOnClickListener(new View.OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(View v) {
|
||||||
|
listener.onBlock(!blocked, id, position);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
avatar.setOnClickListener(new View.OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(View v) {
|
||||||
|
listener.onViewAccount(id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
119
app/src/main/java/com/keylesspalace/tusky/FollowAdapter.java
Normal file
119
app/src/main/java/com/keylesspalace/tusky/FollowAdapter.java
Normal file
|
@ -0,0 +1,119 @@
|
||||||
|
/* Copyright 2017 Andrew Dawson
|
||||||
|
*
|
||||||
|
* This file is part of Tusky.
|
||||||
|
*
|
||||||
|
* Tusky is free software: you can redistribute it and/or modify it under the terms of the GNU
|
||||||
|
* General Public License as published by the Free Software Foundation, either version 3 of the
|
||||||
|
* License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
|
||||||
|
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
|
||||||
|
* Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License along with Tusky. If not, see
|
||||||
|
* <http://www.gnu.org/licenses/>. */
|
||||||
|
|
||||||
|
package com.keylesspalace.tusky;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.support.v7.widget.RecyclerView;
|
||||||
|
import android.view.LayoutInflater;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.widget.TextView;
|
||||||
|
|
||||||
|
import com.android.volley.toolbox.ImageLoader;
|
||||||
|
import com.android.volley.toolbox.NetworkImageView;
|
||||||
|
|
||||||
|
/** Both for follows and following lists. */
|
||||||
|
class FollowAdapter extends AccountAdapter {
|
||||||
|
private static final int VIEW_TYPE_ACCOUNT = 0;
|
||||||
|
private static final int VIEW_TYPE_FOOTER = 1;
|
||||||
|
|
||||||
|
FollowAdapter(AccountActionListener accountActionListener,
|
||||||
|
FooterActionListener footerActionListener) {
|
||||||
|
super(accountActionListener, footerActionListener);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
|
||||||
|
switch (viewType) {
|
||||||
|
default:
|
||||||
|
case VIEW_TYPE_ACCOUNT: {
|
||||||
|
View view = LayoutInflater.from(parent.getContext())
|
||||||
|
.inflate(R.layout.item_account, parent, false);
|
||||||
|
return new AccountViewHolder(view);
|
||||||
|
}
|
||||||
|
case VIEW_TYPE_FOOTER: {
|
||||||
|
View view = LayoutInflater.from(parent.getContext())
|
||||||
|
.inflate(R.layout.item_footer, parent, false);
|
||||||
|
return new FooterViewHolder(view);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onBindViewHolder(RecyclerView.ViewHolder viewHolder, int position) {
|
||||||
|
if (position < accountList.size()) {
|
||||||
|
AccountViewHolder holder = (AccountViewHolder) viewHolder;
|
||||||
|
holder.setupWithAccount(accountList.get(position));
|
||||||
|
holder.setupActionListener(accountActionListener);
|
||||||
|
} else {
|
||||||
|
FooterViewHolder holder = (FooterViewHolder) viewHolder;
|
||||||
|
holder.setState(footerState);
|
||||||
|
holder.setupButton(footerActionListener);
|
||||||
|
holder.setRetryMessage(R.string.footer_retry_accounts);
|
||||||
|
holder.setEndOfTimelineMessage(R.string.footer_end_of_accounts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getItemViewType(int position) {
|
||||||
|
if (position == accountList.size()) {
|
||||||
|
return VIEW_TYPE_FOOTER;
|
||||||
|
} else {
|
||||||
|
return VIEW_TYPE_ACCOUNT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class AccountViewHolder extends RecyclerView.ViewHolder {
|
||||||
|
private View container;
|
||||||
|
private TextView username;
|
||||||
|
private TextView displayName;
|
||||||
|
private TextView note;
|
||||||
|
private NetworkImageView avatar;
|
||||||
|
private String id;
|
||||||
|
|
||||||
|
AccountViewHolder(View itemView) {
|
||||||
|
super(itemView);
|
||||||
|
container = itemView.findViewById(R.id.account_container);
|
||||||
|
username = (TextView) itemView.findViewById(R.id.account_username);
|
||||||
|
displayName = (TextView) itemView.findViewById(R.id.account_display_name);
|
||||||
|
note = (TextView) itemView.findViewById(R.id.account_note);
|
||||||
|
avatar = (NetworkImageView) itemView.findViewById(R.id.account_avatar);
|
||||||
|
avatar.setDefaultImageResId(R.drawable.avatar_default);
|
||||||
|
avatar.setErrorImageResId(R.drawable.avatar_error);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setupWithAccount(Account account) {
|
||||||
|
id = account.id;
|
||||||
|
String format = username.getContext().getString(R.string.status_username_format);
|
||||||
|
String formattedUsername = String.format(format, account.username);
|
||||||
|
username.setText(formattedUsername);
|
||||||
|
displayName.setText(account.displayName);
|
||||||
|
note.setText(account.note);
|
||||||
|
Context context = avatar.getContext();
|
||||||
|
ImageLoader imageLoader = VolleySingleton.getInstance(context).getImageLoader();
|
||||||
|
avatar.setImageUrl(account.avatar, imageLoader);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setupActionListener(final AccountActionListener listener) {
|
||||||
|
container.setOnClickListener(new View.OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(View v) {
|
||||||
|
listener.onViewAccount(id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
57
app/src/main/res/layout/item_blocked_user.xml
Normal file
57
app/src/main/res/layout/item_blocked_user.xml
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content">
|
||||||
|
|
||||||
|
<com.android.volley.toolbox.NetworkImageView
|
||||||
|
android:layout_width="64dp"
|
||||||
|
android:layout_height="64dp"
|
||||||
|
android:scaleType="fitCenter"
|
||||||
|
android:id="@+id/blocked_user_avatar"
|
||||||
|
android:padding="@dimen/status_avatar_padding"
|
||||||
|
android:layout_alignParentLeft="true" />
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="64dp"
|
||||||
|
android:orientation="vertical"
|
||||||
|
android:layout_toRightOf="@id/blocked_user_avatar">
|
||||||
|
|
||||||
|
<Space
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="0dp"
|
||||||
|
android:layout_weight="1" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:id="@+id/blocked_user_display_name"
|
||||||
|
android:text="Display Name" />
|
||||||
|
|
||||||
|
<Space
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="0dp"
|
||||||
|
android:layout_weight="1" />
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:id="@+id/blocked_user_username"
|
||||||
|
android:text="\@username\@example.com" />
|
||||||
|
|
||||||
|
<Space
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="0dp"
|
||||||
|
android:layout_weight="1" />
|
||||||
|
|
||||||
|
</LinearLayout>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:id="@+id/blocked_user_unblock"
|
||||||
|
android:layout_alignParentRight="true"
|
||||||
|
android:layout_centerVertical="true"
|
||||||
|
android:text="@string/action_unblock" />
|
||||||
|
|
||||||
|
</RelativeLayout>
|
|
@ -21,7 +21,7 @@
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:id="@+id/footer_retry_message"
|
android:id="@+id/footer_retry_message"
|
||||||
android:padding="@dimen/footer_text_padding"/>
|
android:padding="@dimen/footer_text_padding" />
|
||||||
|
|
||||||
<Button
|
<Button
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
|
@ -40,6 +40,7 @@
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:id="@+id/footer_end_of_timeline_text"
|
android:id="@+id/footer_end_of_timeline_text"
|
||||||
|
android:padding="@dimen/footer_text_padding"
|
||||||
android:visibility="gone" />
|
android:visibility="gone" />
|
||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
|
@ -53,20 +53,20 @@
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:textAppearance="@android:style/TextAppearance.DeviceDefault.Small"
|
android:textAppearance="@android:style/TextAppearance.DeviceDefault.Small"
|
||||||
android:textStyle="normal|bold" />
|
android:textStyle="normal|bold"
|
||||||
|
android:paddingRight="@dimen/status_display_name_right_padding" />
|
||||||
|
|
||||||
<TextView
|
<TextView
|
||||||
android:id="@+id/status_username"
|
android:id="@+id/status_username"
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:paddingLeft="@dimen/status_username_left_margin"
|
android:textColor="?attr/status_text_color_secondary"
|
||||||
android:textColor="?attr/status_text_color_secondary" />
|
android:paddingRight="@dimen/status_username_right_padding" />
|
||||||
|
|
||||||
<TextView
|
<TextView
|
||||||
android:id="@+id/status_since_created"
|
android:id="@+id/status_since_created"
|
||||||
android:layout_width="wrap_content"
|
android:layout_width="wrap_content"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:paddingLeft="@dimen/status_since_created_left_margin"
|
|
||||||
android:textColor="?attr/status_text_color_secondary" />
|
android:textColor="?attr/status_text_color_secondary" />
|
||||||
|
|
||||||
</com.keylesspalace.tusky.FlowLayout>
|
</com.keylesspalace.tusky.FlowLayout>
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
<resources>
|
<resources>
|
||||||
<dimen name="activity_horizontal_margin">0dp</dimen>
|
<dimen name="activity_horizontal_margin">0dp</dimen>
|
||||||
<dimen name="activity_vertical_margin">0dp</dimen>
|
<dimen name="activity_vertical_margin">0dp</dimen>
|
||||||
<dimen name="status_username_left_margin">4dp</dimen>
|
<dimen name="status_display_name_right_padding">4dp</dimen>
|
||||||
<dimen name="status_since_created_left_margin">4dp</dimen>
|
<dimen name="status_username_right_padding">4dp</dimen>
|
||||||
<dimen name="status_avatar_padding">8dp</dimen>
|
<dimen name="status_avatar_padding">8dp</dimen>
|
||||||
<dimen name="status_reblogged_bar_top_padding">8dp</dimen>
|
<dimen name="status_reblogged_bar_top_padding">8dp</dimen>
|
||||||
<dimen name="status_reblogged_icon_left_padding">40dp</dimen>
|
<dimen name="status_reblogged_icon_left_padding">40dp</dimen>
|
||||||
|
|
Loading…
Reference in a new issue