问题:
在Android中,似乎没有一种方法可以在DCIM文件夹中指定要同步到云的单个文件夹。 Facebook应用程序将图像保存到DCIM / Facebook,因此我保存的所有图像都会自动同步到云中。我可以通过在文件夹中放置.NOMEDIA文件来解决此问题,但这现在意味着我无法进入文件浏览器才能浏览文件,因此,例如,我无法轻松地通过邮件发送这些图像。
目前,我正在开发一个Android应用程序,用于将.NOMDEIA文件以外的文件从DCIM / Facebook自动移动到Pictures / Facebook。我目前在MainActivity类中有执行此操作的代码,并且希望将其转换为将自动运行的服务。
简而言之,我希望该服务在启动时启动,并且始终在不显示活动的情况下运行,因此可以减少我的任何手动工作。再说一次,我不想手动启动任何东西,而且我不需要任何活动,因为我不需要任何用户交互。我已经阅读了许多页面,迷失在文档中,并且去过有关Service,IntentService,JobScheduler等的看似无数的帖子,而我目前正在努力实现和正确使用方法。
对于每几秒钟运行一次的JobSchedule,这是否是一个好案例?它应该是在一个循环中一次又一次地循环运行代码的Service,每次迭代等待几秒钟吗?每当在文件夹中添加内容时,是否可以触发服务?
如上所述,我一直在努力实现。在这一点上,我唯一要做的就是从MainActivity内调用服务,但这是在活动短暂打开和关闭时发生的。这是不希望的行为。此时情况变得有些混乱,因此我选择不包含任何代码,以希望有人可以将我指向正确的文档/教程或提供可以使用的适当框架。我宁可不要一开始就弄混水域。
重要说明:我所进行的活动具有请求READ_EXTERNAL_STORAGE的权限(据我了解,您不需要请求WRITE_EXTERNAL_STORAGE的权限;我的代码无需请求该权限即可工作),但我也不确定如何确保这一点对服务已完成,因为请求权限的方法似乎希望将活动作为参数之一。
编辑:为接收者和服务添加清单和代码
FacebookImageListener.java
public class FacebookImageListener extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
context.startService(new Intent(context, FacebookImageMoverService.class));
Log.i("FacebookImageMoverService", "started");
}
}
FacebookImageMoverService.java
public class FacebookImageMoverService extends Service {
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
while (true) {
//TODO remove this
Log.i("imagemover", "DELETE THIS");
// String path = this.getExternalFilesDir(Environment.DIRECTORY_DCIM).getAbsolutePath() + "/Facebook";
String path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM).getAbsolutePath() + "/Facebook";
Log.d("Files", "Path: " + path);
File directory = new File(path);
File[] files = directory.listFiles();
Log.d("Files", "Size: " + files.length);
for (int i = 0; i < files.length; i++) {
Toast toast = Toast.makeText(this.getApplicationContext(), "FileName:" + files[i].getName(), Toast.LENGTH_LONG);
toast.show();
Log.d("Files", "FileName:" + files[i].getName());
if (!files[i].getName().equals(".NOMEDIA")) {
// moveFile(path + "/", files[i].getName(), this.getExternalFilesDir(Environment.DIRECTORY_PICTURES).getAbsolutePath() + "/Facebook/");
moveFile(path + "/", files[i].getName(), Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES).getAbsolutePath() + "/Facebook/");
}
}
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
//TODO check this out
// return Service.START_NOT_STICKY;
}
@Override
public IBinder onBind(Intent intent) {
//TODO for communication return IBinder implementation
return null;
}
private void moveFile(String inputPath, String inputFile, String outputPath) {
InputStream in = null;
OutputStream out = null;
try {
//create output directory if it doesn't exist
File dir = new File(outputPath);
if (!dir.exists()) {
dir.mkdirs();
}
in = new FileInputStream(inputPath + inputFile);
out = new FileOutputStream(outputPath + inputFile);
byte[] buffer = new byte[1024];
int read;
while ((read = in.read(buffer)) != -1) {
out.write(buffer, 0, read);
}
in.close();
in = null;
// write the output file
out.flush();
out.close();
out = null;
// delete the original file
new File(inputPath + inputFile).delete();
} catch (FileNotFoundException fnfe1) {
Log.e("tag", fnfe1.getMessage());
} catch (Exception e) {
Log.e("tag", e.getMessage());
}
}
}
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="shogg.facebookimagemover"
>
<!--<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />-->
<!--<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />-->
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<application android:label="@string/app_name"
android:icon="@mipmap/ic_launcher"
android:roundIcon="@mipmap/ic_launcher_round">
<receiver android:name=".FacebookImageListener"
android:icon="@mipmap/ic_launcher"
android:exported="true"
android:permission="android.permission.RECEIVE_BOOT_COMPLETED">
<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED" />
<action android:name="android.intent.action.QUICKBOOT_POWERON" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</receiver>
<service android:enabled="true" android:name=".FacebookImageMoverService" />
</application>
</manifest>
答案 0 :(得分:1)
在您的BroadcastReceiver
中注册一个AndroidManifest.xml
,并通过向清单中添加此权限来请求侦听设备引导广播的权限:
android:permission="android.permission.RECEIVE_BOOT_COMPLETED">
然后,您可以使用onReceive()
的{{1}}方法启动服务,而无需启动应用程序的主要活动(以任何方式阻碍用户)。