2017-01-23 06:19:30 +01:00
|
|
|
/* Copyright 2017 Andrew Dawson
|
|
|
|
*
|
2017-04-10 02:12:31 +02:00
|
|
|
* This file is a part of Tusky.
|
2017-01-23 06:19:30 +01:00
|
|
|
*
|
2017-04-10 02:12:31 +02:00
|
|
|
* 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.
|
2017-01-23 06:19:30 +01:00
|
|
|
*
|
|
|
|
* Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
|
2017-04-10 02:12:31 +02:00
|
|
|
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
|
|
|
|
* Public License for more details.
|
2017-01-23 06:19:30 +01:00
|
|
|
*
|
2017-04-10 02:12:31 +02:00
|
|
|
* You should have received a copy of the GNU General Public License along with Tusky; if not,
|
|
|
|
* see <http://www.gnu.org/licenses>. */
|
2017-01-23 06:19:30 +01:00
|
|
|
|
2017-05-05 00:55:34 +02:00
|
|
|
package com.keylesspalace.tusky.fragment;
|
2017-01-23 06:19:30 +01:00
|
|
|
|
2017-09-25 20:15:04 +02:00
|
|
|
import android.content.ClipData;
|
|
|
|
import android.content.ClipboardManager;
|
|
|
|
import android.content.Context;
|
2017-01-23 06:19:30 +01:00
|
|
|
import android.content.Intent;
|
|
|
|
import android.content.SharedPreferences;
|
|
|
|
import android.os.Bundle;
|
2017-11-03 22:17:31 +01:00
|
|
|
import android.support.annotation.NonNull;
|
2017-01-23 06:19:30 +01:00
|
|
|
import android.support.annotation.Nullable;
|
2017-07-14 07:26:58 +02:00
|
|
|
import android.support.v4.app.ActivityOptionsCompat;
|
2017-06-06 23:15:29 +02:00
|
|
|
import android.support.v4.content.LocalBroadcastManager;
|
2017-07-14 07:26:58 +02:00
|
|
|
import android.support.v4.view.ViewCompat;
|
2017-01-23 06:19:30 +01:00
|
|
|
import android.support.v7.widget.PopupMenu;
|
2017-02-27 06:21:46 +01:00
|
|
|
import android.text.Spanned;
|
2017-01-23 06:19:30 +01:00
|
|
|
import android.view.MenuItem;
|
|
|
|
import android.view.View;
|
|
|
|
|
2017-05-05 00:55:34 +02:00
|
|
|
import com.keylesspalace.tusky.AccountActivity;
|
|
|
|
import com.keylesspalace.tusky.BaseActivity;
|
|
|
|
import com.keylesspalace.tusky.ComposeActivity;
|
|
|
|
import com.keylesspalace.tusky.R;
|
|
|
|
import com.keylesspalace.tusky.ReportActivity;
|
2017-06-25 07:07:41 +02:00
|
|
|
import com.keylesspalace.tusky.ViewMediaActivity;
|
2017-05-05 00:55:34 +02:00
|
|
|
import com.keylesspalace.tusky.ViewTagActivity;
|
|
|
|
import com.keylesspalace.tusky.ViewThreadActivity;
|
|
|
|
import com.keylesspalace.tusky.ViewVideoActivity;
|
2017-03-09 16:59:18 +01:00
|
|
|
import com.keylesspalace.tusky.entity.Relationship;
|
2017-03-09 00:27:37 +01:00
|
|
|
import com.keylesspalace.tusky.entity.Status;
|
2017-05-05 00:55:34 +02:00
|
|
|
import com.keylesspalace.tusky.interfaces.AdapterItemRemover;
|
2017-06-22 20:01:25 +02:00
|
|
|
import com.keylesspalace.tusky.network.MastodonApi;
|
2017-06-06 23:15:29 +02:00
|
|
|
import com.keylesspalace.tusky.receiver.TimelineReceiver;
|
2017-05-05 00:55:34 +02:00
|
|
|
import com.keylesspalace.tusky.util.HtmlUtils;
|
2017-01-23 06:19:30 +01:00
|
|
|
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.List;
|
|
|
|
|
2017-03-09 16:59:18 +01:00
|
|
|
import okhttp3.ResponseBody;
|
2017-03-09 00:46:13 +01:00
|
|
|
import retrofit2.Call;
|
|
|
|
import retrofit2.Callback;
|
2017-04-22 02:58:44 +02:00
|
|
|
import retrofit2.Response;
|
2017-03-09 00:46:13 +01:00
|
|
|
|
2017-01-23 06:19:30 +01:00
|
|
|
/* Note from Andrew on Jan. 22, 2017: This class is a design problem for me, so I left it with an
|
|
|
|
* awkward name. TimelineFragment and NotificationFragment have significant overlap but the nature
|
|
|
|
* of that is complicated by how they're coupled with Status and Notification and the corresponding
|
|
|
|
* adapters. I feel like the profile pages and thread viewer, which I haven't made yet, will also
|
|
|
|
* overlap functionality. So, I'm momentarily leaving it and hopefully working on those will clear
|
|
|
|
* up what needs to be where. */
|
2017-07-14 03:31:31 +02:00
|
|
|
public abstract class SFragment extends BaseFragment implements AdapterItemRemover {
|
2017-06-30 08:31:58 +02:00
|
|
|
protected static final int COMPOSE_RESULT = 1;
|
|
|
|
|
2017-01-23 06:19:30 +01:00
|
|
|
protected String loggedInAccountId;
|
|
|
|
protected String loggedInUsername;
|
2017-06-30 08:31:58 +02:00
|
|
|
protected MastodonApi mastodonApi;
|
2017-01-23 06:19:30 +01:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onCreate(@Nullable Bundle savedInstanceState) {
|
|
|
|
super.onCreate(savedInstanceState);
|
|
|
|
|
2017-04-25 13:30:57 +02:00
|
|
|
SharedPreferences preferences = getPrivatePreferences();
|
2017-01-28 04:33:43 +01:00
|
|
|
loggedInAccountId = preferences.getString("loggedInAccountId", null);
|
|
|
|
loggedInUsername = preferences.getString("loggedInAccountUsername", null);
|
2017-03-14 19:24:38 +01:00
|
|
|
}
|
|
|
|
|
2017-04-22 05:16:05 +02:00
|
|
|
@Override
|
|
|
|
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
|
|
|
|
super.onActivityCreated(savedInstanceState);
|
|
|
|
BaseActivity activity = (BaseActivity) getActivity();
|
2017-06-30 08:31:58 +02:00
|
|
|
mastodonApi = activity.mastodonApi;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void startActivity(Intent intent) {
|
|
|
|
super.startActivity(intent);
|
|
|
|
getActivity().overridePendingTransition(R.anim.slide_from_right, R.anim.slide_to_left);
|
2017-01-23 06:19:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
protected void reply(Status status) {
|
2017-03-09 00:27:37 +01:00
|
|
|
String inReplyToId = status.getActionableId();
|
2017-03-31 21:47:41 +02:00
|
|
|
Status actionableStatus = status.getActionableStatus();
|
2017-11-16 19:18:11 +01:00
|
|
|
Status.Visibility replyVisibility = actionableStatus.getVisibility();
|
2017-04-07 20:37:27 +02:00
|
|
|
String contentWarning = actionableStatus.spoilerText;
|
2017-03-31 21:47:41 +02:00
|
|
|
Status.Mention[] mentions = actionableStatus.mentions;
|
2017-01-23 06:19:30 +01:00
|
|
|
List<String> mentionedUsernames = new ArrayList<>();
|
2017-06-27 00:10:01 +02:00
|
|
|
mentionedUsernames.add(actionableStatus.account.username);
|
2017-01-31 05:51:02 +01:00
|
|
|
for (Status.Mention mention : mentions) {
|
2017-03-09 00:27:37 +01:00
|
|
|
mentionedUsernames.add(mention.username);
|
2017-01-23 06:19:30 +01:00
|
|
|
}
|
|
|
|
mentionedUsernames.remove(loggedInUsername);
|
2017-11-01 20:59:29 +01:00
|
|
|
Intent intent = new ComposeActivity.IntentBuilder()
|
|
|
|
.inReplyToId(inReplyToId)
|
|
|
|
.replyVisibility(replyVisibility)
|
|
|
|
.contentWarning(contentWarning)
|
|
|
|
.mentionedUsernames(mentionedUsernames)
|
2017-11-16 19:18:11 +01:00
|
|
|
.repyingStatusAuthor(actionableStatus.account.localUsername)
|
2017-11-01 20:59:29 +01:00
|
|
|
.replyingStatusContent(actionableStatus.content.toString())
|
|
|
|
.build(getContext());
|
2017-04-15 19:44:29 +02:00
|
|
|
startActivityForResult(intent, COMPOSE_RESULT);
|
|
|
|
}
|
|
|
|
|
2017-07-12 21:54:52 +02:00
|
|
|
protected void reblogWithCallback(final Status status, final boolean reblog,
|
|
|
|
Callback<Status> callback) {
|
|
|
|
String id = status.getActionableId();
|
2017-03-09 00:46:13 +01:00
|
|
|
|
2017-03-15 01:34:27 +01:00
|
|
|
Call<Status> call;
|
2017-01-23 06:19:30 +01:00
|
|
|
if (reblog) {
|
2017-06-30 08:31:58 +02:00
|
|
|
call = mastodonApi.reblogStatus(id);
|
2017-01-23 06:19:30 +01:00
|
|
|
} else {
|
2017-06-30 08:31:58 +02:00
|
|
|
call = mastodonApi.unreblogStatus(id);
|
2017-01-23 06:19:30 +01:00
|
|
|
}
|
2017-07-12 21:54:52 +02:00
|
|
|
call.enqueue(callback);
|
2017-01-23 06:19:30 +01:00
|
|
|
}
|
|
|
|
|
2017-07-12 21:54:52 +02:00
|
|
|
protected void favouriteWithCallback(final Status status, final boolean favourite,
|
|
|
|
final Callback<Status> callback) {
|
|
|
|
String id = status.getActionableId();
|
2017-03-09 00:46:13 +01:00
|
|
|
|
2017-03-15 01:34:27 +01:00
|
|
|
Call<Status> call;
|
2017-03-09 00:46:13 +01:00
|
|
|
if (favourite) {
|
2017-06-30 08:31:58 +02:00
|
|
|
call = mastodonApi.favouriteStatus(id);
|
2017-03-09 00:46:13 +01:00
|
|
|
} else {
|
2017-06-30 08:31:58 +02:00
|
|
|
call = mastodonApi.unfavouriteStatus(id);
|
2017-03-09 00:46:13 +01:00
|
|
|
}
|
2017-07-12 21:54:52 +02:00
|
|
|
call.enqueue(callback);
|
2017-03-15 01:34:27 +01:00
|
|
|
callList.add(call);
|
2017-01-23 06:19:30 +01:00
|
|
|
}
|
|
|
|
|
2017-06-28 10:10:56 +02:00
|
|
|
protected void openReblog(@Nullable final Status status) {
|
|
|
|
if (status == null) return;
|
|
|
|
viewAccount(status.account.id);
|
|
|
|
}
|
|
|
|
|
2017-04-22 02:58:44 +02:00
|
|
|
private void mute(String id) {
|
2017-06-30 08:31:58 +02:00
|
|
|
Call<Relationship> call = mastodonApi.muteAccount(id);
|
2017-03-15 01:34:27 +01:00
|
|
|
call.enqueue(new Callback<Relationship>() {
|
2017-03-09 16:59:18 +01:00
|
|
|
@Override
|
2017-11-03 22:17:31 +01:00
|
|
|
public void onResponse(@NonNull Call<Relationship> call, @NonNull Response<Relationship> response) {}
|
2017-03-09 16:59:18 +01:00
|
|
|
|
2017-04-22 02:58:44 +02:00
|
|
|
@Override
|
2017-11-03 22:17:31 +01:00
|
|
|
public void onFailure(@NonNull Call<Relationship> call, @NonNull Throwable t) {}
|
2017-04-22 02:58:44 +02:00
|
|
|
});
|
|
|
|
callList.add(call);
|
2017-06-06 23:15:29 +02:00
|
|
|
Intent intent = new Intent(TimelineReceiver.Types.MUTE_ACCOUNT);
|
|
|
|
intent.putExtra("id", id);
|
|
|
|
LocalBroadcastManager.getInstance(getContext())
|
|
|
|
.sendBroadcast(intent);
|
2017-04-22 02:58:44 +02:00
|
|
|
}
|
2017-03-09 16:59:18 +01:00
|
|
|
|
2017-04-22 02:58:44 +02:00
|
|
|
private void block(String id) {
|
2017-06-30 08:31:58 +02:00
|
|
|
Call<Relationship> call = mastodonApi.blockAccount(id);
|
2017-04-22 02:58:44 +02:00
|
|
|
call.enqueue(new Callback<Relationship>() {
|
2017-03-09 16:59:18 +01:00
|
|
|
@Override
|
2017-11-03 22:17:31 +01:00
|
|
|
public void onResponse(@NonNull Call<Relationship> call, @NonNull retrofit2.Response<Relationship> response) {}
|
2017-03-09 16:59:18 +01:00
|
|
|
|
2017-04-22 02:58:44 +02:00
|
|
|
@Override
|
2017-11-03 22:17:31 +01:00
|
|
|
public void onFailure(@NonNull Call<Relationship> call, @NonNull Throwable t) {}
|
2017-03-09 16:59:18 +01:00
|
|
|
});
|
2017-03-15 01:34:27 +01:00
|
|
|
callList.add(call);
|
2017-06-06 23:15:29 +02:00
|
|
|
Intent intent = new Intent(TimelineReceiver.Types.BLOCK_ACCOUNT);
|
|
|
|
intent.putExtra("id", id);
|
|
|
|
LocalBroadcastManager.getInstance(getContext())
|
|
|
|
.sendBroadcast(intent);
|
2017-01-23 06:19:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private void delete(String id) {
|
2017-06-30 08:31:58 +02:00
|
|
|
Call<ResponseBody> call = mastodonApi.deleteStatus(id);
|
2017-03-15 01:34:27 +01:00
|
|
|
call.enqueue(new Callback<ResponseBody>() {
|
2017-03-09 16:59:18 +01:00
|
|
|
@Override
|
2017-11-03 22:17:31 +01:00
|
|
|
public void onResponse(@NonNull Call<ResponseBody> call, @NonNull retrofit2.Response<ResponseBody> response) {}
|
2017-03-09 16:59:18 +01:00
|
|
|
|
|
|
|
@Override
|
2017-11-03 22:17:31 +01:00
|
|
|
public void onFailure(@NonNull Call<ResponseBody> call, @NonNull Throwable t) {}
|
2017-03-09 16:59:18 +01:00
|
|
|
});
|
2017-03-15 01:34:27 +01:00
|
|
|
callList.add(call);
|
2017-01-23 06:19:30 +01:00
|
|
|
}
|
|
|
|
|
2017-07-14 03:31:31 +02:00
|
|
|
protected void more(final Status status, View view, final int position) {
|
2017-03-09 00:27:37 +01:00
|
|
|
final String id = status.getActionableId();
|
|
|
|
final String accountId = status.getActionableStatus().account.id;
|
|
|
|
final String accountUsename = status.getActionableStatus().account.username;
|
|
|
|
final Spanned content = status.getActionableStatus().content;
|
2017-03-09 17:37:24 +01:00
|
|
|
final String statusUrl = status.getActionableStatus().url;
|
2017-01-23 06:19:30 +01:00
|
|
|
PopupMenu popup = new PopupMenu(getContext(), view);
|
|
|
|
// Give a different menu depending on whether this is the user's own toot or not.
|
|
|
|
if (loggedInAccountId == null || !loggedInAccountId.equals(accountId)) {
|
|
|
|
popup.inflate(R.menu.status_more);
|
|
|
|
} else {
|
|
|
|
popup.inflate(R.menu.status_more_for_user);
|
|
|
|
}
|
|
|
|
popup.setOnMenuItemClickListener(
|
|
|
|
new PopupMenu.OnMenuItemClickListener() {
|
|
|
|
@Override
|
|
|
|
public boolean onMenuItemClick(MenuItem item) {
|
|
|
|
switch (item.getItemId()) {
|
2017-04-12 22:21:48 +02:00
|
|
|
case R.id.status_share_content: {
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
sb.append(status.account.username);
|
|
|
|
sb.append(" - ");
|
|
|
|
sb.append(status.content.toString());
|
|
|
|
|
|
|
|
Intent sendIntent = new Intent();
|
|
|
|
sendIntent.setAction(Intent.ACTION_SEND);
|
|
|
|
sendIntent.putExtra(Intent.EXTRA_TEXT, sb.toString());
|
|
|
|
sendIntent.setType("text/plain");
|
|
|
|
startActivity(Intent.createChooser(sendIntent, getResources().getText(R.string.send_status_content_to)));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
case R.id.status_share_link: {
|
2017-03-09 17:37:24 +01:00
|
|
|
Intent sendIntent = new Intent();
|
|
|
|
sendIntent.setAction(Intent.ACTION_SEND);
|
|
|
|
sendIntent.putExtra(Intent.EXTRA_TEXT, statusUrl);
|
|
|
|
sendIntent.setType("text/plain");
|
2017-04-12 22:21:48 +02:00
|
|
|
startActivity(Intent.createChooser(sendIntent, getResources().getText(R.string.send_status_link_to)));
|
2017-01-23 06:19:30 +01:00
|
|
|
return true;
|
|
|
|
}
|
2017-09-25 20:15:04 +02:00
|
|
|
case R.id.status_copy_link: {
|
|
|
|
ClipboardManager clipboard = (ClipboardManager)
|
|
|
|
getActivity().getSystemService(Context.CLIPBOARD_SERVICE);
|
|
|
|
ClipData clip = ClipData.newPlainText(null, statusUrl);
|
|
|
|
clipboard.setPrimaryClip(clip);
|
|
|
|
return true;
|
|
|
|
}
|
2017-04-22 02:58:44 +02:00
|
|
|
case R.id.status_mute: {
|
|
|
|
mute(accountId);
|
|
|
|
return true;
|
|
|
|
}
|
2017-01-23 06:19:30 +01:00
|
|
|
case R.id.status_block: {
|
|
|
|
block(accountId);
|
|
|
|
return true;
|
|
|
|
}
|
2017-02-27 06:21:46 +01:00
|
|
|
case R.id.status_report: {
|
|
|
|
openReportPage(accountId, accountUsename, id, content);
|
|
|
|
return true;
|
|
|
|
}
|
2017-01-23 06:19:30 +01:00
|
|
|
case R.id.status_delete: {
|
|
|
|
delete(id);
|
2017-07-14 03:31:31 +02:00
|
|
|
removeItem(position);
|
2017-01-23 06:19:30 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
popup.show();
|
|
|
|
}
|
|
|
|
|
2017-07-14 07:26:58 +02:00
|
|
|
protected void viewMedia(String[] urls, int urlIndex, Status.MediaAttachment.Type type,
|
|
|
|
@Nullable View view) {
|
2017-01-23 06:19:30 +01:00
|
|
|
switch (type) {
|
|
|
|
case IMAGE: {
|
2017-06-25 07:07:41 +02:00
|
|
|
Intent intent = new Intent(getContext(), ViewMediaActivity.class);
|
|
|
|
intent.putExtra("urls", urls);
|
|
|
|
intent.putExtra("urlIndex", urlIndex);
|
2017-07-14 07:26:58 +02:00
|
|
|
if (view != null) {
|
|
|
|
String url = urls[urlIndex];
|
|
|
|
ViewCompat.setTransitionName(view, url);
|
|
|
|
ActivityOptionsCompat options =
|
|
|
|
ActivityOptionsCompat.makeSceneTransitionAnimation(getActivity(),
|
|
|
|
view, url);
|
|
|
|
startActivity(intent, options.toBundle());
|
|
|
|
} else {
|
|
|
|
startActivity(intent);
|
|
|
|
}
|
2017-01-23 06:19:30 +01:00
|
|
|
break;
|
|
|
|
}
|
2017-03-09 21:21:55 +01:00
|
|
|
case GIFV:
|
2017-01-23 06:19:30 +01:00
|
|
|
case VIDEO: {
|
|
|
|
Intent intent = new Intent(getContext(), ViewVideoActivity.class);
|
2017-06-25 07:07:41 +02:00
|
|
|
intent.putExtra("url", urls[urlIndex]);
|
2017-01-23 06:19:30 +01:00
|
|
|
startActivity(intent);
|
|
|
|
break;
|
|
|
|
}
|
2017-03-06 01:58:36 +01:00
|
|
|
case UNKNOWN: {
|
|
|
|
/* Intentionally do nothing. This case is here is to handle when new attachment
|
|
|
|
* types are added to the API before code is added here to handle them. So, the
|
|
|
|
* best fallback is to just show the preview and ignore requests to view them. */
|
|
|
|
break;
|
|
|
|
}
|
2017-01-23 06:19:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void viewThread(Status status) {
|
|
|
|
Intent intent = new Intent(getContext(), ViewThreadActivity.class);
|
2017-11-09 10:08:52 +01:00
|
|
|
intent.putExtra("id", status.getActionableId());
|
|
|
|
intent.putExtra("url", status.getActionableStatus().url);
|
2017-01-23 06:19:30 +01:00
|
|
|
startActivity(intent);
|
|
|
|
}
|
2017-01-27 01:34:32 +01:00
|
|
|
|
|
|
|
protected void viewTag(String tag) {
|
|
|
|
Intent intent = new Intent(getContext(), ViewTagActivity.class);
|
|
|
|
intent.putExtra("hashtag", tag);
|
|
|
|
startActivity(intent);
|
|
|
|
}
|
2017-01-28 04:33:43 +01:00
|
|
|
|
2017-03-03 01:25:35 +01:00
|
|
|
protected void viewAccount(String id) {
|
2017-01-28 04:33:43 +01:00
|
|
|
Intent intent = new Intent(getContext(), AccountActivity.class);
|
|
|
|
intent.putExtra("id", id);
|
|
|
|
startActivity(intent);
|
|
|
|
}
|
2017-02-27 06:21:46 +01:00
|
|
|
|
2017-03-03 01:25:35 +01:00
|
|
|
protected void openReportPage(String accountId, String accountUsername, String statusId,
|
2017-06-30 08:31:58 +02:00
|
|
|
Spanned statusContent) {
|
2017-02-27 06:21:46 +01:00
|
|
|
Intent intent = new Intent(getContext(), ReportActivity.class);
|
|
|
|
intent.putExtra("account_id", accountId);
|
2017-03-03 01:25:35 +01:00
|
|
|
intent.putExtra("account_username", accountUsername);
|
2017-02-27 06:21:46 +01:00
|
|
|
intent.putExtra("status_id", statusId);
|
|
|
|
intent.putExtra("status_content", HtmlUtils.toHtml(statusContent));
|
|
|
|
startActivity(intent);
|
|
|
|
}
|
2017-01-23 06:19:30 +01:00
|
|
|
}
|