2
0
mirror of https://github.com/KDE/kdeconnect-android synced 2025-09-05 16:45:08 +00:00
Files
kdeconnect-android/src/org/kde/kdeconnect/Plugins/SharePlugin/SharePlugin.java

336 lines
11 KiB
Java
Raw Normal View History

2014-11-16 23:14:06 -08:00
/*
2020-08-17 16:17:20 +02:00
* SPDX-FileCopyrightText: 2014 Albert Vaca Cintora <albertvaka@gmail.com>
2014-11-16 23:14:06 -08:00
*
2020-08-17 16:17:20 +02:00
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
2018-09-29 20:28:47 +02:00
*/
2014-11-16 23:14:06 -08:00
package org.kde.kdeconnect.Plugins.SharePlugin;
import android.Manifest;
import android.app.Activity;
import android.content.ClipboardManager;
import android.content.Context;
import android.content.Intent;
import android.graphics.drawable.Drawable;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.util.Log;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.annotation.WorkerThread;
import androidx.core.content.ContextCompat;
import org.apache.commons.lang3.ArrayUtils;
import org.apache.commons.lang3.StringUtils;
import org.kde.kdeconnect.Helpers.FilesHelper;
import org.kde.kdeconnect.Helpers.IntentHelper;
import org.kde.kdeconnect.NetworkPacket;
import org.kde.kdeconnect.Plugins.Plugin;
import org.kde.kdeconnect.Plugins.PluginFactory;
import org.kde.kdeconnect.UserInterface.PluginSettingsFragment;
import org.kde.kdeconnect.async.BackgroundJob;
import org.kde.kdeconnect.async.BackgroundJobHandler;
import org.kde.kdeconnect_tp.R;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collections;
/**
* A Plugin for sharing and receiving files and uris.
* <p>
* All of the associated I/O work is scheduled on background
* threads by {@link BackgroundJobHandler}.
* </p>
*/
@PluginFactory.LoadablePlugin
public class SharePlugin extends Plugin {
final static String ACTION_CANCEL_SHARE = "org.kde.kdeconnect.Plugins.SharePlugin.CancelShare";
final static String CANCEL_SHARE_DEVICE_ID_EXTRA = "deviceId";
final static String CANCEL_SHARE_BACKGROUND_JOB_ID_EXTRA = "backgroundJobId";
2018-10-26 23:53:58 +02:00
private final static String PACKET_TYPE_SHARE_REQUEST = "kdeconnect.share.request";
final static String PACKET_TYPE_SHARE_REQUEST_UPDATE = "kdeconnect.share.request.update";
final static String KEY_NUMBER_OF_FILES = "numberOfFiles";
final static String KEY_TOTAL_PAYLOAD_SIZE = "totalPayloadSize";
2023-03-05 22:03:58 +01:00
private final BackgroundJobHandler backgroundJobHandler;
private final Handler handler;
private CompositeReceiveFileJob receiveFileJob;
private CompositeUploadFileJob uploadFileJob;
private final Callback receiveFileJobCallback;
public SharePlugin() {
backgroundJobHandler = BackgroundJobHandler.newFixedThreadPoolBackgroundJobHander(5);
handler = new Handler(Looper.getMainLooper());
receiveFileJobCallback = new Callback();
}
2015-09-11 03:45:31 -07:00
@Override
protected int getOptionalPermissionExplanation() {
return R.string.share_optional_permission_explanation;
}
@Override
public @NonNull String getDisplayName() {
return context.getResources().getString(R.string.pref_plugin_sharereceiver);
}
@Override
public Drawable getIcon() {
return ContextCompat.getDrawable(context, R.drawable.share_plugin_action_24dp);
}
@Override
public @NonNull String getDescription() {
2015-04-12 00:11:30 -07:00
return context.getResources().getString(R.string.pref_plugin_sharereceiver_desc);
}
2015-08-20 00:59:21 -07:00
@Override
public boolean hasMainActivity(Context context) {
2015-08-20 00:59:21 -07:00
return true;
}
@Override
public @NonNull String getActionName() {
2015-08-20 00:59:21 -07:00
return context.getString(R.string.send_files);
}
@Override
public void startMainActivity(Activity parentActivity) {
Intent intent = new Intent(parentActivity, SendFileActivity.class);
intent.putExtra("deviceId", device.getDeviceId());
parentActivity.startActivity(intent);
2015-08-20 00:59:21 -07:00
}
@Override
2015-04-12 00:11:30 -07:00
public boolean hasSettings() {
return true;
}
@Override
@WorkerThread
public boolean onPacketReceived(@NonNull NetworkPacket np) {
try {
if (np.getType().equals(PACKET_TYPE_SHARE_REQUEST_UPDATE)) {
if (receiveFileJob != null && receiveFileJob.isRunning()) {
receiveFileJob.updateTotals(np.getInt(KEY_NUMBER_OF_FILES), np.getLong(KEY_TOTAL_PAYLOAD_SIZE));
} else {
Log.d("SharePlugin", "Received update packet but CompositeUploadJob is null or not running");
}
return true;
}
if (np.has("filename")) {
receiveFile(np);
} else if (np.has("text")) {
Log.i("SharePlugin", "hasText");
receiveText(np);
} else if (np.has("url")) {
receiveUrl(np);
} else {
Log.e("SharePlugin", "Error: Nothing attached!");
}
} catch (Exception e) {
Log.e("SharePlugin", "Exception");
e.printStackTrace();
}
return true;
}
private void receiveUrl(NetworkPacket np) {
String url = np.getString("url");
Log.i("SharePlugin", "hasUrl: " + url);
Intent browserIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
browserIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
IntentHelper.startActivityFromBackgroundOrCreateNotification(context, browserIntent, url);
}
private void receiveText(NetworkPacket np) {
String text = np.getString("text");
ClipboardManager cm = ContextCompat.getSystemService(context, ClipboardManager.class);
cm.setText(text);
handler.post(() -> Toast.makeText(context, R.string.shareplugin_text_saved, Toast.LENGTH_LONG).show());
}
@WorkerThread
private void receiveFile(NetworkPacket np) {
CompositeReceiveFileJob job;
boolean hasNumberOfFiles = np.has(KEY_NUMBER_OF_FILES);
boolean isOpen = np.getBoolean("open", false);
if (hasNumberOfFiles && !isOpen && receiveFileJob != null) {
job = receiveFileJob;
} else {
job = new CompositeReceiveFileJob(device, receiveFileJobCallback);
}
if (!hasNumberOfFiles) {
np.set(KEY_NUMBER_OF_FILES, 1);
np.set(KEY_TOTAL_PAYLOAD_SIZE, np.getPayloadSize());
}
job.addNetworkPacket(np);
if (job != receiveFileJob) {
if (hasNumberOfFiles && !isOpen) {
receiveFileJob = job;
}
backgroundJobHandler.runJob(job);
}
}
2016-09-27 20:26:57 +02:00
@Override
public PluginSettingsFragment getSettingsFragment(Activity activity) {
return ShareSettingsFragment.newInstance(getPluginKey(), R.xml.shareplugin_preferences);
2016-09-27 20:26:57 +02:00
}
void sendUriList(final ArrayList<Uri> uriList) {
2019-04-19 23:37:30 +02:00
CompositeUploadFileJob job;
if (uploadFileJob == null) {
job = new CompositeUploadFileJob(device, this.receiveFileJobCallback);
} else {
job = uploadFileJob;
}
//Read all the data early, as we only have permissions to do it while the activity is alive
for (Uri uri : uriList) {
NetworkPacket np = FilesHelper.uriToNetworkPacket(context, uri, PACKET_TYPE_SHARE_REQUEST);
if (np != null) {
job.addNetworkPacket(np);
}
}
if (job != uploadFileJob) {
uploadFileJob = job;
backgroundJobHandler.runJob(uploadFileJob);
}
}
public void share(Intent intent) {
Bundle extras = intent.getExtras();
ArrayList<Uri> streams = streamsFromIntent(intent, extras);
if (streams != null && !streams.isEmpty()) {
sendUriList(streams);
} else if (extras != null && extras.containsKey(Intent.EXTRA_TEXT)) {
Log.i("SharePlugin", "Intent contains text to share");
String text = extras.getString(Intent.EXTRA_TEXT);
String subject = extras.getString(Intent.EXTRA_SUBJECT);
//Hack: Detect shared youtube videos, so we can open them in the browser instead of as text
if (StringUtils.endsWith(subject, "YouTube")) {
int index = text.indexOf(": http://youtu.be/");
if (index > 0) {
text = text.substring(index + 2); //Skip ": "
}
}
boolean isUrl;
try {
new URL(text);
isUrl = true;
} catch (MalformedURLException e) {
isUrl = false;
}
NetworkPacket np = new NetworkPacket(SharePlugin.PACKET_TYPE_SHARE_REQUEST);
if (isUrl) {
np.set("url", text);
} else {
np.set("text", text);
}
device.sendPacket(np);
} else {
Log.e("SharePlugin", "There's nothing we know how to share");
}
}
private ArrayList<Uri> streamsFromIntent(Intent intent, Bundle extras) {
if (extras == null || !extras.containsKey(Intent.EXTRA_STREAM)) {
return null;
}
Log.i("SharePlugin", "Intent contains streams to share");
ArrayList<Uri> uriList;
if (Intent.ACTION_SEND_MULTIPLE.equals(intent.getAction())) {
uriList = intent.getParcelableArrayListExtra(Intent.EXTRA_STREAM);
} else {
uriList = new ArrayList<>();
uriList.add(extras.getParcelable(Intent.EXTRA_STREAM));
}
uriList.removeAll(Collections.singleton(null));
if (uriList.isEmpty()) {
Log.w("SharePlugin", "All streams were null");
}
return uriList;
}
2015-09-08 14:54:04 -07:00
@Override
public @NonNull String[] getSupportedPacketTypes() {
return new String[]{PACKET_TYPE_SHARE_REQUEST, PACKET_TYPE_SHARE_REQUEST_UPDATE};
2015-09-08 14:54:04 -07:00
}
@Override
public @NonNull String[] getOutgoingPacketTypes() {
return new String[]{PACKET_TYPE_SHARE_REQUEST};
2015-09-08 14:54:04 -07:00
}
@Override
public @NonNull String[] getOptionalPermissions() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
return ArrayUtils.EMPTY_STRING_ARRAY;
} else {
return new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE};
}
}
private class Callback implements BackgroundJob.Callback<Void> {
@Override
public void onResult(@NonNull BackgroundJob job, Void result) {
if (job == receiveFileJob) {
receiveFileJob = null;
} else if (job == uploadFileJob) {
uploadFileJob = null;
}
}
@Override
public void onError(@NonNull BackgroundJob job, @NonNull Throwable error) {
if (job == receiveFileJob) {
receiveFileJob = null;
} else if (job == uploadFileJob) {
uploadFileJob = null;
}
}
}
void cancelJob(long jobId) {
if (backgroundJobHandler.isRunning(jobId)) {
BackgroundJob job = backgroundJobHandler.getJob(jobId);
if (job != null) {
job.cancel();
if (job == receiveFileJob) {
receiveFileJob = null;
} else if (job == uploadFileJob) {
uploadFileJob = null;
}
}
}
}
}