Частые вибрации носимых устройств с Ambient/Blackscreen

Я пытаюсь написать приложение для умных часов, которое имитирует ваше сердцебиение, поэтому оно должно иметь возможность вибрировать с интервалами времени от 1 до 0,5–0,3 с в течение 75 мс.

Это также необходимо сделать, когда приложение переходит в режим окружающей среды или начинает полностью черный экран. Меня не волнует энергопотребление или что-то еще, он должен просто работать 20-30 минут, больше ничего.

На Moto360, на котором я разрабатываю, режим ambient будет превращаться в полный черный экран, если не держать часы в определенном положении, и как только они войдут в черный экран, вибрация прекратится, поэтому мне нужно решение это также будет работать для черного экрана, если только не будет способа остановить часы, перейдя в черный экран, о котором я не знаю.

При просто работе с событиями onEnterAmbient и т.п. часы перестают вибрировать, когда не держишь под правильным углом из-за перехода в черный экран, а также вибрация не всегда постоянная, из-за необходимости вызывать vibrator.vibrate( ) повторно при вводе/обновлении и т. д., поэтому обычно это приводит к небольшой несогласованности в образце вибрации, чего следует избегать. Так что подход с использованием ambientmode, я думаю, не работает.

Вот прежний Кодекс:

[...]
private long[] vibrationPattern_StandardPulse = {0, 75, 1000};
private long[] vibrationPattern_AcceleratedPulse = {0, 75, 600};
[...]
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    setAmbientEnabled();
    mContainerView = (BoxInsetLayout) findViewById(R.id.container);
    vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);
[...]
@Override
public void onEnterAmbient(Bundle ambientDetails) {
    super.onEnterAmbient(ambientDetails);
    keepVibrating();
}

@Override
public void onUpdateAmbient() {
    super.onUpdateAmbient();
    keepVibrating();
}

@Override
public void onExitAmbient() {
    super.onExitAmbient();
    keepVibrating();
[...]
 public void keepVibrating() {
    if (standardPulse) {
        vibrator.vibrate(vibrationPattern_StandardPulse, 0);
    } else {
        vibrator.vibrate(vibrationPattern_AcceleratedPulse, 0);
    }
}
[...]

Поэтому я попробовал еще раз с помощью аварийного менеджера (придерживаясь этой ссылки: https://developer.android.com/training/wearables/apps/always-on.html#UpdateContent), чтобы иметь возможность также отправлять вибрации в режиме Blackscreen, но, как оказалось, таймер для диспетчера будильника нельзя ставить такие маленькие интервалы? Он вибрирует только каждые 5 секунд, как предполагается, 1 секунду и меньше, независимо от того, находится ли он на черном экране, в обычном или окружающем режиме ...

Код:

protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    setAmbientEnabled();
    mAmbientStateAlarmManager =
            (AlarmManager) getSystemService(Context.ALARM_SERVICE);
    Intent ambientStateIntent =
            new Intent(getApplicationContext(), MainActivity.class);

    mAmbientStatePendingIntent = PendingIntent.getActivity(
            getApplicationContext(),
            0,
            ambientStateIntent,
            PendingIntent.FLAG_UPDATE_CURRENT);

    mContainerView = (BoxInsetLayout) findViewById(R.id.container);
    //mClockView = (TextView) findViewById(R.id.clock);
    vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);
[...]}
@Override
public void onNewIntent(Intent intent) {
    super.onNewIntent(intent);

    setIntent(intent);

    refreshDisplayAndSetNextUpdate();
}

private long AMBIENT_INTERVAL_MS = 0;

private void refreshDisplayAndSetNextUpdate() {
    if (standardPulse) {
        AMBIENT_INTERVAL_MS = 1000;
    } else {
        AMBIENT_INTERVAL_MS = 600;
    }

    if (isAmbient()) {
        // Implement data retrieval and update the screen for ambient mode
        vibrator.vibrate(75);
    } else {
        // Implement data retrieval and update the screen for interactive mode
        vibrator.vibrate(75);
    }

    long timeMs = System.currentTimeMillis();

    // Schedule a new alarm
    if (isAmbient()) {
        // Calculate the next trigger time
        long triggerTimeMs = timeMs + AMBIENT_INTERVAL_MS;

        mAmbientStateAlarmManager.setExact(
                AlarmManager.RTC_WAKEUP,
                triggerTimeMs,
                mAmbientStatePendingIntent);

    } else {
        // Calculate the next trigger time for interactive mode
        long triggerTimeMs = timeMs + AMBIENT_INTERVAL_MS;

        mAmbientStateAlarmManager.setExact(
                AlarmManager.RTC_WAKEUP,
                triggerTimeMs,
                mAmbientStatePendingIntent);
    }
}

Затем RefreshUpdateAndSetNextAlarm также вызывается в событиях onEnterAmbient и т. д., но не работает должным образом. Есть ли способ делать такие вещи на умных часах, особенно на Moto 360?

Может быть, если я буду постоянно отправлять сообщения со смартфона через messageAPI и установить их как срочные?


person Alex_W    schedule 04.11.2016    source источник


Ответы (1)


Решение заключалось в использовании CPU Wakelock:

PowerManager powerManager = (PowerManager) getSystemService(POWER_SERVICE);
PowerManager.WakeLock wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK,
            "MyWakelockTag");
wakeLock.acquire();

и экран держать на флаге:

getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
person Alex_W    schedule 14.11.2016