2017-08-13 16:31:11 +02:00
|
|
|
/* Copyright (C) 2017 Andreas Shimokawa
|
|
|
|
|
|
|
|
This file is part of Gadgetbridge.
|
|
|
|
|
|
|
|
Gadgetbridge is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU Affero General Public License as published
|
|
|
|
by the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
Gadgetbridge is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU Affero General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
|
|
|
package nodomain.freeyourgadget.gadgetbridge.service.devices.amazfitbip;
|
|
|
|
|
2017-08-25 00:21:47 +02:00
|
|
|
import android.net.Uri;
|
|
|
|
import android.widget.Toast;
|
|
|
|
|
2017-08-21 23:47:47 +02:00
|
|
|
import org.slf4j.Logger;
|
|
|
|
import org.slf4j.LoggerFactory;
|
|
|
|
|
2017-08-19 20:58:13 +02:00
|
|
|
import java.io.IOException;
|
|
|
|
import java.nio.ByteBuffer;
|
|
|
|
import java.nio.ByteOrder;
|
|
|
|
|
2017-08-15 17:23:12 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.deviceevents.GBDeviceEventCallControl;
|
2017-08-21 23:47:47 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.devices.amazfitbip.AmazfitBipIcon;
|
2017-08-19 20:58:13 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.devices.amazfitbip.AmazfitBipService;
|
2017-08-19 23:49:19 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.devices.amazfitbip.AmazfitBipWeatherConditions;
|
2017-08-16 20:55:20 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.model.CallSpec;
|
2017-08-18 16:21:54 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.model.NotificationSpec;
|
|
|
|
import nodomain.freeyourgadget.gadgetbridge.model.NotificationType;
|
2017-08-19 20:58:13 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.model.WeatherSpec;
|
|
|
|
import nodomain.freeyourgadget.gadgetbridge.service.btle.TransactionBuilder;
|
2017-08-21 23:47:47 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.service.btle.profiles.alertnotification.AlertCategory;
|
|
|
|
import nodomain.freeyourgadget.gadgetbridge.service.btle.profiles.alertnotification.AlertNotificationProfile;
|
|
|
|
import nodomain.freeyourgadget.gadgetbridge.service.btle.profiles.alertnotification.NewAlert;
|
2017-08-25 00:21:47 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.amazfitbip.operations.AmazfitBipUpdateFirmwareOperation;
|
2017-08-13 16:31:11 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.miband.NotificationStrategy;
|
|
|
|
import nodomain.freeyourgadget.gadgetbridge.service.devices.miband2.MiBand2Support;
|
2017-08-25 00:21:47 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
2017-08-18 16:21:54 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.util.StringUtils;
|
2017-08-25 23:35:16 +02:00
|
|
|
import nodomain.freeyourgadget.gadgetbridge.util.Version;
|
2017-08-13 16:31:11 +02:00
|
|
|
|
|
|
|
public class AmazfitBipSupport extends MiBand2Support {
|
2017-08-21 23:47:47 +02:00
|
|
|
|
|
|
|
private static final Logger LOG = LoggerFactory.getLogger(AmazfitBipSupport.class);
|
|
|
|
|
2017-08-13 16:31:11 +02:00
|
|
|
@Override
|
|
|
|
public NotificationStrategy getNotificationStrategy() {
|
|
|
|
return new AmazfitBipTextNotificationStrategy(this);
|
|
|
|
}
|
|
|
|
|
2017-08-18 16:21:54 +02:00
|
|
|
@Override
|
|
|
|
public void onNotification(NotificationSpec notificationSpec) {
|
|
|
|
if (notificationSpec.type == NotificationType.GENERIC_ALARM_CLOCK) {
|
|
|
|
onAlarmClock(notificationSpec);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-08-20 23:58:41 +02:00
|
|
|
String senderOrTiltle = StringUtils.getFirstOf(notificationSpec.sender, notificationSpec.title);
|
2017-08-18 16:21:54 +02:00
|
|
|
|
2017-08-21 23:47:47 +02:00
|
|
|
String message = StringUtils.truncate(senderOrTiltle, 32) + "\0";
|
2017-08-20 23:58:41 +02:00
|
|
|
if (notificationSpec.subject != null) {
|
|
|
|
message += StringUtils.truncate(notificationSpec.subject, 128) + "\n\n";
|
2017-08-18 16:21:54 +02:00
|
|
|
}
|
2017-08-20 23:58:41 +02:00
|
|
|
if (notificationSpec.body != null) {
|
|
|
|
message += StringUtils.truncate(notificationSpec.body, 128);
|
|
|
|
}
|
|
|
|
|
2017-08-21 23:47:47 +02:00
|
|
|
try {
|
|
|
|
TransactionBuilder builder = performInitialized("new notification");
|
|
|
|
AlertNotificationProfile<?> profile = new AlertNotificationProfile(this);
|
|
|
|
profile.setMaxLength(255); // TODO: find out real limit, certainly it is more than 18 which is default
|
|
|
|
|
|
|
|
int customIconId = AmazfitBipIcon.mapToIconId(notificationSpec.type);
|
|
|
|
|
2017-08-23 23:12:44 +02:00
|
|
|
AlertCategory alertCategory = AlertCategory.CustomMiBand2;
|
2017-08-21 23:47:47 +02:00
|
|
|
|
|
|
|
// The SMS icon for AlertCategory.SMS is unique and not available as iconId
|
|
|
|
if (notificationSpec.type == NotificationType.GENERIC_SMS) {
|
|
|
|
alertCategory = AlertCategory.SMS;
|
|
|
|
}
|
|
|
|
|
|
|
|
NewAlert alert = new NewAlert(alertCategory, 1, message, customIconId);
|
|
|
|
profile.newAlert(builder, alert);
|
|
|
|
builder.queue(getQueue());
|
|
|
|
} catch (IOException ex) {
|
|
|
|
LOG.error("Unable to send notification to Amazfit Bip", ex);
|
|
|
|
}
|
2017-08-18 16:21:54 +02:00
|
|
|
}
|
|
|
|
|
2017-08-13 16:31:11 +02:00
|
|
|
@Override
|
|
|
|
public void onFindDevice(boolean start) {
|
2017-08-16 20:55:20 +02:00
|
|
|
CallSpec callSpec = new CallSpec();
|
|
|
|
callSpec.command = start ? CallSpec.CALL_INCOMING : CallSpec.CALL_END;
|
|
|
|
callSpec.name = "Gadgetbridge";
|
|
|
|
onSetCallState(callSpec);
|
2017-08-13 16:31:11 +02:00
|
|
|
}
|
2017-08-15 17:23:12 +02:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void handleButtonPressed(byte[] value) {
|
|
|
|
if (value == null || value.length != 1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
GBDeviceEventCallControl callCmd = new GBDeviceEventCallControl();
|
|
|
|
|
|
|
|
if (value[0] == 0x07) {
|
|
|
|
callCmd.event = GBDeviceEventCallControl.Event.REJECT;
|
|
|
|
} else if (value[0] == 0x09) {
|
|
|
|
callCmd.event = GBDeviceEventCallControl.Event.ACCEPT;
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
evaluateGBDeviceEvent(callCmd);
|
|
|
|
}
|
2017-08-19 20:58:13 +02:00
|
|
|
|
|
|
|
@Override
|
2017-08-25 00:21:47 +02:00
|
|
|
public void onInstallApp(Uri uri) {
|
|
|
|
try {
|
|
|
|
new AmazfitBipUpdateFirmwareOperation(uri, this).perform();
|
|
|
|
} catch (IOException ex) {
|
|
|
|
GB.toast(getContext(), "Firmware cannot be installed: " + ex.getMessage(), Toast.LENGTH_LONG, GB.ERROR, ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-19 20:58:13 +02:00
|
|
|
public void onSendWeather(WeatherSpec weatherSpec) {
|
|
|
|
try {
|
|
|
|
TransactionBuilder builder = performInitialized("Sending weather forecast");
|
2017-08-25 23:35:16 +02:00
|
|
|
Version version = new Version(gbDevice.getFirmwareVersion());
|
|
|
|
|
|
|
|
boolean supportsConditionString = false;
|
|
|
|
if (version.compareTo(new Version("0.0.8.74")) >= 0) {
|
|
|
|
supportsConditionString = true;
|
|
|
|
}
|
|
|
|
|
2017-08-19 20:58:13 +02:00
|
|
|
final byte NR_DAYS = 2;
|
2017-08-25 23:35:16 +02:00
|
|
|
int bytesPerDay = 4;
|
|
|
|
int conditionsLength = 0;
|
|
|
|
if (supportsConditionString) {
|
|
|
|
bytesPerDay = 5;
|
|
|
|
conditionsLength = weatherSpec.currentCondition.getBytes().length;
|
|
|
|
}
|
|
|
|
int length = 7 + bytesPerDay * NR_DAYS + conditionsLength;
|
|
|
|
ByteBuffer buf = ByteBuffer.allocate(length);
|
|
|
|
|
2017-08-19 20:58:13 +02:00
|
|
|
buf.order(ByteOrder.LITTLE_ENDIAN);
|
|
|
|
buf.put((byte) 1);
|
|
|
|
buf.putInt(weatherSpec.timestamp);
|
|
|
|
buf.put((byte) 0);
|
|
|
|
|
|
|
|
buf.put(NR_DAYS);
|
|
|
|
|
2017-08-19 23:49:19 +02:00
|
|
|
byte condition = AmazfitBipWeatherConditions.mapToAmazfitBipWeatherCode(weatherSpec.currentConditionCode);
|
|
|
|
buf.put(condition);
|
|
|
|
buf.put(condition);
|
2017-08-19 20:58:13 +02:00
|
|
|
buf.put((byte) (weatherSpec.todayMaxTemp - 273));
|
|
|
|
buf.put((byte) (weatherSpec.todayMinTemp - 273));
|
2017-08-25 23:35:16 +02:00
|
|
|
if (supportsConditionString) {
|
|
|
|
buf.put(weatherSpec.currentCondition.getBytes());
|
|
|
|
buf.put((byte) 0); //
|
|
|
|
}
|
2017-08-19 23:49:19 +02:00
|
|
|
condition = AmazfitBipWeatherConditions.mapToAmazfitBipWeatherCode(weatherSpec.tomorrowConditionCode);
|
|
|
|
|
|
|
|
buf.put(condition);
|
|
|
|
buf.put(condition);
|
2017-08-19 20:58:13 +02:00
|
|
|
buf.put((byte) (weatherSpec.tomorrowMaxTemp - 273));
|
|
|
|
buf.put((byte) (weatherSpec.tomorrowMinTemp - 273));
|
2017-08-25 23:35:16 +02:00
|
|
|
if (supportsConditionString) {
|
|
|
|
buf.put((byte) 0); // not yet in weatherspec
|
|
|
|
}
|
2017-08-19 20:58:13 +02:00
|
|
|
|
|
|
|
builder.write(getCharacteristic(AmazfitBipService.UUID_CHARACTERISTIC_WEATHER), buf.array());
|
|
|
|
builder.queue(getQueue());
|
|
|
|
} catch (IOException ignore) {
|
|
|
|
}
|
|
|
|
}
|
2017-08-13 16:31:11 +02:00
|
|
|
}
|