Как мне получить обслуживание для Android, чтобы работать, когда телефон находится в режиме ожидания / дремал / заблокирована
Добрый вечер,
Я уже несколько дней пытаюсь чего-то достичь и буквально не знаю, что еще попробовать, я в основном перепробовал все, что нашел в интернете, и это все еще не работает, но у меня есть чувство, что как только я найду решение, оно должно быть чем-то довольно простым.
Я работаю над большим проектом, но здесь я просто пытаюсь получить очень простой пример для работы, который я позже смогу адаптировать к своему проекту.
Что я хочу сделать
Я хочу, чтобы счетчик добавлял +1 Каждую секунду в течение 1200 секунд (20 минут) и записывал в файл каждый раз, когда он считает. В итоге я должен получить файл с 1200 строками и меткой времени на образец.
Как я пытаюсь это сделать
Я перепробовал миллион вещей, но здесь я просто вернулся к очень простому примеру, чтобы продемонстрировать его и попросить о помощи:
- У меня есть основная актичность с одной кнопкой запуска.
- При нажатии на кнопку я запускаю новый процесс из действия. Это процесс переднего плана.
- Я начинаю считать по кругу. Когда я достигаю 1200, отсчет прекращается.
Моя проблема
Пока экран телефона включен, все работает просто отлично, но как только я блокирую экран и кладу телефон в карман, в зависимости от телефона, на котором я тестирую, он начинает отказывать.
Чего бы мне хотелось
Если бы кто-нибудь мог сказать мне, что изменить / добавить / изменить или даже сделать это за меня, и написать ответ здесь (это действительно простой проект) Я был бы очень благодарен, я теряю кучу времени и просто не могу нажать нужную клавишу.
Я знаю, что прошу слишком многого, но сейчас я в полном отчаянии. Я был бы очень рад, если бы кто-нибудь смог придумать рабочее решение.
Большое спасибо.
Что я уже пробовал:
Поскольку это очень простой проект я просто скопирую 4 части из которых он состоит прямо здесь:
XML activity_main.xml
<?xml version="1.0" encoding="utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity"> <Button android:id="@+id/start_button" android:layout_width="150dp" android:layout_height="50dp" android:layout_marginStart="52dp" android:layout_marginBottom="116dp" android:onClick="startProcess" android:text="@string/button_label_start" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintStart_toStartOf="parent" /> <TextView android:id="@+id/counter_textView" android:layout_width="383dp" android:layout_height="412dp" android:gravity="center_horizontal|center_vertical" android:text="@string/counter_label_value" android:textSize="160sp" android:textStyle="bold" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintHorizontal_bias="0.428" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="parent" app:layout_constraintVertical_bias="0.169" /> </androidx.constraintlayout.widget.ConstraintLayout>
МАНИФЕСТ AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.helloworld"> <uses-permission android:name="android.permission.FOREGROUND_SERVICE" /> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:roundIcon="@mipmap/ic_launcher_round" android:supportsRtl="true" android:theme="@style/AppTheme"> <service android:name=".ForegroundService" android:enabled="true" android:exported="true" android:process=":externalProcess"> </service> <activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> </manifest>
ГЛАВНЫЙ MainActivity.java
package com.example.helloworld; import androidx.annotation.Nullable; import androidx.appcompat.app.AppCompatActivity; import androidx.core.content.ContextCompat; import android.app.PendingIntent; import android.content.Intent; import android.os.Bundle; import android.view.View; import android.widget.TextView; public class MainActivity extends AppCompatActivity { public static final String CHANNEL_ID = "ForegroundServiceChannel"; private TextView mShowCount; @Override protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) { super.onActivityResult(requestCode, resultCode, data); int mCount = data.getIntExtra(ForegroundService.FOREGROUND_MESSAGE, -1); if(mShowCount != null) mShowCount.setText(Integer.toString(mCount)); } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mShowCount = findViewById(R.id.counter_textView); } @Override protected void onResume() { super.onResume(); } @Override protected void onStart() { super.onStart(); } @Override protected void onPause() { super.onPause(); } @Override protected void onStop() { super.onStop(); } @Override protected void onDestroy() { super.onDestroy(); } @Override protected void onRestart() { super.onRestart(); } public void startProcess(View view) { PendingIntent pendingResult = createPendingResult(100, new Intent(), 0); Intent serviceIntent = new Intent(this, ForegroundService.class); serviceIntent.putExtra("pendingIntent", pendingResult); ContextCompat.startForegroundService(this, serviceIntent); } public void stopProcess(View view) { Intent serviceIntent = new Intent(this, ForegroundService.class); stopService(serviceIntent); } }
СЛУЖБА ПЕРЕДНЕГО ПЛАНА ForegroundService.java
package com.example.helloworld; import android.app.Notification; import android.app.NotificationChannel; import android.app.NotificationManager; import android.app.PendingIntent; import android.app.Service; import android.content.Intent; import android.os.Build; import android.os.IBinder; import android.os.SystemClock; import android.util.Log; import androidx.core.app.NotificationCompat; import java.io.File; import java.io.FileWriter; import java.util.Date; public class ForegroundService extends Service { public static final String CHANNEL_ID = "ForegroundServiceChannel"; public static final String FOREGROUND_MESSAGE = "com.example.helloworld.FOREGROUND_MESSAGE"; private PendingIntent data; private int mCount; private File basePath; public ForegroundService() { } @Override protected void finalize() throws Throwable { super.finalize(); } @Override public void onCreate() { super.onCreate(); } @Override public boolean stopService(Intent name) { return super.stopService(name); } @Override public int onStartCommand(Intent intent, int flags, int startId) { String input = intent.getStringExtra("inputExtra"); createNotificationChannel(); Intent notificationIntent = new Intent(this, MainActivity.class); PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0); Notification notification = new NotificationCompat.Builder(this, MainActivity.CHANNEL_ID) .setContentTitle("Foreground Service") .setContentText(input) .setContentIntent(pendingIntent) .build(); startForeground(1, notification); data = intent.getParcelableExtra("pendingIntent"); this.basePath = this.getExternalFilesDir("recs"); mCount = 0; new Thread(new Runnable() { public void run() { try { while(mCount < 1200) { Intent resultIntent = new Intent(); resultIntent.putExtra(FOREGROUND_MESSAGE, ++mCount); writeFile((new Date().getTime() / 1000) + " Increasing counter: " + mCount + "\n"); data.send(ForegroundService.this, 200, resultIntent); SystemClock.sleep(1000); } }catch (Exception ignored){} } }).start(); //stopSelf(); return START_NOT_STICKY; } private void createNotificationChannel() { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { NotificationChannel serviceChannel = new NotificationChannel( CHANNEL_ID, "Foreground Service Channel", NotificationManager.IMPORTANCE_DEFAULT ); NotificationManager manager = getSystemService(NotificationManager.class); manager.createNotificationChannel(serviceChannel); } } @Override public void onDestroy(){ super.onDestroy(); } @Override public IBinder onBind(Intent intent) { return null; } private void writeFile(String data) { File file = new File (basePath,"test"); if (!file.exists()) { boolean mkdirs = file.mkdirs(); if (!mkdirs) { Log.e("RECORDING", "Error creating SAVE BASE PATH"); } } try{ File counter_file = new File(file, "counter.txt"); FileWriter writer = new FileWriter(counter_file, true); writer.append(data); writer.flush(); writer.close(); } catch (Exception e){ e.printStackTrace(); } } }
Gerry Schmitz
Может быть, это "по замыслу"; удержание какого-то мошеннического приложения от высасывания всей энергии.
Louen
Я думал, что это может быть проблемой, но потом я понял, что у меня есть несколько приложений (от других разработчиков), которые делают именно то, что мне нужно... поэтому это должно быть возможно каким-то образом :(
0x01AA
Может быть, здесь вы найдете подсказку (см. Пример внизу для услуг): Обращение с выключенным экраном и экраном на намерения | думаю Андроид[^]