Merge branch 'dev' into dev
This commit is contained in:
commit
4a481bf294
35 changed files with 655 additions and 1246 deletions
Binary file not shown.
|
@ -32,11 +32,10 @@
|
||||||
android:icon="${appIcon}"
|
android:icon="${appIcon}"
|
||||||
android:label="@string/app_name"
|
android:label="@string/app_name"
|
||||||
android:supportsRtl="true"
|
android:supportsRtl="true"
|
||||||
android:theme="@style/AppTheme">
|
android:theme="@style/AppTheme.NoActionBar">
|
||||||
<activity android:name=".MainActivity">
|
<activity android:name=".MainActivity">
|
||||||
<intent-filter>
|
<intent-filter>
|
||||||
<action android:name="android.intent.action.MAIN" />
|
<action android:name="android.intent.action.MAIN" />
|
||||||
|
|
||||||
<category android:name="android.intent.category.LAUNCHER" />
|
<category android:name="android.intent.category.LAUNCHER" />
|
||||||
</intent-filter>
|
</intent-filter>
|
||||||
</activity>
|
</activity>
|
||||||
|
@ -172,7 +171,10 @@
|
||||||
android:name=".setupwizard.SetupWizardActivity"
|
android:name=".setupwizard.SetupWizardActivity"
|
||||||
android:configChanges="orientation|keyboardHidden|screenSize"
|
android:configChanges="orientation|keyboardHidden|screenSize"
|
||||||
android:theme="@style/AppTheme.SetupWizard"
|
android:theme="@style/AppTheme.SetupWizard"
|
||||||
android:label="@string/title_activity_setup_wizard"></activity>
|
android:label="@string/title_activity_setup_wizard" />
|
||||||
|
|
||||||
|
<activity android:name=".SingleFragmentActivity"
|
||||||
|
android:theme="@style/AppTheme" />
|
||||||
</application>
|
</application>
|
||||||
|
|
||||||
</manifest>
|
</manifest>
|
|
@ -1,30 +1,35 @@
|
||||||
package info.nightscout.androidaps;
|
package info.nightscout.androidaps;
|
||||||
|
|
||||||
import android.app.Activity;
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.DialogInterface;
|
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.pm.PackageManager;
|
import android.content.pm.PackageManager;
|
||||||
import android.graphics.Rect;
|
import android.graphics.Rect;
|
||||||
import android.os.Build;
|
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
|
import android.os.PersistableBundle;
|
||||||
import android.os.PowerManager;
|
import android.os.PowerManager;
|
||||||
|
import android.support.annotation.Nullable;
|
||||||
|
import android.support.design.widget.NavigationView;
|
||||||
|
import android.support.design.widget.TabLayout;
|
||||||
import android.support.v4.app.ActivityCompat;
|
import android.support.v4.app.ActivityCompat;
|
||||||
import android.support.v4.view.ViewPager;
|
import android.support.v4.view.ViewPager;
|
||||||
|
import android.support.v4.widget.DrawerLayout;
|
||||||
|
import android.support.v7.app.ActionBarDrawerToggle;
|
||||||
import android.support.v7.app.AlertDialog;
|
import android.support.v7.app.AlertDialog;
|
||||||
import android.support.v7.app.AppCompatActivity;
|
import android.support.v7.app.AppCompatActivity;
|
||||||
import android.support.v7.widget.PopupMenu;
|
import android.support.v7.widget.Toolbar;
|
||||||
import android.text.SpannableString;
|
import android.text.SpannableString;
|
||||||
import android.text.method.LinkMovementMethod;
|
import android.text.method.LinkMovementMethod;
|
||||||
import android.text.util.Linkify;
|
import android.text.util.Linkify;
|
||||||
import android.view.MenuInflater;
|
import android.util.Log;
|
||||||
|
import android.util.TypedValue;
|
||||||
|
import android.view.Menu;
|
||||||
import android.view.MenuItem;
|
import android.view.MenuItem;
|
||||||
import android.view.MotionEvent;
|
import android.view.MotionEvent;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.view.WindowManager;
|
import android.view.WindowManager;
|
||||||
import android.view.inputmethod.InputMethodManager;
|
import android.view.inputmethod.InputMethodManager;
|
||||||
import android.widget.EditText;
|
import android.widget.EditText;
|
||||||
import android.widget.ImageButton;
|
import android.widget.LinearLayout;
|
||||||
import android.widget.TextView;
|
import android.widget.TextView;
|
||||||
|
|
||||||
import com.joanzapata.iconify.Iconify;
|
import com.joanzapata.iconify.Iconify;
|
||||||
|
@ -37,7 +42,6 @@ import org.slf4j.LoggerFactory;
|
||||||
import info.nightscout.androidaps.data.Profile;
|
import info.nightscout.androidaps.data.Profile;
|
||||||
import info.nightscout.androidaps.events.EventAppExit;
|
import info.nightscout.androidaps.events.EventAppExit;
|
||||||
import info.nightscout.androidaps.events.EventFeatureRunning;
|
import info.nightscout.androidaps.events.EventFeatureRunning;
|
||||||
import info.nightscout.androidaps.events.EventPreferenceChange;
|
|
||||||
import info.nightscout.androidaps.events.EventRefreshGui;
|
import info.nightscout.androidaps.events.EventRefreshGui;
|
||||||
import info.nightscout.androidaps.interfaces.PluginBase;
|
import info.nightscout.androidaps.interfaces.PluginBase;
|
||||||
import info.nightscout.androidaps.plugins.ConfigBuilder.ConfigBuilderPlugin;
|
import info.nightscout.androidaps.plugins.ConfigBuilder.ConfigBuilderPlugin;
|
||||||
|
@ -45,7 +49,6 @@ import info.nightscout.androidaps.plugins.Food.FoodPlugin;
|
||||||
import info.nightscout.androidaps.plugins.Overview.events.EventSetWakeLock;
|
import info.nightscout.androidaps.plugins.Overview.events.EventSetWakeLock;
|
||||||
import info.nightscout.androidaps.plugins.Treatments.TreatmentsPlugin;
|
import info.nightscout.androidaps.plugins.Treatments.TreatmentsPlugin;
|
||||||
import info.nightscout.androidaps.setupwizard.SetupWizardActivity;
|
import info.nightscout.androidaps.setupwizard.SetupWizardActivity;
|
||||||
import info.nightscout.androidaps.tabs.SlidingTabLayout;
|
|
||||||
import info.nightscout.androidaps.tabs.TabPageAdapter;
|
import info.nightscout.androidaps.tabs.TabPageAdapter;
|
||||||
import info.nightscout.utils.AndroidPermission;
|
import info.nightscout.utils.AndroidPermission;
|
||||||
import info.nightscout.utils.ImportExportPrefs;
|
import info.nightscout.utils.ImportExportPrefs;
|
||||||
|
@ -55,13 +58,15 @@ import info.nightscout.utils.OKDialog;
|
||||||
import info.nightscout.utils.PasswordProtection;
|
import info.nightscout.utils.PasswordProtection;
|
||||||
import info.nightscout.utils.SP;
|
import info.nightscout.utils.SP;
|
||||||
|
|
||||||
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
|
public class MainActivity extends AppCompatActivity {
|
||||||
private static Logger log = LoggerFactory.getLogger(MainActivity.class);
|
private static Logger log = LoggerFactory.getLogger(MainActivity.class);
|
||||||
|
|
||||||
ImageButton menuButton;
|
|
||||||
|
|
||||||
protected PowerManager.WakeLock mWakeLock;
|
protected PowerManager.WakeLock mWakeLock;
|
||||||
|
|
||||||
|
private ActionBarDrawerToggle actionBarDrawerToggle;
|
||||||
|
|
||||||
|
private MenuItem pluginPreferencesMenuItem;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onCreate(Bundle savedInstanceState) {
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
|
@ -71,16 +76,54 @@ public class MainActivity extends AppCompatActivity implements View.OnClickListe
|
||||||
|
|
||||||
Iconify.with(new FontAwesomeModule());
|
Iconify.with(new FontAwesomeModule());
|
||||||
LocaleHelper.onCreate(this, "en");
|
LocaleHelper.onCreate(this, "en");
|
||||||
|
|
||||||
setContentView(R.layout.activity_main);
|
setContentView(R.layout.activity_main);
|
||||||
menuButton = (ImageButton) findViewById(R.id.overview_menuButton);
|
setSupportActionBar(findViewById(R.id.toolbar));
|
||||||
menuButton.setOnClickListener(this);
|
getSupportActionBar().setDisplayShowTitleEnabled(false);
|
||||||
|
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
|
||||||
|
getSupportActionBar().setHomeButtonEnabled(true);
|
||||||
|
|
||||||
|
DrawerLayout drawerLayout = findViewById(R.id.drawer_layout);
|
||||||
|
actionBarDrawerToggle = new ActionBarDrawerToggle(this, drawerLayout, R.string.open_navigation, R.string.close_navigation);
|
||||||
|
drawerLayout.addDrawerListener(actionBarDrawerToggle);
|
||||||
|
actionBarDrawerToggle.syncState();
|
||||||
|
|
||||||
onStatusEvent(new EventSetWakeLock(SP.getBoolean("lockscreen", false)));
|
onStatusEvent(new EventSetWakeLock(SP.getBoolean("lockscreen", false)));
|
||||||
|
|
||||||
doMigrations();
|
doMigrations();
|
||||||
|
|
||||||
registerBus();
|
registerBus();
|
||||||
setUpTabs(false);
|
setupTabs();
|
||||||
|
setupViews(false);
|
||||||
|
|
||||||
|
final ViewPager viewPager = findViewById(R.id.pager);
|
||||||
|
viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
|
||||||
|
@Override
|
||||||
|
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onPageSelected(int position) {
|
||||||
|
checkPluginPreferences(viewPager);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onPageScrollStateChanged(int state) {
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void checkPluginPreferences(ViewPager viewPager) {
|
||||||
|
if (pluginPreferencesMenuItem == null) return;
|
||||||
|
if (((TabPageAdapter) viewPager.getAdapter()).getPluginAt(viewPager.getCurrentItem()).getPreferencesId() != -1)
|
||||||
|
pluginPreferencesMenuItem.setEnabled(true);
|
||||||
|
else pluginPreferencesMenuItem.setEnabled(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onPostCreate(@Nullable Bundle savedInstanceState, @Nullable PersistableBundle persistentState) {
|
||||||
|
super.onPostCreate(savedInstanceState, persistentState);
|
||||||
|
actionBarDrawerToggle.syncState();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -127,39 +170,74 @@ public class MainActivity extends AppCompatActivity implements View.OnClickListe
|
||||||
public void onStatusEvent(final EventRefreshGui ev) {
|
public void onStatusEvent(final EventRefreshGui ev) {
|
||||||
String lang = SP.getString("language", "en");
|
String lang = SP.getString("language", "en");
|
||||||
LocaleHelper.setLocale(getApplicationContext(), lang);
|
LocaleHelper.setLocale(getApplicationContext(), lang);
|
||||||
runOnUiThread(new Runnable() {
|
runOnUiThread(() -> {
|
||||||
@Override
|
if (ev.recreate) {
|
||||||
public void run() {
|
recreate();
|
||||||
if (ev.recreate) {
|
} else {
|
||||||
recreate();
|
try { // activity may be destroyed
|
||||||
} else {
|
setupTabs();
|
||||||
try { // activity may be destroyed
|
setupViews(true);
|
||||||
setUpTabs(true);
|
} catch (IllegalStateException e) {
|
||||||
} catch (IllegalStateException e) {
|
log.error("Unhandled exception", e);
|
||||||
log.error("Unhandled exception", e);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean lockScreen = BuildConfig.NSCLIENTOLNY && SP.getBoolean("lockscreen", false);
|
|
||||||
if (lockScreen)
|
|
||||||
getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
|
|
||||||
else
|
|
||||||
getWindow().clearFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
boolean lockScreen = BuildConfig.NSCLIENTOLNY && SP.getBoolean("lockscreen", false);
|
||||||
|
if (lockScreen)
|
||||||
|
getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
|
||||||
|
else
|
||||||
|
getWindow().clearFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void setUpTabs(boolean switchToLast) {
|
private void setupViews(boolean switchToLast) {
|
||||||
TabPageAdapter pageAdapter = new TabPageAdapter(getSupportFragmentManager(), this);
|
TabPageAdapter pageAdapter = new TabPageAdapter(getSupportFragmentManager(), this);
|
||||||
|
NavigationView navigationView = findViewById(R.id.navigation_view);
|
||||||
|
navigationView.setNavigationItemSelectedListener(menuItem -> {
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
Menu menu = navigationView.getMenu();
|
||||||
|
menu.clear();
|
||||||
for (PluginBase p : MainApp.getPluginsList()) {
|
for (PluginBase p : MainApp.getPluginsList()) {
|
||||||
pageAdapter.registerNewFragment(p);
|
pageAdapter.registerNewFragment(p);
|
||||||
|
if (p.hasFragment() && !p.isFragmentVisible() && p.isEnabled(p.pluginDescription.getType())) {
|
||||||
|
MenuItem menuItem = menu.add(p.getName());
|
||||||
|
menuItem.setCheckable(true);
|
||||||
|
menuItem.setOnMenuItemClickListener(item -> {
|
||||||
|
Intent intent = new Intent(this, SingleFragmentActivity.class);
|
||||||
|
intent.putExtra("plugin", MainApp.getPluginsList().indexOf(p));
|
||||||
|
startActivity(intent);
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
ViewPager mPager = (ViewPager) findViewById(R.id.pager);
|
ViewPager mPager = findViewById(R.id.pager);
|
||||||
mPager.setAdapter(pageAdapter);
|
mPager.setAdapter(pageAdapter);
|
||||||
SlidingTabLayout mTabs = (SlidingTabLayout) findViewById(R.id.tabs);
|
|
||||||
mTabs.setViewPager(mPager);
|
|
||||||
if (switchToLast)
|
if (switchToLast)
|
||||||
mPager.setCurrentItem(pageAdapter.getCount() - 1, false);
|
mPager.setCurrentItem(pageAdapter.getCount() - 1, false);
|
||||||
|
checkPluginPreferences(mPager);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setupTabs() {
|
||||||
|
ViewPager viewPager = findViewById(R.id.pager);
|
||||||
|
TabLayout normalTabs = findViewById(R.id.tabs_normal);
|
||||||
|
normalTabs.setupWithViewPager(viewPager, true);
|
||||||
|
TabLayout compactTabs = findViewById(R.id.tabs_compact);
|
||||||
|
compactTabs.setupWithViewPager(viewPager, true);
|
||||||
|
Toolbar toolbar = findViewById(R.id.toolbar);
|
||||||
|
if (SP.getBoolean("short_tabtitles", false)) {
|
||||||
|
normalTabs.setVisibility(View.GONE);
|
||||||
|
compactTabs.setVisibility(View.VISIBLE);
|
||||||
|
toolbar.setLayoutParams(new LinearLayout.LayoutParams(Toolbar.LayoutParams.MATCH_PARENT, (int) getResources().getDimension(R.dimen.compact_height)));
|
||||||
|
} else {
|
||||||
|
normalTabs.setVisibility(View.VISIBLE);
|
||||||
|
compactTabs.setVisibility(View.GONE);
|
||||||
|
TypedValue typedValue = new TypedValue();
|
||||||
|
if (getTheme().resolveAttribute(R.attr.actionBarSize, typedValue, true)) {
|
||||||
|
toolbar.setLayoutParams(new LinearLayout.LayoutParams(Toolbar.LayoutParams.MATCH_PARENT,
|
||||||
|
TypedValue.complexToDimensionPixelSize(typedValue.data, getResources().getDisplayMetrics())));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void registerBus() {
|
private void registerBus() {
|
||||||
|
@ -257,99 +335,96 @@ public class MainActivity extends AppCompatActivity implements View.OnClickListe
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onClick(final View v) {
|
public boolean onCreateOptionsMenu(Menu menu) {
|
||||||
final Activity activity = this;
|
getMenuInflater().inflate(R.menu.menu_main, menu);
|
||||||
switch (v.getId()) {
|
pluginPreferencesMenuItem = menu.findItem(R.id.nav_plugin_preferences);
|
||||||
case R.id.overview_menuButton:
|
checkPluginPreferences(findViewById(R.id.pager));
|
||||||
PopupMenu popup = new PopupMenu(v.getContext(), v);
|
return true;
|
||||||
MenuInflater inflater = popup.getMenuInflater();
|
}
|
||||||
inflater.inflate(R.menu.menu_main, popup.getMenu());
|
|
||||||
popup.setOnMenuItemClickListener(new PopupMenu.OnMenuItemClickListener() {
|
@Override
|
||||||
@Override
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
public boolean onMenuItemClick(MenuItem item) {
|
int id = item.getItemId();
|
||||||
int id = item.getItemId();
|
switch (id) {
|
||||||
switch (id) {
|
case R.id.nav_preferences:
|
||||||
case R.id.nav_preferences:
|
PasswordProtection.QueryPassword(this, R.string.settings_password, "settings_password", () -> {
|
||||||
PasswordProtection.QueryPassword(v.getContext(), R.string.settings_password, "settings_password", new Runnable() {
|
Intent i = new Intent(this, PreferencesActivity.class);
|
||||||
@Override
|
i.putExtra("id", -1);
|
||||||
public void run() {
|
startActivity(i);
|
||||||
Intent i = new Intent(v.getContext(), PreferencesActivity.class);
|
}, null);
|
||||||
i.putExtra("id", -1);
|
return true;
|
||||||
startActivity(i);
|
case R.id.nav_historybrowser:
|
||||||
}
|
startActivity(new Intent(this, HistoryBrowseActivity.class));
|
||||||
}, null);
|
return true;
|
||||||
break;
|
case R.id.nav_setupwizard:
|
||||||
case R.id.nav_historybrowser:
|
startActivity(new Intent(this, SetupWizardActivity.class));
|
||||||
startActivity(new Intent(v.getContext(), HistoryBrowseActivity.class));
|
return true;
|
||||||
break;
|
case R.id.nav_resetdb:
|
||||||
case R.id.nav_setupwizard:
|
new AlertDialog.Builder(this)
|
||||||
startActivity(new Intent(v.getContext(), SetupWizardActivity.class));
|
.setTitle(R.string.nav_resetdb)
|
||||||
break;
|
.setMessage(R.string.reset_db_confirm)
|
||||||
case R.id.nav_resetdb:
|
.setNegativeButton(android.R.string.cancel, null)
|
||||||
new AlertDialog.Builder(v.getContext())
|
.setPositiveButton(android.R.string.ok, (dialog, which) -> {
|
||||||
.setTitle(R.string.nav_resetdb)
|
MainApp.getDbHelper().resetDatabases();
|
||||||
.setMessage(R.string.reset_db_confirm)
|
// should be handled by Plugin-Interface and
|
||||||
.setNegativeButton(android.R.string.cancel, null)
|
// additional service interface and plugin registry
|
||||||
.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
|
FoodPlugin.getPlugin().getService().resetFood();
|
||||||
@Override
|
TreatmentsPlugin.getPlugin().getService().resetTreatments();
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
})
|
||||||
MainApp.getDbHelper().resetDatabases();
|
.create()
|
||||||
// should be handled by Plugin-Interface and
|
.show();
|
||||||
// additional service interface and plugin registry
|
return true;
|
||||||
FoodPlugin.getPlugin().getService().resetFood();
|
case R.id.nav_export:
|
||||||
TreatmentsPlugin.getPlugin().getService().resetTreatments();
|
ImportExportPrefs.verifyStoragePermissions(this);
|
||||||
}
|
ImportExportPrefs.exportSharedPreferences(this);
|
||||||
})
|
return true;
|
||||||
.create()
|
case R.id.nav_import:
|
||||||
.show();
|
ImportExportPrefs.verifyStoragePermissions(this);
|
||||||
break;
|
ImportExportPrefs.importSharedPreferences(this);
|
||||||
case R.id.nav_export:
|
return true;
|
||||||
ImportExportPrefs.verifyStoragePermissions(activity);
|
case R.id.nav_show_logcat:
|
||||||
ImportExportPrefs.exportSharedPreferences(activity);
|
LogDialog.showLogcat(this);
|
||||||
break;
|
return true;
|
||||||
case R.id.nav_import:
|
case R.id.nav_about:
|
||||||
ImportExportPrefs.verifyStoragePermissions(activity);
|
AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
||||||
ImportExportPrefs.importSharedPreferences(activity);
|
builder.setTitle(MainApp.gs(R.string.app_name) + " " + BuildConfig.VERSION);
|
||||||
break;
|
if (Config.NSCLIENT || Config.G5UPLOADER)
|
||||||
case R.id.nav_show_logcat:
|
builder.setIcon(R.mipmap.yellowowl);
|
||||||
LogDialog.showLogcat(v.getContext());
|
else
|
||||||
break;
|
builder.setIcon(R.mipmap.blueowl);
|
||||||
case R.id.nav_about:
|
String message = "Build: " + BuildConfig.BUILDVERSION + "\n";
|
||||||
AlertDialog.Builder builder = new AlertDialog.Builder(v.getContext());
|
message += "Flavor: " + BuildConfig.FLAVOR + BuildConfig.BUILD_TYPE + "\n";
|
||||||
builder.setTitle(MainApp.gs(R.string.app_name) + " " + BuildConfig.VERSION);
|
message += MainApp.gs(R.string.configbuilder_nightscoutversion_label) + " " + ConfigBuilderPlugin.nightscoutVersionName;
|
||||||
if (Config.NSCLIENT || Config.G5UPLOADER)
|
if (MainApp.engineeringMode)
|
||||||
builder.setIcon(R.mipmap.yellowowl);
|
message += "\n" + MainApp.gs(R.string.engineering_mode_enabled);
|
||||||
else
|
message += MainApp.gs(R.string.about_link_urls);
|
||||||
builder.setIcon(R.mipmap.blueowl);
|
final SpannableString messageSpanned = new SpannableString(message);
|
||||||
String message = "Build: " + BuildConfig.BUILDVERSION + "\n";
|
Linkify.addLinks(messageSpanned, Linkify.WEB_URLS);
|
||||||
message += "Flavor: " + BuildConfig.FLAVOR + BuildConfig.BUILD_TYPE + "\n";
|
builder.setMessage(messageSpanned);
|
||||||
message += MainApp.gs(R.string.configbuilder_nightscoutversion_label) + " " + ConfigBuilderPlugin.nightscoutVersionName;
|
builder.setPositiveButton(MainApp.gs(R.string.ok), null);
|
||||||
if (MainApp.engineeringMode)
|
AlertDialog alertDialog = builder.create();
|
||||||
message += "\n" + MainApp.gs(R.string.engineering_mode_enabled);
|
alertDialog.show();
|
||||||
message += MainApp.gs(R.string.about_link_urls);
|
((TextView) alertDialog.findViewById(android.R.id.message)).setMovementMethod(LinkMovementMethod.getInstance());
|
||||||
final SpannableString messageSpanned = new SpannableString(message);
|
return true;
|
||||||
Linkify.addLinks(messageSpanned, Linkify.WEB_URLS);
|
case R.id.nav_exit:
|
||||||
builder.setMessage(messageSpanned);
|
log.debug("Exiting");
|
||||||
builder.setPositiveButton(MainApp.gs(R.string.ok), null);
|
MainApp.instance().stopKeepAliveService();
|
||||||
AlertDialog alertDialog = builder.create();
|
MainApp.bus().post(new EventAppExit());
|
||||||
alertDialog.show();
|
MainApp.closeDbHelper();
|
||||||
((TextView) alertDialog.findViewById(android.R.id.message)).setMovementMethod(LinkMovementMethod.getInstance());
|
finish();
|
||||||
break;
|
System.runFinalization();
|
||||||
case R.id.nav_exit:
|
System.exit(0);
|
||||||
log.debug("Exiting");
|
return true;
|
||||||
MainApp.instance().stopKeepAliveService();
|
case R.id.nav_plugin_preferences:
|
||||||
MainApp.bus().post(new EventAppExit());
|
ViewPager viewPager = findViewById(R.id.pager);
|
||||||
MainApp.closeDbHelper();
|
final PluginBase plugin = ((TabPageAdapter) viewPager.getAdapter()).getPluginAt(viewPager.getCurrentItem());
|
||||||
finish();
|
PasswordProtection.QueryPassword(this, R.string.settings_password, "settings_password", () -> {
|
||||||
System.runFinalization();
|
Intent i = new Intent(this, PreferencesActivity.class);
|
||||||
System.exit(0);
|
i.putExtra("id", plugin.getPreferencesId());
|
||||||
break;
|
startActivity(i);
|
||||||
}
|
}, null);
|
||||||
return false;
|
return true;
|
||||||
}
|
|
||||||
});
|
|
||||||
popup.show();
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
return actionBarDrawerToggle.onOptionsItemSelected(item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,59 @@
|
||||||
|
package info.nightscout.androidaps;
|
||||||
|
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.support.annotation.Nullable;
|
||||||
|
import android.support.v4.app.Fragment;
|
||||||
|
import android.support.v4.view.ViewPager;
|
||||||
|
import android.support.v7.app.AppCompatActivity;
|
||||||
|
import android.view.Menu;
|
||||||
|
import android.view.MenuItem;
|
||||||
|
|
||||||
|
import info.nightscout.androidaps.interfaces.PluginBase;
|
||||||
|
import info.nightscout.androidaps.tabs.TabPageAdapter;
|
||||||
|
import info.nightscout.utils.PasswordProtection;
|
||||||
|
|
||||||
|
public class SingleFragmentActivity extends AppCompatActivity {
|
||||||
|
|
||||||
|
private PluginBase plugin;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onCreate(@Nullable Bundle savedInstanceState) {
|
||||||
|
super.onCreate(savedInstanceState);
|
||||||
|
setContentView(R.layout.activity_single_fragment);
|
||||||
|
|
||||||
|
this.plugin = MainApp.getPluginsList().get(getIntent().getIntExtra("plugin", -1));
|
||||||
|
setTitle(plugin.getName());
|
||||||
|
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
|
||||||
|
getSupportActionBar().setDisplayShowHomeEnabled(true);
|
||||||
|
|
||||||
|
if (savedInstanceState == null) {
|
||||||
|
getSupportFragmentManager().beginTransaction().replace(R.id.frame_layout,
|
||||||
|
Fragment.instantiate(this, plugin.pluginDescription.getFragmentClass())).commit();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
|
if (item.getItemId() == android.R.id.home) {
|
||||||
|
finish();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else if (item.getItemId() == R.id.nav_plugin_preferences) {
|
||||||
|
PasswordProtection.QueryPassword(this, R.string.settings_password, "settings_password", () -> {
|
||||||
|
Intent i = new Intent(this, PreferencesActivity.class);
|
||||||
|
i.putExtra("id", plugin.getPreferencesId());
|
||||||
|
startActivity(i);
|
||||||
|
}, null);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean onCreateOptionsMenu(Menu menu) {
|
||||||
|
if (plugin.getPreferencesId() != -1)
|
||||||
|
getMenuInflater().inflate(R.menu.menu_single_fragment, menu);
|
||||||
|
return super.onCreateOptionsMenu(menu);
|
||||||
|
}
|
||||||
|
}
|
|
@ -226,6 +226,10 @@ public class Profile {
|
||||||
basal_v.setValueAt(i, description.basalMinimumRate);
|
basal_v.setValueAt(i, description.basalMinimumRate);
|
||||||
if (notify)
|
if (notify)
|
||||||
sendBelowMinimumNotification(from);
|
sendBelowMinimumNotification(from);
|
||||||
|
} else if (basal_v.valueAt(i) > description.basalMaximumRate) {
|
||||||
|
basal_v.setValueAt(i, description.basalMaximumRate);
|
||||||
|
if (notify)
|
||||||
|
sendAboveMaximumNotification(from);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -243,6 +247,10 @@ public class Profile {
|
||||||
MainApp.bus().post(new EventNewNotification(new Notification(Notification.MINIMAL_BASAL_VALUE_REPLACED, String.format(MainApp.gs(R.string.minimalbasalvaluereplaced), from), Notification.NORMAL)));
|
MainApp.bus().post(new EventNewNotification(new Notification(Notification.MINIMAL_BASAL_VALUE_REPLACED, String.format(MainApp.gs(R.string.minimalbasalvaluereplaced), from), Notification.NORMAL)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected void sendAboveMaximumNotification(String from) {
|
||||||
|
MainApp.bus().post(new EventNewNotification(new Notification(Notification.MAXIMUM_BASAL_VALUE_REPLACED, String.format(MainApp.gs(R.string.maximumbasalvaluereplaced), from), Notification.NORMAL)));
|
||||||
|
}
|
||||||
|
|
||||||
private void validate(LongSparseArray array) {
|
private void validate(LongSparseArray array) {
|
||||||
if (array.size() == 0) {
|
if (array.size() == 0) {
|
||||||
isValid = false;
|
isValid = false;
|
||||||
|
|
|
@ -35,6 +35,7 @@ public class PumpDescription {
|
||||||
public boolean isSetBasalProfileCapable = true;
|
public boolean isSetBasalProfileCapable = true;
|
||||||
public double basalStep = 0.01d;
|
public double basalStep = 0.01d;
|
||||||
public double basalMinimumRate = 0.04d;
|
public double basalMinimumRate = 0.04d;
|
||||||
|
public double basalMaximumRate = 25d;
|
||||||
|
|
||||||
public boolean isRefillingCapable = false;
|
public boolean isRefillingCapable = false;
|
||||||
|
|
||||||
|
|
|
@ -222,7 +222,7 @@ public class CareportalFragment extends SubscriberFragment implements View.OnCli
|
||||||
|
|
||||||
double cageUrgent = nsSettings.getExtendedWarnValue("cage", "urgent", 72);
|
double cageUrgent = nsSettings.getExtendedWarnValue("cage", "urgent", 72);
|
||||||
double cageWarn = nsSettings.getExtendedWarnValue("cage", "warn", 48);
|
double cageWarn = nsSettings.getExtendedWarnValue("cage", "warn", 48);
|
||||||
handleAge(sage, CareportalEvent.SITECHANGE, cageWarn, cageUrgent);
|
handleAge(cage, CareportalEvent.SITECHANGE, cageWarn, cageUrgent);
|
||||||
|
|
||||||
double sageUrgent = nsSettings.getExtendedWarnValue("sage", "urgent", 166);
|
double sageUrgent = nsSettings.getExtendedWarnValue("sage", "urgent", 166);
|
||||||
double sageWarn = nsSettings.getExtendedWarnValue("sage", "warn", 164);
|
double sageWarn = nsSettings.getExtendedWarnValue("sage", "warn", 164);
|
||||||
|
|
|
@ -87,9 +87,9 @@ public class ConfigBuilderPlugin extends PluginBase {
|
||||||
super(new PluginDescription()
|
super(new PluginDescription()
|
||||||
.mainType(PluginType.GENERAL)
|
.mainType(PluginType.GENERAL)
|
||||||
.fragmentClass(ConfigBuilderFragment.class.getName())
|
.fragmentClass(ConfigBuilderFragment.class.getName())
|
||||||
.showInList(false)
|
.showInList(true)
|
||||||
.alwaysEnabled(true)
|
.alwaysEnabled(true)
|
||||||
.alwayVisible(true)
|
.alwayVisible(false)
|
||||||
.pluginName(R.string.configbuilder)
|
.pluginName(R.string.configbuilder)
|
||||||
.shortName(R.string.configbuilder_shortname)
|
.shortName(R.string.configbuilder_shortname)
|
||||||
);
|
);
|
||||||
|
|
|
@ -64,6 +64,7 @@ public class Notification {
|
||||||
public static final int PERMISSION_LOCATION = 36;
|
public static final int PERMISSION_LOCATION = 36;
|
||||||
public static final int PERMISSION_BATTERY = 37;
|
public static final int PERMISSION_BATTERY = 37;
|
||||||
public static final int PERMISSION_SMS = 38;
|
public static final int PERMISSION_SMS = 38;
|
||||||
|
public static final int MAXIMUM_BASAL_VALUE_REPLACED = 39;
|
||||||
|
|
||||||
|
|
||||||
public int id;
|
public int id;
|
||||||
|
|
|
@ -1,20 +0,0 @@
|
||||||
package info.nightscout.androidaps.plugins.PumpInsight;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Created by jamorham on 25/01/2018.
|
|
||||||
*
|
|
||||||
* Async command status
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
enum Cstatus {
|
|
||||||
UNKNOWN,
|
|
||||||
PENDING,
|
|
||||||
SUCCESS,
|
|
||||||
FAILURE,
|
|
||||||
TIMEOUT;
|
|
||||||
|
|
||||||
boolean success() {
|
|
||||||
return this == SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,95 +0,0 @@
|
||||||
package info.nightscout.androidaps.plugins.PumpInsight;
|
|
||||||
|
|
||||||
import android.os.PowerManager;
|
|
||||||
|
|
||||||
import com.squareup.otto.Subscribe;
|
|
||||||
|
|
||||||
import java.util.UUID;
|
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
|
||||||
|
|
||||||
import info.nightscout.androidaps.MainApp;
|
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.events.EventInsightCallback;
|
|
||||||
|
|
||||||
import static info.nightscout.androidaps.plugins.PumpInsight.utils.Helpers.getWakeLock;
|
|
||||||
import static info.nightscout.androidaps.plugins.PumpInsight.utils.Helpers.msSince;
|
|
||||||
import static info.nightscout.androidaps.plugins.PumpInsight.utils.Helpers.releaseWakeLock;
|
|
||||||
import static info.nightscout.androidaps.plugins.PumpInsight.utils.Helpers.tsl;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Created by jamorham on 25/01/2018.
|
|
||||||
*
|
|
||||||
* Asynchronous adapter
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
public class InsightAsyncAdapter {
|
|
||||||
|
|
||||||
private final ConcurrentHashMap<UUID, EventInsightCallback> commandResults = new ConcurrentHashMap<>();
|
|
||||||
|
|
||||||
InsightAsyncAdapter() {
|
|
||||||
MainApp.bus().register(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
// just log during debugging
|
|
||||||
private static void log(String msg) {
|
|
||||||
android.util.Log.e("INSIGHTPUMPASYNC", msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Subscribe
|
|
||||||
public void onStatusEvent(final EventInsightCallback ev) {
|
|
||||||
log("Received callback event: " + ev.toString());
|
|
||||||
commandResults.put(ev.request_uuid, ev);
|
|
||||||
}
|
|
||||||
|
|
||||||
// poll command result
|
|
||||||
private Cstatus checkCommandResult(UUID uuid) {
|
|
||||||
if (uuid == null) return Cstatus.FAILURE;
|
|
||||||
if (commandResults.containsKey(uuid)) {
|
|
||||||
if (commandResults.get(uuid).success) {
|
|
||||||
return Cstatus.SUCCESS;
|
|
||||||
} else {
|
|
||||||
return Cstatus.FAILURE;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return Cstatus.PENDING;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// blocking call to wait for result callback
|
|
||||||
private Cstatus busyWaitForCommandInternal(final UUID uuid, long wait_time) {
|
|
||||||
final PowerManager.WakeLock wl = getWakeLock("insight-wait-cmd", 60000);
|
|
||||||
try {
|
|
||||||
log("busy wait for command " + uuid);
|
|
||||||
if (uuid == null) return Cstatus.FAILURE;
|
|
||||||
final long start_time = tsl();
|
|
||||||
Cstatus status = checkCommandResult(uuid);
|
|
||||||
while ((status == Cstatus.PENDING) && msSince(start_time) < wait_time) {
|
|
||||||
//log("command result waiting");
|
|
||||||
try {
|
|
||||||
Thread.sleep(200);
|
|
||||||
} catch (InterruptedException e) {
|
|
||||||
log("Got interrupted exception! " + e);
|
|
||||||
}
|
|
||||||
status = checkCommandResult(uuid);
|
|
||||||
}
|
|
||||||
if (status == Cstatus.PENDING) {
|
|
||||||
return Cstatus.TIMEOUT;
|
|
||||||
} else {
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
releaseWakeLock(wl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// wait for and then package result, cleanup and return
|
|
||||||
Mstatus busyWaitForCommandResult(final UUID uuid, long wait_time) {
|
|
||||||
final Mstatus mstatus = new Mstatus();
|
|
||||||
mstatus.cstatus = busyWaitForCommandInternal(uuid, wait_time);
|
|
||||||
mstatus.event = commandResults.get(uuid);
|
|
||||||
commandResults.remove(uuid);
|
|
||||||
return mstatus;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
|
@ -9,7 +9,6 @@ import java.text.DecimalFormat;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
import info.nightscout.androidaps.BuildConfig;
|
import info.nightscout.androidaps.BuildConfig;
|
||||||
import info.nightscout.androidaps.Config;
|
import info.nightscout.androidaps.Config;
|
||||||
|
@ -21,7 +20,6 @@ import info.nightscout.androidaps.data.PumpEnactResult;
|
||||||
import info.nightscout.androidaps.db.ExtendedBolus;
|
import info.nightscout.androidaps.db.ExtendedBolus;
|
||||||
import info.nightscout.androidaps.db.Source;
|
import info.nightscout.androidaps.db.Source;
|
||||||
import info.nightscout.androidaps.db.TemporaryBasal;
|
import info.nightscout.androidaps.db.TemporaryBasal;
|
||||||
import info.nightscout.androidaps.plugins.Treatments.Treatment;
|
|
||||||
import info.nightscout.androidaps.interfaces.Constraint;
|
import info.nightscout.androidaps.interfaces.Constraint;
|
||||||
import info.nightscout.androidaps.interfaces.ConstraintsInterface;
|
import info.nightscout.androidaps.interfaces.ConstraintsInterface;
|
||||||
import info.nightscout.androidaps.interfaces.PluginBase;
|
import info.nightscout.androidaps.interfaces.PluginBase;
|
||||||
|
@ -33,7 +31,8 @@ import info.nightscout.androidaps.plugins.Overview.events.EventDismissNotificati
|
||||||
import info.nightscout.androidaps.plugins.Overview.events.EventNewNotification;
|
import info.nightscout.androidaps.plugins.Overview.events.EventNewNotification;
|
||||||
import info.nightscout.androidaps.plugins.Overview.events.EventOverviewBolusProgress;
|
import info.nightscout.androidaps.plugins.Overview.events.EventOverviewBolusProgress;
|
||||||
import info.nightscout.androidaps.plugins.Overview.notifications.Notification;
|
import info.nightscout.androidaps.plugins.Overview.notifications.Notification;
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.connector.CancelBolusTaskRunner;
|
import info.nightscout.androidaps.plugins.PumpInsight.connector.CancelBolusSilentlyTaskRunner;
|
||||||
|
import info.nightscout.androidaps.plugins.PumpInsight.connector.CancelTBRSilentlyTaskRunner;
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.connector.Connector;
|
import info.nightscout.androidaps.plugins.PumpInsight.connector.Connector;
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.connector.SetTBRTaskRunner;
|
import info.nightscout.androidaps.plugins.PumpInsight.connector.SetTBRTaskRunner;
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.connector.StatusTaskRunner;
|
import info.nightscout.androidaps.plugins.PumpInsight.connector.StatusTaskRunner;
|
||||||
|
@ -44,12 +43,14 @@ import info.nightscout.androidaps.plugins.PumpInsight.history.HistoryReceiver;
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.history.LiveHistory;
|
import info.nightscout.androidaps.plugins.PumpInsight.history.LiveHistory;
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.utils.Helpers;
|
import info.nightscout.androidaps.plugins.PumpInsight.utils.Helpers;
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.utils.StatusItem;
|
import info.nightscout.androidaps.plugins.PumpInsight.utils.StatusItem;
|
||||||
|
import info.nightscout.androidaps.plugins.Treatments.Treatment;
|
||||||
import info.nightscout.androidaps.plugins.Treatments.TreatmentsPlugin;
|
import info.nightscout.androidaps.plugins.Treatments.TreatmentsPlugin;
|
||||||
import info.nightscout.utils.DateUtil;
|
import info.nightscout.utils.DateUtil;
|
||||||
import info.nightscout.utils.NSUpload;
|
import info.nightscout.utils.NSUpload;
|
||||||
import info.nightscout.utils.SP;
|
import info.nightscout.utils.SP;
|
||||||
import sugar.free.sightparser.applayer.descriptors.ActiveBolus;
|
import sugar.free.sightparser.applayer.descriptors.ActiveBolus;
|
||||||
import sugar.free.sightparser.applayer.descriptors.ActiveBolusType;
|
import sugar.free.sightparser.applayer.descriptors.ActiveBolusType;
|
||||||
|
import sugar.free.sightparser.applayer.descriptors.MessagePriority;
|
||||||
import sugar.free.sightparser.applayer.descriptors.PumpStatus;
|
import sugar.free.sightparser.applayer.descriptors.PumpStatus;
|
||||||
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.BRProfileBlock;
|
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.BRProfileBlock;
|
||||||
import sugar.free.sightparser.applayer.messages.AppLayerMessage;
|
import sugar.free.sightparser.applayer.messages.AppLayerMessage;
|
||||||
|
@ -94,7 +95,6 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
private static boolean initialized = false;
|
private static boolean initialized = false;
|
||||||
private static volatile boolean update_pending = false;
|
private static volatile boolean update_pending = false;
|
||||||
private static Logger log = LoggerFactory.getLogger(InsightPlugin.class);
|
private static Logger log = LoggerFactory.getLogger(InsightPlugin.class);
|
||||||
private final InsightAsyncAdapter async = new InsightAsyncAdapter();
|
|
||||||
private StatusTaskRunner.Result statusResult;
|
private StatusTaskRunner.Result statusResult;
|
||||||
private long statusResultTime = -1;
|
private long statusResultTime = -1;
|
||||||
private Date lastDataTime = new Date(0);
|
private Date lastDataTime = new Date(0);
|
||||||
|
@ -115,10 +115,10 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
);
|
);
|
||||||
log("InsightPlugin instantiated");
|
log("InsightPlugin instantiated");
|
||||||
pumpDescription.isBolusCapable = true;
|
pumpDescription.isBolusCapable = true;
|
||||||
pumpDescription.bolusStep = 0.05d; // specification says 0.05U up to 2U then 0.1U @ 2-5U 0.2U @ 10-20U 0.5U 10-20U (are these just UI restrictions?)
|
pumpDescription.bolusStep = 0.01d; // specification says 0.05U up to 2U then 0.1U @ 2-5U 0.2U @ 10-20U 0.5U 10-20U (are these just UI restrictions? Yes, they are!)
|
||||||
|
|
||||||
pumpDescription.isExtendedBolusCapable = true;
|
pumpDescription.isExtendedBolusCapable = true;
|
||||||
pumpDescription.extendedBolusStep = 0.05d; // specification probably same as above
|
pumpDescription.extendedBolusStep = 0.01d; // specification probably same as above
|
||||||
pumpDescription.extendedBolusDurationStep = 15; // 15 minutes up to 24 hours
|
pumpDescription.extendedBolusDurationStep = 15; // 15 minutes up to 24 hours
|
||||||
pumpDescription.extendedBolusMaxDuration = 24 * 60;
|
pumpDescription.extendedBolusMaxDuration = 24 * 60;
|
||||||
|
|
||||||
|
@ -138,6 +138,7 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
pumpDescription.is30minBasalRatesCapable = true;
|
pumpDescription.is30minBasalRatesCapable = true;
|
||||||
pumpDescription.basalStep = 0.01d;
|
pumpDescription.basalStep = 0.01d;
|
||||||
pumpDescription.basalMinimumRate = 0.02d;
|
pumpDescription.basalMinimumRate = 0.02d;
|
||||||
|
pumpDescription.basalMaximumRate = 25d;
|
||||||
|
|
||||||
pumpDescription.isRefillingCapable = true;
|
pumpDescription.isRefillingCapable = true;
|
||||||
|
|
||||||
|
@ -191,7 +192,7 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean isFakingTempsByExtendedBoluses() {
|
public boolean isFakingTempsByExtendedBoluses() {
|
||||||
return false;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -287,17 +288,15 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
log("getPumpStatus");
|
log("getPumpStatus");
|
||||||
if (Connector.get().isPumpConnected()) {
|
if (Connector.get().isPumpConnected()) {
|
||||||
log("is connected.. requesting status");
|
log("is connected.. requesting status");
|
||||||
final UUID uuid = aSyncTaskRunner(new StatusTaskRunner(connector.getServiceConnector()), "Status");
|
try {
|
||||||
Mstatus mstatus = async.busyWaitForCommandResult(uuid, BUSY_WAIT_TIME);
|
setStatusResult(fetchTaskRunner(new StatusTaskRunner(connector.getServiceConnector()), StatusTaskRunner.Result.class));
|
||||||
if (mstatus.success()) {
|
|
||||||
log("GOT STATUS RESULT!!! PARTY WOOHOO!!!");
|
log("GOT STATUS RESULT!!! PARTY WOOHOO!!!");
|
||||||
setStatusResult((StatusTaskRunner.Result) mstatus.getResponseObject());
|
|
||||||
statusResultTime = Helpers.tsl();
|
statusResultTime = Helpers.tsl();
|
||||||
processStatusResult();
|
processStatusResult();
|
||||||
updateGui();
|
updateGui();
|
||||||
connector.requestHistoryReSync();
|
connector.requestHistoryReSync();
|
||||||
connector.requestHistorySync();
|
connector.requestHistorySync();
|
||||||
} else {
|
} catch (Exception e) {
|
||||||
log("StatusTaskRunner wasn't successful.");
|
log("StatusTaskRunner wasn't successful.");
|
||||||
if (connector.getServiceConnector().isConnectedToService() && connector.getServiceConnector().getStatus() != Status.CONNECTED) {
|
if (connector.getServiceConnector().isConnectedToService() && connector.getServiceConnector().getStatus() != Status.CONNECTED) {
|
||||||
if (Helpers.ratelimit("insight-reconnect", 2)) {
|
if (Helpers.ratelimit("insight-reconnect", 2)) {
|
||||||
|
@ -313,6 +312,8 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
|
|
||||||
public void setStatusResult(StatusTaskRunner.Result result) {
|
public void setStatusResult(StatusTaskRunner.Result result) {
|
||||||
this.statusResult = result;
|
this.statusResult = result;
|
||||||
|
this.pumpDescription.basalMinimumRate = result.minimumBasalAmount;
|
||||||
|
this.pumpDescription.basalMaximumRate = result.maximumBasalAmount;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -335,9 +336,8 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
profileBlocks.add(new BRProfileBlock.ProfileBlock((((nextValue != null ? nextValue.timeAsSeconds : 24 * 60 * 60) - basalValue.timeAsSeconds) / 60), Helpers.roundDouble(basalValue.value, 2)));
|
profileBlocks.add(new BRProfileBlock.ProfileBlock((((nextValue != null ? nextValue.timeAsSeconds : 24 * 60 * 60) - basalValue.timeAsSeconds) / 60), Helpers.roundDouble(basalValue.value, 2)));
|
||||||
log("setNewBasalProfile: " + basalValue.value + " for " + Integer.toString(((nextValue != null ? nextValue.timeAsSeconds : 24 * 60 * 60) - basalValue.timeAsSeconds) / 60));
|
log("setNewBasalProfile: " + basalValue.value + " for " + Integer.toString(((nextValue != null ? nextValue.timeAsSeconds : 24 * 60 * 60) - basalValue.timeAsSeconds) / 60));
|
||||||
}
|
}
|
||||||
final UUID uuid = aSyncTaskRunner(new WriteBasalProfileTaskRunner(connector.getServiceConnector(), profileBlocks), "Write basal profile");
|
try {
|
||||||
final Mstatus ms = async.busyWaitForCommandResult(uuid, BUSY_WAIT_TIME);
|
fetchTaskRunner(new WriteBasalProfileTaskRunner(connector.getServiceConnector(), profileBlocks));
|
||||||
if (ms.success()) {
|
|
||||||
MainApp.bus().post(new EventDismissNotification(Notification.FAILED_UDPATE_PROFILE));
|
MainApp.bus().post(new EventDismissNotification(Notification.FAILED_UDPATE_PROFILE));
|
||||||
Notification notification = new Notification(Notification.PROFILE_SET_OK, MainApp.gs(R.string.profile_set_ok), Notification.INFO, 60);
|
Notification notification = new Notification(Notification.PROFILE_SET_OK, MainApp.gs(R.string.profile_set_ok), Notification.INFO, 60);
|
||||||
MainApp.bus().post(new EventNewNotification(notification));
|
MainApp.bus().post(new EventNewNotification(notification));
|
||||||
|
@ -345,7 +345,7 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
result.enacted = true;
|
result.enacted = true;
|
||||||
result.comment = "OK";
|
result.comment = "OK";
|
||||||
this.profileBlocks = profileBlocks;
|
this.profileBlocks = profileBlocks;
|
||||||
} else {
|
} catch (Exception e) {
|
||||||
Notification notification = new Notification(Notification.FAILED_UDPATE_PROFILE, MainApp.gs(R.string.failedupdatebasalprofile), Notification.URGENT);
|
Notification notification = new Notification(Notification.FAILED_UDPATE_PROFILE, MainApp.gs(R.string.failedupdatebasalprofile), Notification.URGENT);
|
||||||
MainApp.bus().post(new EventNewNotification(notification));
|
MainApp.bus().post(new EventNewNotification(notification));
|
||||||
result.comment = MainApp.gs(R.string.failedupdatebasalprofile);
|
result.comment = MainApp.gs(R.string.failedupdatebasalprofile);
|
||||||
|
@ -403,17 +403,13 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
|
|
||||||
// is there an insulin component to the treatment?
|
// is there an insulin component to the treatment?
|
||||||
if (detailedBolusInfo.insulin > 0) {
|
if (detailedBolusInfo.insulin > 0) {
|
||||||
final UUID cmd = deliverBolus(detailedBolusInfo.insulin); // actually request delivery
|
try {
|
||||||
if (cmd == null) {
|
bolusId = deliverBolus(detailedBolusInfo.insulin);
|
||||||
|
result.success = true;
|
||||||
|
detailedBolusInfo.pumpId = getRecordUniqueID(bolusId);
|
||||||
|
} catch (Exception e) {
|
||||||
return pumpEnactFailure();
|
return pumpEnactFailure();
|
||||||
}
|
}
|
||||||
final Mstatus ms = async.busyWaitForCommandResult(cmd, BUSY_WAIT_TIME);
|
|
||||||
|
|
||||||
result.success = ms.success();
|
|
||||||
if (ms.success()) {
|
|
||||||
detailedBolusInfo.pumpId = getRecordUniqueID(ms.getResponseID());
|
|
||||||
bolusId = ms.getResponseID();
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
result.success = true; // always true with carb only treatments
|
result.success = true; // always true with carb only treatments
|
||||||
}
|
}
|
||||||
|
@ -440,20 +436,11 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
updateGui();
|
updateGui();
|
||||||
connector.tryToGetPumpStatusAgain();
|
connector.tryToGetPumpStatusAgain();
|
||||||
|
|
||||||
connector.requestHistorySync(30000);
|
|
||||||
|
|
||||||
if (result.success) while (true) {
|
if (result.success) while (true) {
|
||||||
try {
|
try {
|
||||||
Thread.sleep(200);
|
Thread.sleep(500);
|
||||||
} catch (InterruptedException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
final UUID uuid = aSyncSingleCommand(new ActiveBolusesMessage(), "Active boluses");
|
|
||||||
Mstatus mstatus = async.busyWaitForCommandResult(uuid, BUSY_WAIT_TIME);
|
|
||||||
if (mstatus.success()) {
|
|
||||||
final EventOverviewBolusProgress bolusingEvent = EventOverviewBolusProgress.getInstance();
|
final EventOverviewBolusProgress bolusingEvent = EventOverviewBolusProgress.getInstance();
|
||||||
ActiveBolusesMessage activeBolusesMessage = (ActiveBolusesMessage) mstatus.getResponseObject();
|
ActiveBolusesMessage activeBolusesMessage = fetchSingleMessage(new ActiveBolusesMessage(), ActiveBolusesMessage.class);
|
||||||
ActiveBolus activeBolus = null;
|
ActiveBolus activeBolus = null;
|
||||||
if (activeBolusesMessage.getBolus1() != null && activeBolusesMessage.getBolus1().getBolusID() == bolusingEvent.bolusId)
|
if (activeBolusesMessage.getBolus1() != null && activeBolusesMessage.getBolus1().getBolusID() == bolusingEvent.bolusId)
|
||||||
activeBolus = activeBolusesMessage.getBolus1();
|
activeBolus = activeBolusesMessage.getBolus1();
|
||||||
|
@ -463,87 +450,60 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
activeBolus = activeBolusesMessage.getBolus3();
|
activeBolus = activeBolusesMessage.getBolus3();
|
||||||
if (activeBolus == null) break;
|
if (activeBolus == null) break;
|
||||||
else {
|
else {
|
||||||
|
int percentBefore = bolusingEvent.percent;
|
||||||
bolusingEvent.percent = (int) (100D / activeBolus.getInitialAmount() * (activeBolus.getInitialAmount() - activeBolus.getLeftoverAmount()));
|
bolusingEvent.percent = (int) (100D / activeBolus.getInitialAmount() * (activeBolus.getInitialAmount() - activeBolus.getLeftoverAmount()));
|
||||||
bolusingEvent.status = String.format(MainApp.gs(R.string.bolusdelivering), activeBolus.getInitialAmount() - activeBolus.getLeftoverAmount());
|
bolusingEvent.status = String.format(MainApp.gs(R.string.bolusdelivering), activeBolus.getInitialAmount() - activeBolus.getLeftoverAmount());
|
||||||
MainApp.bus().post(bolusingEvent);
|
if (percentBefore != bolusingEvent.percent) MainApp.bus().post(bolusingEvent);
|
||||||
}
|
}
|
||||||
} else break;
|
} catch (Exception e) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
connector.requestHistorySync(2000);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void stopBolusDelivering() {
|
public void stopBolusDelivering() {
|
||||||
CancelBolusMessage cancelBolusMessage = new CancelBolusMessage();
|
CancelBolusMessage cancelBolusMessage = new CancelBolusMessage();
|
||||||
|
cancelBolusMessage.setMessagePriority(MessagePriority.HIGHEST);
|
||||||
cancelBolusMessage.setBolusId(EventOverviewBolusProgress.getInstance().bolusId);
|
cancelBolusMessage.setBolusId(EventOverviewBolusProgress.getInstance().bolusId);
|
||||||
final UUID cmd = aSyncSingleCommand(cancelBolusMessage, "Cancel standard bolus");
|
try {
|
||||||
|
fetchSingleMessage(cancelBolusMessage);
|
||||||
if (cmd == null) {
|
} catch (Exception e) {
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
final Mstatus cs = async.busyWaitForCommandResult(cmd, BUSY_WAIT_TIME);
|
|
||||||
log("Got command status: " + cs);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Temporary Basals
|
// Temporary Basals
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public PumpEnactResult setTempBasalAbsolute(Double absoluteRate, Integer durationInMinutes, Profile profile, boolean enforceNew) {
|
public PumpEnactResult setTempBasalAbsolute(Double absoluteRate, Integer durationInMinutes, Profile profile, boolean enforceNew) {
|
||||||
absoluteRate = Helpers.roundDouble(absoluteRate, 3);
|
|
||||||
log("Set TBR absolute: " + absoluteRate);
|
log("Set TBR absolute: " + absoluteRate);
|
||||||
final double base_basal = getBaseBasalRate();
|
if (getBaseBasalRate() == 0) {
|
||||||
if (base_basal == 0) {
|
|
||||||
log("Base basal rate appears to be zero!");
|
log("Base basal rate appears to be zero!");
|
||||||
return pumpEnactFailure();
|
return pumpEnactFailure();
|
||||||
}
|
}
|
||||||
int percent_amount = (int) Math.round(100d / base_basal * absoluteRate);
|
double percent = 100D / getBaseBasalRate() * absoluteRate;
|
||||||
log("Calculated requested rate: " + absoluteRate + " base rate: " + base_basal + " percentage: " + percent_amount + "%");
|
log("Calculated requested rate: " + absoluteRate + " base rate: " + getBaseBasalRate() + " percentage: " + percent + "%");
|
||||||
percent_amount = (int) Math.round(((double) percent_amount) / 10d) * 10;
|
try {
|
||||||
log("Calculated final rate: " + percent_amount + "%");
|
if (percent > 250) {
|
||||||
|
log ("Calculated rate is above 250%, switching to emulation using extended boluses");
|
||||||
if (percent_amount == 100) {
|
cancelTempBasal(true);
|
||||||
return cancelTempBasal(false);
|
if (!setExtendedBolus((absoluteRate - getBaseBasalRate()) / 60D * ((double) durationInMinutes), durationInMinutes).success) {
|
||||||
}
|
//Fallback to TBR if setting an extended bolus didn't work
|
||||||
|
log ("Setting an extended bolus didn't work, falling back to normal TBR");
|
||||||
if (percent_amount > 250) percent_amount = 250;
|
return setTempBasalPercent((int) percent, durationInMinutes, profile, true);
|
||||||
|
}
|
||||||
|
return new PumpEnactResult().success(true).enacted(true).absolute(absoluteRate).duration(durationInMinutes);
|
||||||
final SetTBRTaskRunner task = new SetTBRTaskRunner(connector.getServiceConnector(), percent_amount, durationInMinutes);
|
} else {
|
||||||
final UUID cmd = aSyncTaskRunner(task, "Set TBR abs: " + absoluteRate + " " + durationInMinutes + "m");
|
log ("Calculated rate is below or equal to 250%, using normal TBRs");
|
||||||
|
cancelExtendedBolus();
|
||||||
if (cmd == null) {
|
return setTempBasalPercent((int) percent, durationInMinutes, profile, true);
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
return pumpEnactFailure();
|
return pumpEnactFailure();
|
||||||
}
|
}
|
||||||
|
|
||||||
Mstatus ms = async.busyWaitForCommandResult(cmd, BUSY_WAIT_TIME);
|
|
||||||
log("Got command status: " + ms);
|
|
||||||
|
|
||||||
PumpEnactResult pumpEnactResult = new PumpEnactResult().enacted(true).isPercent(true).duration(durationInMinutes);
|
|
||||||
pumpEnactResult.percent = percent_amount;
|
|
||||||
pumpEnactResult.success = ms.success();
|
|
||||||
pumpEnactResult.comment = ms.getCommandComment();
|
|
||||||
|
|
||||||
|
|
||||||
if (pumpEnactResult.success) {
|
|
||||||
// create log entry
|
|
||||||
final TemporaryBasal tempBasal = new TemporaryBasal()
|
|
||||||
.date(System.currentTimeMillis())
|
|
||||||
.percent(percent_amount)
|
|
||||||
.duration(durationInMinutes)
|
|
||||||
.source(Source.USER);
|
|
||||||
TreatmentsPlugin.getPlugin().addToHistoryTempBasal(tempBasal);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Config.logPumpComm)
|
|
||||||
log.debug("Setting temp basal absolute: " + pumpEnactResult.success);
|
|
||||||
|
|
||||||
updateGui();
|
|
||||||
|
|
||||||
connector.requestHistorySync(5000);
|
|
||||||
connector.tryToGetPumpStatusAgain();
|
|
||||||
|
|
||||||
return pumpEnactResult;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -555,42 +515,24 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
if (percent == 100) {
|
if (percent == 100) {
|
||||||
// This would cause a cancel if a tbr is in progress so treat as a cancel
|
// This would cause a cancel if a tbr is in progress so treat as a cancel
|
||||||
return cancelTempBasal(false);
|
return cancelTempBasal(false);
|
||||||
}
|
} else if (percent > 250) percent = 250;
|
||||||
|
|
||||||
|
try {
|
||||||
final UUID cmd = aSyncTaskRunner(new SetTBRTaskRunner(connector.getServiceConnector(), percent, durationInMinutes), "Set TBR " + percent + "%" + " " + durationInMinutes + "m");
|
fetchTaskRunner(new SetTBRTaskRunner(connector.getServiceConnector(), percent, durationInMinutes));
|
||||||
|
|
||||||
if (cmd == null) {
|
|
||||||
return pumpEnactFailure();
|
|
||||||
}
|
|
||||||
|
|
||||||
final Mstatus ms = async.busyWaitForCommandResult(cmd, BUSY_WAIT_TIME);
|
|
||||||
log("Got command status: " + ms);
|
|
||||||
|
|
||||||
PumpEnactResult pumpEnactResult = new PumpEnactResult().enacted(true).isPercent(true).duration(durationInMinutes);
|
|
||||||
pumpEnactResult.percent = percent;
|
|
||||||
pumpEnactResult.success = ms.success();
|
|
||||||
pumpEnactResult.comment = ms.getCommandComment();
|
|
||||||
|
|
||||||
if (pumpEnactResult.success) {
|
|
||||||
// create log entry
|
|
||||||
final TemporaryBasal tempBasal = new TemporaryBasal()
|
final TemporaryBasal tempBasal = new TemporaryBasal()
|
||||||
.date(System.currentTimeMillis())
|
.date(System.currentTimeMillis())
|
||||||
.percent(percent)
|
.percent(percent)
|
||||||
.duration(durationInMinutes)
|
.duration(durationInMinutes)
|
||||||
.source(Source.USER); // TODO check this is correct
|
.source(Source.USER);
|
||||||
TreatmentsPlugin.getPlugin().addToHistoryTempBasal(tempBasal);
|
TreatmentsPlugin.getPlugin().addToHistoryTempBasal(tempBasal);
|
||||||
|
updateGui();
|
||||||
|
if (Config.logPumpComm) log.debug("Set temp basal " + percent + "% for " + durationInMinutes + "m");
|
||||||
|
connector.requestHistorySync(5000);
|
||||||
|
connector.tryToGetPumpStatusAgain();
|
||||||
|
return new PumpEnactResult().success(true).enacted(true).percent(percent);
|
||||||
|
} catch (Exception e) {
|
||||||
|
return pumpEnactFailure();
|
||||||
}
|
}
|
||||||
|
|
||||||
updateGui();
|
|
||||||
|
|
||||||
if (Config.logPumpComm)
|
|
||||||
log.debug("Set temp basal " + percent + "% for " + durationInMinutes + "m");
|
|
||||||
|
|
||||||
connector.requestHistorySync(5000);
|
|
||||||
connector.tryToGetPumpStatusAgain();
|
|
||||||
|
|
||||||
return pumpEnactResult;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -598,35 +540,24 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
public PumpEnactResult cancelTempBasal(boolean enforceNew) {
|
public PumpEnactResult cancelTempBasal(boolean enforceNew) {
|
||||||
log("Cancel TBR");
|
log("Cancel TBR");
|
||||||
|
|
||||||
|
try {
|
||||||
fauxTBRcancel = !SP.getBoolean("insight_real_tbr_cancel", false);
|
cancelExtendedBolus();
|
||||||
|
realTBRCancel();
|
||||||
final UUID cmd;
|
updateGui();
|
||||||
|
if (Config.logPumpComm) log.debug("Canceling temp basal");
|
||||||
if (fauxTBRcancel) {
|
connector.requestHistorySync(5000);
|
||||||
cmd = aSyncTaskRunner(new SetTBRTaskRunner(connector.getServiceConnector(), 100, 1), "Faux Cancel TBR - setting " + "90%" + " 1m");
|
connector.tryToGetPumpStatusAgain();
|
||||||
} else {
|
return new PumpEnactResult().success(true).enacted(true).isTempCancel(true);
|
||||||
cmd = aSyncSingleCommand(new CancelTBRMessage(), "Cancel Temp Basal");
|
} catch (Exception e) {
|
||||||
}
|
|
||||||
if (cmd == null) {
|
|
||||||
return pumpEnactFailure();
|
return pumpEnactFailure();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// TODO isn't conditional on one apparently being in progress only the history change
|
private void realTBRCancel() throws Exception {
|
||||||
final Mstatus ms = async.busyWaitForCommandResult(cmd, BUSY_WAIT_TIME);
|
if (fetchTaskRunner(new CancelTBRSilentlyTaskRunner(connector.getServiceConnector()), Boolean.class) && TreatmentsPlugin.getPlugin().isTempBasalInProgress()) {
|
||||||
|
|
||||||
if (TreatmentsPlugin.getPlugin().isTempBasalInProgress()) {
|
|
||||||
TemporaryBasal tempStop = new TemporaryBasal().date(System.currentTimeMillis()).source(Source.USER);
|
TemporaryBasal tempStop = new TemporaryBasal().date(System.currentTimeMillis()).source(Source.USER);
|
||||||
TreatmentsPlugin.getPlugin().addToHistoryTempBasal(tempStop);
|
TreatmentsPlugin.getPlugin().addToHistoryTempBasal(tempStop);
|
||||||
}
|
}
|
||||||
updateGui();
|
|
||||||
if (Config.logPumpComm)
|
|
||||||
log.debug("Canceling temp basal: "); // TODO get more info
|
|
||||||
|
|
||||||
connector.requestHistorySync(5000);
|
|
||||||
connector.tryToGetPumpStatusAgain();
|
|
||||||
|
|
||||||
return new PumpEnactResult().success(ms.success()).enacted(true).isTempCancel(true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -635,88 +566,60 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
@Override
|
@Override
|
||||||
public PumpEnactResult setExtendedBolus(Double insulin, Integer durationInMinutes) {
|
public PumpEnactResult setExtendedBolus(Double insulin, Integer durationInMinutes) {
|
||||||
log("Set Extended bolus " + insulin + " " + durationInMinutes);
|
log("Set Extended bolus " + insulin + " " + durationInMinutes);
|
||||||
ExtendedBolusMessage extendedBolusMessage = new ExtendedBolusMessage();
|
try {
|
||||||
extendedBolusMessage.setAmount(insulin);
|
ExtendedBolusMessage extendedBolusMessage = new ExtendedBolusMessage();
|
||||||
extendedBolusMessage.setDuration(durationInMinutes);
|
extendedBolusMessage.setAmount(insulin);
|
||||||
final UUID cmd = aSyncSingleCommand(extendedBolusMessage, "Extended bolus U" + insulin + " mins:" + durationInMinutes);
|
extendedBolusMessage.setDuration(durationInMinutes);
|
||||||
if (cmd == null) {
|
BolusMessage bolusMessage = fetchSingleMessage(extendedBolusMessage, BolusMessage.class);
|
||||||
return pumpEnactFailure();
|
|
||||||
}
|
|
||||||
|
|
||||||
final Mstatus ms = async.busyWaitForCommandResult(cmd, BUSY_WAIT_TIME);
|
|
||||||
log("Got command status: " + ms);
|
|
||||||
|
|
||||||
PumpEnactResult pumpEnactResult = new PumpEnactResult().enacted(true).bolusDelivered(insulin).duration(durationInMinutes);
|
|
||||||
pumpEnactResult.success = ms.success();
|
|
||||||
pumpEnactResult.comment = ms.getCommandComment();
|
|
||||||
|
|
||||||
if (pumpEnactResult.success) {
|
|
||||||
// create log entry
|
|
||||||
final ExtendedBolus extendedBolus = new ExtendedBolus();
|
final ExtendedBolus extendedBolus = new ExtendedBolus();
|
||||||
extendedBolus.date = System.currentTimeMillis();
|
extendedBolus.date = System.currentTimeMillis();
|
||||||
extendedBolus.insulin = insulin;
|
extendedBolus.insulin = insulin;
|
||||||
extendedBolus.durationInMinutes = durationInMinutes;
|
extendedBolus.durationInMinutes = durationInMinutes;
|
||||||
extendedBolus.source = Source.USER;
|
extendedBolus.source = Source.USER;
|
||||||
extendedBolus.pumpId = getRecordUniqueID(ms.getResponseID());
|
extendedBolus.pumpId = getRecordUniqueID(bolusMessage.getBolusId());
|
||||||
TreatmentsPlugin.getPlugin().addToHistoryExtendedBolus(extendedBolus);
|
TreatmentsPlugin.getPlugin().addToHistoryExtendedBolus(extendedBolus);
|
||||||
|
updateGui();
|
||||||
|
connector.requestHistorySync(30000);
|
||||||
|
connector.tryToGetPumpStatusAgain();
|
||||||
|
if (Config.logPumpComm)
|
||||||
|
log.debug("Setting extended bolus: " + insulin + " mins:" + durationInMinutes);
|
||||||
|
return new PumpEnactResult().success(true).enacted(true).duration(durationInMinutes).bolusDelivered(insulin);
|
||||||
|
} catch (Exception e) {
|
||||||
|
return pumpEnactFailure();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Config.logPumpComm)
|
|
||||||
log.debug("Setting extended bolus: " + insulin + " mins:" + durationInMinutes + " " + pumpEnactResult.comment);
|
|
||||||
|
|
||||||
updateGui();
|
|
||||||
|
|
||||||
connector.requestHistorySync(30000);
|
|
||||||
connector.tryToGetPumpStatusAgain();
|
|
||||||
|
|
||||||
return pumpEnactResult;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public PumpEnactResult cancelExtendedBolus() {
|
public PumpEnactResult cancelExtendedBolus() {
|
||||||
|
|
||||||
log("Cancel Extended bolus");
|
log("Cancel Extended bolus");
|
||||||
|
|
||||||
// TODO note always sends cancel to pump but only changes history if present
|
Integer bolusId = null;
|
||||||
|
|
||||||
final UUID cmd = aSyncTaskRunner(new CancelBolusTaskRunner(connector.getServiceConnector(), ActiveBolusType.EXTENDED), "Cancel extended bolus");
|
try {
|
||||||
|
bolusId = fetchTaskRunner(new CancelBolusSilentlyTaskRunner(connector.getServiceConnector(), ActiveBolusType.EXTENDED), Integer.class);
|
||||||
if (cmd == null) {
|
if (TreatmentsPlugin.getPlugin().isInHistoryExtendedBoluslInProgress()) {
|
||||||
|
ExtendedBolus exStop = new ExtendedBolus(System.currentTimeMillis());
|
||||||
|
exStop.source = Source.USER;
|
||||||
|
TreatmentsPlugin.getPlugin().addToHistoryExtendedBolus(exStop);
|
||||||
|
}
|
||||||
|
if (Config.logPumpComm) log.debug("Cancel extended bolus:");
|
||||||
|
if (bolusId != null) connector.requestHistorySync(5000);
|
||||||
|
connector.tryToGetPumpStatusAgain();
|
||||||
|
updateGui();
|
||||||
|
return new PumpEnactResult().success(true).enacted(bolusId != null);
|
||||||
|
} catch (Exception e) {
|
||||||
return pumpEnactFailure();
|
return pumpEnactFailure();
|
||||||
}
|
}
|
||||||
|
|
||||||
final Mstatus ms = async.busyWaitForCommandResult(cmd, BUSY_WAIT_TIME);
|
|
||||||
|
|
||||||
if (TreatmentsPlugin.getPlugin().isInHistoryExtendedBoluslInProgress()) {
|
|
||||||
ExtendedBolus exStop = new ExtendedBolus(System.currentTimeMillis());
|
|
||||||
exStop.source = Source.USER;
|
|
||||||
TreatmentsPlugin.getPlugin().addToHistoryExtendedBolus(exStop);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Config.logPumpComm)
|
|
||||||
log.debug("Cancel extended bolus:");
|
|
||||||
|
|
||||||
updateGui();
|
|
||||||
|
|
||||||
connector.requestHistorySync(5000);
|
|
||||||
connector.tryToGetPumpStatusAgain();
|
|
||||||
|
|
||||||
return new PumpEnactResult().success(ms.success()).enacted(true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private synchronized UUID deliverBolus(double bolusValue) {
|
private int deliverBolus(double bolusValue) throws Exception {
|
||||||
log("DeliverBolus: " + bolusValue);
|
log("DeliverBolus: " + bolusValue);
|
||||||
|
|
||||||
if (bolusValue == 0) return null;
|
|
||||||
if (bolusValue < 0) return null;
|
|
||||||
|
|
||||||
// TODO check limits here or they already occur via a previous constraint interface?
|
|
||||||
|
|
||||||
final StandardBolusMessage message = new StandardBolusMessage();
|
final StandardBolusMessage message = new StandardBolusMessage();
|
||||||
message.setAmount(bolusValue);
|
message.setAmount(bolusValue);
|
||||||
|
|
||||||
return aSyncSingleCommand(message, "Deliver Bolus " + bolusValue);
|
return fetchSingleMessage(message, BolusMessage.class).getBolusId();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -931,82 +834,34 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Utility
|
private void fetchTaskRunner(TaskRunner taskRunner) throws Exception {
|
||||||
|
fetchTaskRunner(taskRunner, Object.class);
|
||||||
private synchronized UUID aSyncSingleCommand(final AppLayerMessage msg, final String name) {
|
|
||||||
// if (!isConnected()) return false;
|
|
||||||
//if (isBusy()) return false;
|
|
||||||
log("asyncSinglecommand called: " + name);
|
|
||||||
final EventInsightCallback event = new EventInsightCallback();
|
|
||||||
new Thread() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
log("asyncSingleCommand thread");
|
|
||||||
final SingleMessageTaskRunner singleMessageTaskRunner = new SingleMessageTaskRunner(connector.getServiceConnector(), msg);
|
|
||||||
try {
|
|
||||||
singleMessageTaskRunner.fetch(new TaskRunner.ResultCallback() {
|
|
||||||
@Override
|
|
||||||
public void onResult(Object o) {
|
|
||||||
lastDataTime = new Date();
|
|
||||||
log(name + " success");
|
|
||||||
event.response_object = o;
|
|
||||||
if (o instanceof BolusMessage) {
|
|
||||||
event.response_id = ((BolusMessage) o).getBolusId();
|
|
||||||
}
|
|
||||||
event.success = true;
|
|
||||||
pushCallbackEvent(event);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onError(Exception e) {
|
|
||||||
log(name + " error");
|
|
||||||
event.message = e.getMessage();
|
|
||||||
pushCallbackEvent(event);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
} catch (Exception e) {
|
|
||||||
log("EXCEPTION" + e.toString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}.start();
|
|
||||||
return event.request_uuid;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private synchronized UUID aSyncTaskRunner(final TaskRunner task, final String name) {
|
private void fetchSingleMessage(AppLayerMessage message) throws Exception {
|
||||||
// if (!isConnected()) return false;
|
fetchSingleMessage(message, AppLayerMessage.class);
|
||||||
//if (isBusy()) return false;
|
}
|
||||||
log("asyncTaskRunner called: " + name);
|
|
||||||
final EventInsightCallback event = new EventInsightCallback();
|
|
||||||
new Thread() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
log("asyncTaskRunner thread");
|
|
||||||
try {
|
|
||||||
task.fetch(new TaskRunner.ResultCallback() {
|
|
||||||
@Override
|
|
||||||
public void onResult(Object o) {
|
|
||||||
lastDataTime = new Date();
|
|
||||||
log(name + " success");
|
|
||||||
event.response_object = o;
|
|
||||||
event.success = true;
|
|
||||||
pushCallbackEvent(event);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
private <T> T fetchTaskRunner(TaskRunner taskRunner, Class<T> resultType) throws Exception {
|
||||||
public void onError(Exception e) {
|
try {
|
||||||
log(name + " error");
|
T result = (T) taskRunner.fetchAndWaitUsingLatch(BUSY_WAIT_TIME);
|
||||||
event.message = e.getMessage();
|
lastDataTime = new Date();
|
||||||
pushCallbackEvent(event);
|
return result;
|
||||||
}
|
} catch (Exception e) {
|
||||||
});
|
log("Error while fetching " + taskRunner.getClass().getSimpleName() + ": " + e.getClass().getSimpleName());
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
} catch (Exception e) {
|
private <T extends AppLayerMessage> T fetchSingleMessage(AppLayerMessage message, Class<T> resultType) throws Exception {
|
||||||
log("EXCEPTION" + e.toString());
|
try {
|
||||||
}
|
T result = (T) new SingleMessageTaskRunner(connector.getServiceConnector(), message).fetchAndWaitUsingLatch(BUSY_WAIT_TIME);
|
||||||
}
|
lastDataTime = new Date();
|
||||||
}.start();
|
return result;
|
||||||
return event.request_uuid;
|
} catch (Exception e) {
|
||||||
|
log("Error while fetching " + message.getClass().getSimpleName() + ": " + e.getClass().getSimpleName());
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1016,14 +871,6 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
|
|
||||||
// Constraints
|
// Constraints
|
||||||
|
|
||||||
@Override
|
|
||||||
public Constraint<Double> applyBasalConstraints(Constraint<Double> absoluteRate, Profile profile) {
|
|
||||||
if (statusResult != null) {
|
|
||||||
absoluteRate.setIfSmaller(statusResult.maximumBasalAmount, String.format(MainApp.gs(R.string.limitingbasalratio), statusResult.maximumBasalAmount, MainApp.gs(R.string.pumplimit)), this);
|
|
||||||
}
|
|
||||||
return absoluteRate;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Constraint<Integer> applyBasalPercentConstraints(Constraint<Integer> percentRate, Profile profile) {
|
public Constraint<Integer> applyBasalPercentConstraints(Constraint<Integer> percentRate, Profile profile) {
|
||||||
percentRate.setIfGreater(0, String.format(MainApp.gs(R.string.limitingpercentrate), 0, MainApp.gs(R.string.itmustbepositivevalue)), this);
|
percentRate.setIfGreater(0, String.format(MainApp.gs(R.string.limitingpercentrate), 0, MainApp.gs(R.string.itmustbepositivevalue)), this);
|
||||||
|
@ -1034,8 +881,10 @@ public class InsightPlugin extends PluginBase implements PumpInterface, Constrai
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Constraint<Double> applyBolusConstraints(Constraint<Double> insulin) {
|
public Constraint<Double> applyBolusConstraints(Constraint<Double> insulin) {
|
||||||
if (statusResult != null)
|
if (statusResult != null) {
|
||||||
insulin.setIfSmaller(statusResult.maximumBolusAmount, String.format(MainApp.gs(R.string.limitingbolus), statusResult.maximumBolusAmount, MainApp.gs(R.string.pumplimit)), this);
|
insulin.setIfSmaller(statusResult.maximumBolusAmount, String.format(MainApp.gs(R.string.limitingbolus), statusResult.maximumBolusAmount, MainApp.gs(R.string.pumplimit)), this);
|
||||||
|
insulin.setIfGreater(statusResult.minimumBolusAmount, String.format(MainApp.gs(R.string.limitingbolus), statusResult.maximumBolusAmount, MainApp.gs(R.string.pumplimit)), this);
|
||||||
|
}
|
||||||
return insulin;
|
return insulin;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,50 +0,0 @@
|
||||||
package info.nightscout.androidaps.plugins.PumpInsight;
|
|
||||||
|
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.events.EventInsightCallback;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Created by jamorham on 01/02/2018.
|
|
||||||
*
|
|
||||||
* Encapsulates results from commands
|
|
||||||
*/
|
|
||||||
|
|
||||||
class Mstatus {
|
|
||||||
|
|
||||||
Cstatus cstatus = Cstatus.UNKNOWN;
|
|
||||||
EventInsightCallback event;
|
|
||||||
|
|
||||||
// comment field preparation for results
|
|
||||||
String getCommandComment() {
|
|
||||||
if (success()) {
|
|
||||||
return "OK";
|
|
||||||
} else {
|
|
||||||
return (event == null) ? "EVENT DATA IS NULL - ERROR OR FIREWALL ENABLED?" : event.message;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean success() {
|
|
||||||
return cstatus.success();
|
|
||||||
}
|
|
||||||
|
|
||||||
int getResponseID() {
|
|
||||||
if (success()) {
|
|
||||||
return event.response_id;
|
|
||||||
} else {
|
|
||||||
return -2; // invalid
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Object getResponseObject() {
|
|
||||||
if (success()) {
|
|
||||||
return event.response_object;
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
return cstatus + " " + event;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -0,0 +1,66 @@
|
||||||
|
package info.nightscout.androidaps.plugins.PumpInsight.connector;
|
||||||
|
|
||||||
|
import sugar.free.sightparser.applayer.descriptors.ActiveBolusType;
|
||||||
|
import sugar.free.sightparser.applayer.descriptors.MessagePriority;
|
||||||
|
import sugar.free.sightparser.applayer.descriptors.alerts.Warning38BolusCancelled;
|
||||||
|
import sugar.free.sightparser.applayer.messages.AppLayerMessage;
|
||||||
|
import sugar.free.sightparser.applayer.messages.remote_control.CancelBolusMessage;
|
||||||
|
import sugar.free.sightparser.applayer.messages.remote_control.DismissAlertMessage;
|
||||||
|
import sugar.free.sightparser.applayer.messages.status.ActiveAlertMessage;
|
||||||
|
import sugar.free.sightparser.applayer.messages.status.ActiveBolusesMessage;
|
||||||
|
import sugar.free.sightparser.handling.SightServiceConnector;
|
||||||
|
import sugar.free.sightparser.handling.TaskRunner;
|
||||||
|
|
||||||
|
// by Tebbe Ubben
|
||||||
|
|
||||||
|
public class CancelBolusSilentlyTaskRunner extends TaskRunner {
|
||||||
|
|
||||||
|
private ActiveBolusType bolusType;
|
||||||
|
private long cancelledAt;
|
||||||
|
private int bolusId;
|
||||||
|
|
||||||
|
public CancelBolusSilentlyTaskRunner(SightServiceConnector serviceConnector, ActiveBolusType bolusType) {
|
||||||
|
super(serviceConnector);
|
||||||
|
this.bolusType = bolusType;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected AppLayerMessage run(AppLayerMessage message) throws Exception {
|
||||||
|
if (message == null) return new ActiveBolusesMessage();
|
||||||
|
else if (message instanceof ActiveBolusesMessage) {
|
||||||
|
ActiveBolusesMessage bolusesMessage = (ActiveBolusesMessage) message;
|
||||||
|
CancelBolusMessage cancelBolusMessage = new CancelBolusMessage();
|
||||||
|
if (bolusesMessage.getBolus1().getBolusType() == bolusType)
|
||||||
|
bolusId = bolusesMessage.getBolus1().getBolusID();
|
||||||
|
else if (bolusesMessage.getBolus2().getBolusType() == bolusType)
|
||||||
|
bolusId = bolusesMessage.getBolus2().getBolusID();
|
||||||
|
else if (bolusesMessage.getBolus3().getBolusType() == bolusType)
|
||||||
|
bolusId = bolusesMessage.getBolus3().getBolusID();
|
||||||
|
else finish(null);
|
||||||
|
cancelBolusMessage.setBolusId(bolusId);
|
||||||
|
return cancelBolusMessage;
|
||||||
|
} else if (message instanceof CancelBolusMessage) {
|
||||||
|
cancelledAt = System.currentTimeMillis();
|
||||||
|
ActiveAlertMessage activeAlertMessage = new ActiveAlertMessage();
|
||||||
|
activeAlertMessage.setMessagePriority(MessagePriority.HIGHER);
|
||||||
|
return activeAlertMessage;
|
||||||
|
} else if (message instanceof ActiveAlertMessage) {
|
||||||
|
ActiveAlertMessage activeAlertMessage = (ActiveAlertMessage) message;
|
||||||
|
if (activeAlertMessage.getAlert() == null) {
|
||||||
|
if (System.currentTimeMillis() - cancelledAt >= 10000) finish(bolusId);
|
||||||
|
else {
|
||||||
|
ActiveAlertMessage activeAlertMessage2 = new ActiveAlertMessage();
|
||||||
|
activeAlertMessage2.setMessagePriority(MessagePriority.HIGHER);
|
||||||
|
return activeAlertMessage2;
|
||||||
|
}
|
||||||
|
} else if (!(activeAlertMessage.getAlert() instanceof Warning38BolusCancelled)) finish(bolusId);
|
||||||
|
else {
|
||||||
|
DismissAlertMessage dismissAlertMessage = new DismissAlertMessage();
|
||||||
|
dismissAlertMessage.setAlertID(activeAlertMessage.getAlertID());
|
||||||
|
dismissAlertMessage.setMessagePriority(MessagePriority.HIGHER);
|
||||||
|
return dismissAlertMessage;
|
||||||
|
}
|
||||||
|
} else if (message instanceof DismissAlertMessage) finish(bolusId);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,38 +0,0 @@
|
||||||
package info.nightscout.androidaps.plugins.PumpInsight.connector;
|
|
||||||
|
|
||||||
import sugar.free.sightparser.applayer.messages.AppLayerMessage;
|
|
||||||
import sugar.free.sightparser.applayer.descriptors.ActiveBolusType;
|
|
||||||
import sugar.free.sightparser.applayer.messages.remote_control.CancelBolusMessage;
|
|
||||||
import sugar.free.sightparser.applayer.messages.status.ActiveBolusesMessage;
|
|
||||||
import sugar.free.sightparser.handling.SightServiceConnector;
|
|
||||||
import sugar.free.sightparser.handling.TaskRunner;
|
|
||||||
|
|
||||||
// by Tebbe Ubben
|
|
||||||
|
|
||||||
public class CancelBolusTaskRunner extends TaskRunner {
|
|
||||||
|
|
||||||
private ActiveBolusType bolusType;
|
|
||||||
|
|
||||||
public CancelBolusTaskRunner(SightServiceConnector serviceConnector, ActiveBolusType bolusType) {
|
|
||||||
super(serviceConnector);
|
|
||||||
this.bolusType = bolusType;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected AppLayerMessage run(AppLayerMessage message) throws Exception {
|
|
||||||
if (message == null) return new ActiveBolusesMessage();
|
|
||||||
else if (message instanceof ActiveBolusesMessage) {
|
|
||||||
ActiveBolusesMessage bolusesMessage = (ActiveBolusesMessage) message;
|
|
||||||
CancelBolusMessage cancelBolusMessage = new CancelBolusMessage();
|
|
||||||
if (bolusesMessage.getBolus1().getBolusType() == bolusType)
|
|
||||||
cancelBolusMessage.setBolusId(bolusesMessage.getBolus1().getBolusID());
|
|
||||||
else if (bolusesMessage.getBolus2().getBolusType() == bolusType)
|
|
||||||
cancelBolusMessage.setBolusId(bolusesMessage.getBolus2().getBolusID());
|
|
||||||
else if (bolusesMessage.getBolus3().getBolusType() == bolusType)
|
|
||||||
cancelBolusMessage.setBolusId(bolusesMessage.getBolus3().getBolusID());
|
|
||||||
else finish(null);
|
|
||||||
return cancelBolusMessage;
|
|
||||||
} else if (message instanceof CancelBolusMessage) finish(null);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -0,0 +1,50 @@
|
||||||
|
package info.nightscout.androidaps.plugins.PumpInsight.connector;
|
||||||
|
|
||||||
|
import sugar.free.sightparser.applayer.descriptors.MessagePriority;
|
||||||
|
import sugar.free.sightparser.applayer.descriptors.alerts.Warning36TBRCancelled;
|
||||||
|
import sugar.free.sightparser.applayer.messages.AppLayerMessage;
|
||||||
|
import sugar.free.sightparser.applayer.messages.remote_control.CancelTBRMessage;
|
||||||
|
import sugar.free.sightparser.applayer.messages.remote_control.DismissAlertMessage;
|
||||||
|
import sugar.free.sightparser.applayer.messages.status.ActiveAlertMessage;
|
||||||
|
import sugar.free.sightparser.applayer.messages.status.CurrentTBRMessage;
|
||||||
|
import sugar.free.sightparser.handling.SightServiceConnector;
|
||||||
|
import sugar.free.sightparser.handling.TaskRunner;
|
||||||
|
|
||||||
|
public class CancelTBRSilentlyTaskRunner extends TaskRunner {
|
||||||
|
|
||||||
|
private long cancelledAt;
|
||||||
|
|
||||||
|
public CancelTBRSilentlyTaskRunner(SightServiceConnector serviceConnector) {
|
||||||
|
super(serviceConnector);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected AppLayerMessage run(AppLayerMessage message) throws Exception {
|
||||||
|
if (message == null) return new CurrentTBRMessage();
|
||||||
|
else if (message instanceof CurrentTBRMessage) {
|
||||||
|
if (((CurrentTBRMessage) message).getPercentage() == 100) finish(false);
|
||||||
|
else return new CancelTBRMessage();
|
||||||
|
} else if (message instanceof CancelTBRMessage) {
|
||||||
|
ActiveAlertMessage activeAlertMessage = new ActiveAlertMessage();
|
||||||
|
activeAlertMessage.setMessagePriority(MessagePriority.HIGHER);
|
||||||
|
return activeAlertMessage;
|
||||||
|
} else if (message instanceof ActiveAlertMessage) {
|
||||||
|
ActiveAlertMessage activeAlertMessage = (ActiveAlertMessage) message;
|
||||||
|
if (activeAlertMessage.getAlert() == null) {
|
||||||
|
if (System.currentTimeMillis() - cancelledAt >= 10000) finish(true);
|
||||||
|
else {
|
||||||
|
ActiveAlertMessage activeAlertMessage2 = new ActiveAlertMessage();
|
||||||
|
activeAlertMessage2.setMessagePriority(MessagePriority.HIGHER);
|
||||||
|
return activeAlertMessage2;
|
||||||
|
}
|
||||||
|
} else if (!(activeAlertMessage.getAlert() instanceof Warning36TBRCancelled)) finish(true);
|
||||||
|
else {
|
||||||
|
DismissAlertMessage dismissAlertMessage = new DismissAlertMessage();
|
||||||
|
dismissAlertMessage.setAlertID(activeAlertMessage.getAlertID());
|
||||||
|
dismissAlertMessage.setMessagePriority(MessagePriority.HIGHER);
|
||||||
|
return dismissAlertMessage;
|
||||||
|
}
|
||||||
|
} else if (message instanceof DismissAlertMessage) finish(true);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
|
@ -25,25 +25,15 @@ public class SetTBRTaskRunner extends TaskRunner {
|
||||||
if (message == null) return new CurrentTBRMessage();
|
if (message == null) return new CurrentTBRMessage();
|
||||||
else if (message instanceof CurrentTBRMessage) {
|
else if (message instanceof CurrentTBRMessage) {
|
||||||
if (((CurrentTBRMessage) message).getPercentage() == 100) {
|
if (((CurrentTBRMessage) message).getPercentage() == 100) {
|
||||||
if (amount == 100) finish(amount);
|
SetTBRMessage setTBRMessage = new SetTBRMessage();
|
||||||
else {
|
setTBRMessage.setDuration(duration);
|
||||||
SetTBRMessage setTBRMessage = new SetTBRMessage();
|
setTBRMessage.setAmount(amount);
|
||||||
setTBRMessage.setDuration(duration);
|
return setTBRMessage;
|
||||||
setTBRMessage.setAmount(amount);
|
|
||||||
return setTBRMessage;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
if (amount == 100) {
|
ChangeTBRMessage changeTBRMessage = new ChangeTBRMessage();
|
||||||
ChangeTBRMessage changeTBRMessage = new ChangeTBRMessage();
|
changeTBRMessage.setDuration(duration);
|
||||||
changeTBRMessage.setDuration(1);
|
changeTBRMessage.setAmount(amount);
|
||||||
changeTBRMessage.setAmount(90);
|
return changeTBRMessage;
|
||||||
return changeTBRMessage;
|
|
||||||
} else {
|
|
||||||
ChangeTBRMessage changeTBRMessage = new ChangeTBRMessage();
|
|
||||||
changeTBRMessage.setDuration(duration);
|
|
||||||
changeTBRMessage.setAmount(amount);
|
|
||||||
return changeTBRMessage;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else if (message instanceof SetTBRMessage) finish(amount);
|
} else if (message instanceof SetTBRMessage) finish(amount);
|
||||||
return null;
|
return null;
|
||||||
|
|
|
@ -12,6 +12,8 @@ import sugar.free.sightparser.applayer.descriptors.configuration_blocks.BRProfil
|
||||||
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.BRProfile5Block;
|
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.BRProfile5Block;
|
||||||
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.BRProfileBlock;
|
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.BRProfileBlock;
|
||||||
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.ConfigurationBlock;
|
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.ConfigurationBlock;
|
||||||
|
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.FactoryMinBRAmountBlock;
|
||||||
|
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.FactoryMinBolusAmountBlock;
|
||||||
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.MaxBRAmountBlock;
|
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.MaxBRAmountBlock;
|
||||||
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.MaxBolusAmountBlock;
|
import sugar.free.sightparser.applayer.descriptors.configuration_blocks.MaxBolusAmountBlock;
|
||||||
import sugar.free.sightparser.applayer.messages.AppLayerMessage;
|
import sugar.free.sightparser.applayer.messages.AppLayerMessage;
|
||||||
|
@ -102,6 +104,16 @@ public class StatusTaskRunner extends TaskRunner {
|
||||||
return readMessage;
|
return readMessage;
|
||||||
} else if (configurationBlock instanceof MaxBRAmountBlock) {
|
} else if (configurationBlock instanceof MaxBRAmountBlock) {
|
||||||
result.maximumBasalAmount = ((MaxBRAmountBlock) configurationBlock).getMaximumAmount();
|
result.maximumBasalAmount = ((MaxBRAmountBlock) configurationBlock).getMaximumAmount();
|
||||||
|
ReadConfigurationBlockMessage readMessage = new ReadConfigurationBlockMessage();
|
||||||
|
readMessage.setConfigurationBlockID(FactoryMinBRAmountBlock.ID);
|
||||||
|
return readMessage;
|
||||||
|
} else if (configurationBlock instanceof FactoryMinBRAmountBlock) {
|
||||||
|
result.minimumBasalAmount = ((FactoryMinBRAmountBlock) configurationBlock).getMinimumAmount();
|
||||||
|
ReadConfigurationBlockMessage readMessage = new ReadConfigurationBlockMessage();
|
||||||
|
readMessage.setConfigurationBlockID(FactoryMinBolusAmountBlock.ID);
|
||||||
|
return readMessage;
|
||||||
|
} else if (configurationBlock instanceof FactoryMinBolusAmountBlock) {
|
||||||
|
result.minimumBolusAmount = ((FactoryMinBolusAmountBlock) configurationBlock).getMinimumAmount();
|
||||||
finish(result);
|
finish(result);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -122,5 +134,7 @@ public class StatusTaskRunner extends TaskRunner {
|
||||||
public List<BRProfileBlock.ProfileBlock> basalProfile;
|
public List<BRProfileBlock.ProfileBlock> basalProfile;
|
||||||
public double maximumBolusAmount;
|
public double maximumBolusAmount;
|
||||||
public double maximumBasalAmount;
|
public double maximumBasalAmount;
|
||||||
|
public double minimumBolusAmount;
|
||||||
|
public double minimumBasalAmount;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -58,13 +58,8 @@ class HistoryIntentAdapter {
|
||||||
|
|
||||||
final long record_unique_id = getRecordUniqueID(pump_serial_number, pump_record_id);
|
final long record_unique_id = getRecordUniqueID(pump_serial_number, pump_record_id);
|
||||||
|
|
||||||
// other sanity checks
|
log("Creating TBR record: " + pump_tbr_percent + "% " + pump_tbr_duration + "m" + " id:" + record_unique_id);
|
||||||
if ((pump_tbr_percent == 90) && (pump_tbr_duration <= 1)) {
|
logAdapter.createTBRrecord(start_time, pump_tbr_percent, pump_tbr_duration, record_unique_id);
|
||||||
log("Not creating TBR record for faux cancel");
|
|
||||||
} else {
|
|
||||||
log("Creating TBR record: " + pump_tbr_percent + "% " + pump_tbr_duration + "m" + " id:" + record_unique_id);
|
|
||||||
logAdapter.createTBRrecord(start_time, pump_tbr_percent, pump_tbr_duration, record_unique_id);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void processDeliveredBolusIntent(Intent intent) {
|
void processDeliveredBolusIntent(Intent intent) {
|
||||||
|
@ -160,8 +155,10 @@ class HistoryIntentAdapter {
|
||||||
if (SP.getBoolean("insight_automatic_careportal_events", false)) {
|
if (SP.getBoolean("insight_automatic_careportal_events", false)) {
|
||||||
Date date = getDateExtra(intent, HistoryBroadcast.EXTRA_EVENT_TIME);
|
Date date = getDateExtra(intent, HistoryBroadcast.EXTRA_EVENT_TIME);
|
||||||
String alertType = intent.getStringExtra(HistoryBroadcast.EXTRA_ALERT_TYPE);
|
String alertType = intent.getStringExtra(HistoryBroadcast.EXTRA_ALERT_TYPE);
|
||||||
|
int alertText = getAlertText(alertType);
|
||||||
|
if (alertText == 0) return;
|
||||||
if (MainApp.getDbHelper().getCareportalEventFromTimestamp(date.getTime()) != null) return;
|
if (MainApp.getDbHelper().getCareportalEventFromTimestamp(date.getTime()) != null) return;
|
||||||
logNote(date, MainApp.gs(getAlertText(alertType)));
|
logNote(date, MainApp.gs(alertText));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -231,8 +228,8 @@ class HistoryIntentAdapter {
|
||||||
if (type.equals("Warning32BatteryLow")) return R.string.alert_w32;
|
if (type.equals("Warning32BatteryLow")) return R.string.alert_w32;
|
||||||
if (type.equals("Warning33InvalidDateTime")) return R.string.alert_w33;
|
if (type.equals("Warning33InvalidDateTime")) return R.string.alert_w33;
|
||||||
if (type.equals("Warning34EndOfWarranty")) return R.string.alert_w34;
|
if (type.equals("Warning34EndOfWarranty")) return R.string.alert_w34;
|
||||||
if (type.equals("Warning36TBRCancelled")) return R.string.alert_w36;
|
if (type.equals("Warning36TBRCancelled")) return 0;
|
||||||
if (type.equals("Warning38BolusCancelled")) return R.string.alert_w38;
|
if (type.equals("Warning38BolusCancelled")) return 0;
|
||||||
if (type.equals("Warning39LoantimeWarning")) return R.string.alert_w39;
|
if (type.equals("Warning39LoantimeWarning")) return R.string.alert_w39;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,322 +0,0 @@
|
||||||
package info.nightscout.androidaps.tabs;/*
|
|
||||||
* Copyright 2014 Google Inc. All rights reserved.
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.graphics.Typeface;
|
|
||||||
import android.support.v4.view.PagerAdapter;
|
|
||||||
import android.support.v4.view.ViewPager;
|
|
||||||
import android.util.AttributeSet;
|
|
||||||
import android.util.SparseArray;
|
|
||||||
import android.util.TypedValue;
|
|
||||||
import android.view.Gravity;
|
|
||||||
import android.view.LayoutInflater;
|
|
||||||
import android.view.View;
|
|
||||||
import android.view.ViewGroup;
|
|
||||||
import android.widget.HorizontalScrollView;
|
|
||||||
import android.widget.LinearLayout;
|
|
||||||
import android.widget.TextView;
|
|
||||||
|
|
||||||
import info.nightscout.androidaps.R;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* To be used with ViewPager to provide a tab indicator component which give constant feedback as to
|
|
||||||
* the user's scroll progress.
|
|
||||||
* <p>
|
|
||||||
* To use the component, simply add it to your view hierarchy. Then in your
|
|
||||||
* {@link android.app.Activity} or {@link android.support.v4.app.Fragment} call
|
|
||||||
* {@link #setViewPager(ViewPager)} providing it the ViewPager this layout is being used for.
|
|
||||||
* <p>
|
|
||||||
* The colors can be customized in two ways. The first and simplest is to provide an array of colors
|
|
||||||
* via {@link #setSelectedIndicatorColors(int...)}. The
|
|
||||||
* alternative is via the {@link TabColorizer} interface which provides you complete control over
|
|
||||||
* which color is used for any individual position.
|
|
||||||
* <p>
|
|
||||||
* The views used as tabs can be customized by calling {@link #setCustomTabView(int, int)},
|
|
||||||
* providing the layout ID of your custom layout.
|
|
||||||
*/
|
|
||||||
public class SlidingTabLayout extends HorizontalScrollView {
|
|
||||||
/**
|
|
||||||
* Allows complete control over the colors drawn in the tab layout. Set with
|
|
||||||
* {@link #setCustomTabColorizer(TabColorizer)}.
|
|
||||||
*/
|
|
||||||
public interface TabColorizer {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return return the color of the indicator used when {@code position} is selected.
|
|
||||||
*/
|
|
||||||
int getIndicatorColor(int position);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private static final int TITLE_OFFSET_DIPS = 24;
|
|
||||||
private static final int TAB_VIEW_PADDING_DIPS = 9;
|
|
||||||
private static final int TAB_VIEW_TEXT_SIZE_SP = 12;
|
|
||||||
|
|
||||||
private int mTitleOffset;
|
|
||||||
|
|
||||||
private int mTabViewLayoutId;
|
|
||||||
private int mTabViewTextViewId;
|
|
||||||
private boolean mDistributeEvenly;
|
|
||||||
|
|
||||||
private ViewPager mViewPager;
|
|
||||||
private SparseArray<String> mContentDescriptions = new SparseArray<String>();
|
|
||||||
private ViewPager.OnPageChangeListener mViewPagerPageChangeListener;
|
|
||||||
|
|
||||||
private final SlidingTabStrip mTabStrip;
|
|
||||||
|
|
||||||
public SlidingTabLayout(Context context) {
|
|
||||||
this(context, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public SlidingTabLayout(Context context, AttributeSet attrs) {
|
|
||||||
this(context, attrs, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
public SlidingTabLayout(Context context, AttributeSet attrs, int defStyle) {
|
|
||||||
super(context, attrs, defStyle);
|
|
||||||
|
|
||||||
// Disable the Scroll Bar
|
|
||||||
setHorizontalScrollBarEnabled(false);
|
|
||||||
// Make sure that the Tab Strips fills this View
|
|
||||||
setFillViewport(true);
|
|
||||||
|
|
||||||
mTitleOffset = (int) (TITLE_OFFSET_DIPS * getResources().getDisplayMetrics().density);
|
|
||||||
|
|
||||||
mTabStrip = new SlidingTabStrip(context);
|
|
||||||
addView(mTabStrip, LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT);
|
|
||||||
setBackgroundColor(context.getResources().getColor(R.color.tabBgColor));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the custom {@link TabColorizer} to be used.
|
|
||||||
*
|
|
||||||
* If you only require simple custmisation then you can use
|
|
||||||
* {@link #setSelectedIndicatorColors(int...)} to achieve
|
|
||||||
* similar effects.
|
|
||||||
*/
|
|
||||||
public void setCustomTabColorizer(TabColorizer tabColorizer) {
|
|
||||||
mTabStrip.setCustomTabColorizer(tabColorizer);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setDistributeEvenly(boolean distributeEvenly) {
|
|
||||||
mDistributeEvenly = distributeEvenly;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets the colors to be used for indicating the selected tab. These colors are treated as a
|
|
||||||
* circular array. Providing one color will mean that all tabs are indicated with the same color.
|
|
||||||
*/
|
|
||||||
public void setSelectedIndicatorColors(int... colors) {
|
|
||||||
mTabStrip.setSelectedIndicatorColors(colors);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the {@link ViewPager.OnPageChangeListener}. When using {@link SlidingTabLayout} you are
|
|
||||||
* required to set any {@link ViewPager.OnPageChangeListener} through this method. This is so
|
|
||||||
* that the layout can update it's scroll position correctly.
|
|
||||||
*
|
|
||||||
* @see ViewPager#setOnPageChangeListener(ViewPager.OnPageChangeListener)
|
|
||||||
*/
|
|
||||||
public void setOnPageChangeListener(ViewPager.OnPageChangeListener listener) {
|
|
||||||
mViewPagerPageChangeListener = listener;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the custom layout to be inflated for the tab views.
|
|
||||||
*
|
|
||||||
* @param layoutResId Layout id to be inflated
|
|
||||||
* @param textViewId id of the {@link TextView} in the inflated view
|
|
||||||
*/
|
|
||||||
public void setCustomTabView(int layoutResId, int textViewId) {
|
|
||||||
mTabViewLayoutId = layoutResId;
|
|
||||||
mTabViewTextViewId = textViewId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets the associated view pager. Note that the assumption here is that the pager content
|
|
||||||
* (number of tabs and tab titles) does not change after this call has been made.
|
|
||||||
*/
|
|
||||||
public void setViewPager(ViewPager viewPager) {
|
|
||||||
mTabStrip.removeAllViews();
|
|
||||||
|
|
||||||
mViewPager = viewPager;
|
|
||||||
if (viewPager != null) {
|
|
||||||
viewPager.setOnPageChangeListener(new InternalViewPagerListener());
|
|
||||||
populateTabStrip();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a default view to be used for tabs. This is called if a custom tab view is not set via
|
|
||||||
* {@link #setCustomTabView(int, int)}.
|
|
||||||
*/
|
|
||||||
protected TextView createDefaultTabView(Context context) {
|
|
||||||
TextView textView = new TextView(context);
|
|
||||||
textView.setGravity(Gravity.CENTER);
|
|
||||||
textView.setTextSize(TypedValue.COMPLEX_UNIT_SP, TAB_VIEW_TEXT_SIZE_SP);
|
|
||||||
textView.setTypeface(Typeface.DEFAULT_BOLD);
|
|
||||||
textView.setLayoutParams(new LinearLayout.LayoutParams(
|
|
||||||
ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT));
|
|
||||||
|
|
||||||
TypedValue outValue = new TypedValue();
|
|
||||||
getContext().getTheme().resolveAttribute(android.R.attr.selectableItemBackground,
|
|
||||||
outValue, true);
|
|
||||||
textView.setBackgroundResource(outValue.resourceId);
|
|
||||||
textView.setAllCaps(true);
|
|
||||||
|
|
||||||
int padding = (int) (TAB_VIEW_PADDING_DIPS * getResources().getDisplayMetrics().density);
|
|
||||||
textView.setPadding(padding, padding, padding, padding);
|
|
||||||
|
|
||||||
return textView;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void populateTabStrip() {
|
|
||||||
final PagerAdapter adapter = mViewPager.getAdapter();
|
|
||||||
final View.OnClickListener tabClickListener = new TabClickListener();
|
|
||||||
|
|
||||||
for (int i = 0; i < adapter.getCount(); i++) {
|
|
||||||
View tabView = null;
|
|
||||||
TextView tabTitleView = null;
|
|
||||||
|
|
||||||
if (mTabViewLayoutId != 0) {
|
|
||||||
// If there is a custom tab view layout id set, try and inflate it
|
|
||||||
tabView = LayoutInflater.from(getContext()).inflate(mTabViewLayoutId, mTabStrip,
|
|
||||||
false);
|
|
||||||
tabTitleView = (TextView) tabView.findViewById(mTabViewTextViewId);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tabView == null) {
|
|
||||||
tabView = createDefaultTabView(getContext());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tabTitleView == null && TextView.class.isInstance(tabView)) {
|
|
||||||
tabTitleView = (TextView) tabView;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mDistributeEvenly) {
|
|
||||||
LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams) tabView.getLayoutParams();
|
|
||||||
lp.width = 0;
|
|
||||||
lp.weight = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
tabTitleView.setText(adapter.getPageTitle(i));
|
|
||||||
tabView.setOnClickListener(tabClickListener);
|
|
||||||
String desc = mContentDescriptions.get(i, null);
|
|
||||||
if (desc != null) {
|
|
||||||
tabView.setContentDescription(desc);
|
|
||||||
}
|
|
||||||
|
|
||||||
mTabStrip.addView(tabView);
|
|
||||||
if (i == mViewPager.getCurrentItem()) {
|
|
||||||
tabView.setSelected(true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setContentDescription(int i, String desc) {
|
|
||||||
mContentDescriptions.put(i, desc);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onAttachedToWindow() {
|
|
||||||
super.onAttachedToWindow();
|
|
||||||
|
|
||||||
if (mViewPager != null) {
|
|
||||||
scrollToTab(mViewPager.getCurrentItem(), 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void scrollToTab(int tabIndex, int positionOffset) {
|
|
||||||
final int tabStripChildCount = mTabStrip.getChildCount();
|
|
||||||
if (tabStripChildCount == 0 || tabIndex < 0 || tabIndex >= tabStripChildCount) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
View selectedChild = mTabStrip.getChildAt(tabIndex);
|
|
||||||
if (selectedChild != null) {
|
|
||||||
int targetScrollX = selectedChild.getLeft() + positionOffset;
|
|
||||||
|
|
||||||
if (tabIndex > 0 || positionOffset > 0) {
|
|
||||||
// If we're not at the first child and are mid-scroll, make sure we obey the offset
|
|
||||||
targetScrollX -= mTitleOffset;
|
|
||||||
}
|
|
||||||
|
|
||||||
scrollTo(targetScrollX, 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private class InternalViewPagerListener implements ViewPager.OnPageChangeListener {
|
|
||||||
private int mScrollState;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
|
|
||||||
int tabStripChildCount = mTabStrip.getChildCount();
|
|
||||||
if ((tabStripChildCount == 0) || (position < 0) || (position >= tabStripChildCount)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
mTabStrip.onViewPagerPageChanged(position, positionOffset);
|
|
||||||
|
|
||||||
View selectedTitle = mTabStrip.getChildAt(position);
|
|
||||||
int extraOffset = (selectedTitle != null)
|
|
||||||
? (int) (positionOffset * selectedTitle.getWidth())
|
|
||||||
: 0;
|
|
||||||
scrollToTab(position, extraOffset);
|
|
||||||
|
|
||||||
if (mViewPagerPageChangeListener != null) {
|
|
||||||
mViewPagerPageChangeListener.onPageScrolled(position, positionOffset,
|
|
||||||
positionOffsetPixels);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onPageScrollStateChanged(int state) {
|
|
||||||
mScrollState = state;
|
|
||||||
|
|
||||||
if (mViewPagerPageChangeListener != null) {
|
|
||||||
mViewPagerPageChangeListener.onPageScrollStateChanged(state);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onPageSelected(int position) {
|
|
||||||
if (mScrollState == ViewPager.SCROLL_STATE_IDLE) {
|
|
||||||
mTabStrip.onViewPagerPageChanged(position, 0f);
|
|
||||||
scrollToTab(position, 0);
|
|
||||||
}
|
|
||||||
for (int i = 0; i < mTabStrip.getChildCount(); i++) {
|
|
||||||
mTabStrip.getChildAt(i).setSelected(position == i);
|
|
||||||
}
|
|
||||||
if (mViewPagerPageChangeListener != null) {
|
|
||||||
mViewPagerPageChangeListener.onPageSelected(position);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private class TabClickListener implements View.OnClickListener {
|
|
||||||
@Override
|
|
||||||
public void onClick(View v) {
|
|
||||||
for (int i = 0; i < mTabStrip.getChildCount(); i++) {
|
|
||||||
if (v == mTabStrip.getChildAt(i)) {
|
|
||||||
mViewPager.setCurrentItem(i);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,165 +0,0 @@
|
||||||
package info.nightscout.androidaps.tabs;/*
|
|
||||||
* Copyright 2014 Google Inc. All rights reserved.
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.graphics.Canvas;
|
|
||||||
import android.graphics.Color;
|
|
||||||
import android.graphics.Paint;
|
|
||||||
import android.util.AttributeSet;
|
|
||||||
import android.util.TypedValue;
|
|
||||||
import android.view.View;
|
|
||||||
import android.widget.LinearLayout;
|
|
||||||
|
|
||||||
import info.nightscout.androidaps.MainApp;
|
|
||||||
import info.nightscout.androidaps.R;
|
|
||||||
|
|
||||||
class SlidingTabStrip extends LinearLayout {
|
|
||||||
|
|
||||||
private static final int DEFAULT_BOTTOM_BORDER_THICKNESS_DIPS = 0;
|
|
||||||
private static final byte DEFAULT_BOTTOM_BORDER_COLOR_ALPHA = 0x26;
|
|
||||||
private static final int SELECTED_INDICATOR_THICKNESS_DIPS = 3;
|
|
||||||
|
|
||||||
private final int mBottomBorderThickness;
|
|
||||||
private final Paint mBottomBorderPaint;
|
|
||||||
|
|
||||||
private final int mSelectedIndicatorThickness;
|
|
||||||
private final Paint mSelectedIndicatorPaint;
|
|
||||||
|
|
||||||
private int mSelectedPosition;
|
|
||||||
private float mSelectionOffset;
|
|
||||||
|
|
||||||
private SlidingTabLayout.TabColorizer mCustomTabColorizer;
|
|
||||||
private final SimpleTabColorizer mDefaultTabColorizer;
|
|
||||||
|
|
||||||
SlidingTabStrip(Context context) {
|
|
||||||
this(context, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
SlidingTabStrip(Context context, AttributeSet attrs) {
|
|
||||||
super(context, attrs);
|
|
||||||
setWillNotDraw(false);
|
|
||||||
|
|
||||||
final float density = getResources().getDisplayMetrics().density;
|
|
||||||
|
|
||||||
TypedValue outValue = new TypedValue();
|
|
||||||
context.getTheme().resolveAttribute(android.R.attr.colorForeground, outValue, true);
|
|
||||||
final int themeForegroundColor = outValue.data;
|
|
||||||
|
|
||||||
int defaultBottomBorderColor = setColorAlpha(themeForegroundColor,
|
|
||||||
DEFAULT_BOTTOM_BORDER_COLOR_ALPHA);
|
|
||||||
|
|
||||||
mDefaultTabColorizer = new SimpleTabColorizer();
|
|
||||||
mDefaultTabColorizer.setIndicatorColors(MainApp.gc(R.color.tabBgColorSelected));
|
|
||||||
|
|
||||||
mBottomBorderThickness = (int) (DEFAULT_BOTTOM_BORDER_THICKNESS_DIPS * density);
|
|
||||||
mBottomBorderPaint = new Paint();
|
|
||||||
mBottomBorderPaint.setColor(defaultBottomBorderColor);
|
|
||||||
|
|
||||||
mSelectedIndicatorThickness = (int) (SELECTED_INDICATOR_THICKNESS_DIPS * density);
|
|
||||||
mSelectedIndicatorPaint = new Paint();
|
|
||||||
}
|
|
||||||
|
|
||||||
void setCustomTabColorizer(SlidingTabLayout.TabColorizer customTabColorizer) {
|
|
||||||
mCustomTabColorizer = customTabColorizer;
|
|
||||||
invalidate();
|
|
||||||
}
|
|
||||||
|
|
||||||
void setSelectedIndicatorColors(int... colors) {
|
|
||||||
// Make sure that the custom colorizer is removed
|
|
||||||
mCustomTabColorizer = null;
|
|
||||||
mDefaultTabColorizer.setIndicatorColors(colors);
|
|
||||||
invalidate();
|
|
||||||
}
|
|
||||||
|
|
||||||
void onViewPagerPageChanged(int position, float positionOffset) {
|
|
||||||
mSelectedPosition = position;
|
|
||||||
mSelectionOffset = positionOffset;
|
|
||||||
invalidate();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onDraw(Canvas canvas) {
|
|
||||||
final int height = getHeight();
|
|
||||||
final int childCount = getChildCount();
|
|
||||||
final SlidingTabLayout.TabColorizer tabColorizer = mCustomTabColorizer != null
|
|
||||||
? mCustomTabColorizer
|
|
||||||
: mDefaultTabColorizer;
|
|
||||||
|
|
||||||
// Thick colored underline below the current selection
|
|
||||||
if (childCount > 0) {
|
|
||||||
View selectedTitle = getChildAt(mSelectedPosition);
|
|
||||||
int left = selectedTitle.getLeft();
|
|
||||||
int right = selectedTitle.getRight();
|
|
||||||
int color = tabColorizer.getIndicatorColor(mSelectedPosition);
|
|
||||||
|
|
||||||
if (mSelectionOffset > 0f && mSelectedPosition < (getChildCount() - 1)) {
|
|
||||||
int nextColor = tabColorizer.getIndicatorColor(mSelectedPosition + 1);
|
|
||||||
if (color != nextColor) {
|
|
||||||
color = blendColors(nextColor, color, mSelectionOffset);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Draw the selection partway between the tabs
|
|
||||||
View nextTitle = getChildAt(mSelectedPosition + 1);
|
|
||||||
left = (int) (mSelectionOffset * nextTitle.getLeft() +
|
|
||||||
(1.0f - mSelectionOffset) * left);
|
|
||||||
right = (int) (mSelectionOffset * nextTitle.getRight() +
|
|
||||||
(1.0f - mSelectionOffset) * right);
|
|
||||||
}
|
|
||||||
|
|
||||||
mSelectedIndicatorPaint.setColor(color);
|
|
||||||
|
|
||||||
canvas.drawRect(left, height - mSelectedIndicatorThickness, right,
|
|
||||||
height, mSelectedIndicatorPaint);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Thin underline along the entire bottom edge
|
|
||||||
canvas.drawRect(0, height - mBottomBorderThickness, getWidth(), height, mBottomBorderPaint);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the alpha value of the {@code color} to be the given {@code alpha} value.
|
|
||||||
*/
|
|
||||||
private static int setColorAlpha(int color, byte alpha) {
|
|
||||||
return Color.argb(alpha, Color.red(color), Color.green(color), Color.blue(color));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Blend {@code color1} and {@code color2} using the given ratio.
|
|
||||||
*
|
|
||||||
* @param ratio of which to blend. 1.0 will return {@code color1}, 0.5 will give an even blend,
|
|
||||||
* 0.0 will return {@code color2}.
|
|
||||||
*/
|
|
||||||
private static int blendColors(int color1, int color2, float ratio) {
|
|
||||||
final float inverseRation = 1f - ratio;
|
|
||||||
float r = (Color.red(color1) * ratio) + (Color.red(color2) * inverseRation);
|
|
||||||
float g = (Color.green(color1) * ratio) + (Color.green(color2) * inverseRation);
|
|
||||||
float b = (Color.blue(color1) * ratio) + (Color.blue(color2) * inverseRation);
|
|
||||||
return Color.rgb((int) r, (int) g, (int) b);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class SimpleTabColorizer implements SlidingTabLayout.TabColorizer {
|
|
||||||
private int[] mIndicatorColors;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public final int getIndicatorColor(int position) {
|
|
||||||
return mIndicatorColors[position % mIndicatorColors.length];
|
|
||||||
}
|
|
||||||
|
|
||||||
void setIndicatorColors(int... colors) {
|
|
||||||
mIndicatorColors = colors;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -39,6 +39,10 @@ public class TabPageAdapter extends FragmentStatePagerAdapter {
|
||||||
return Fragment.instantiate(context, visibleFragmentList.get(position).pluginDescription.getFragmentClass());
|
return Fragment.instantiate(context, visibleFragmentList.get(position).pluginDescription.getFragmentClass());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PluginBase getPluginAt(int position) {
|
||||||
|
return visibleFragmentList.get(position);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void finishUpdate(ViewGroup container) {
|
public void finishUpdate(ViewGroup container) {
|
||||||
try{
|
try{
|
||||||
|
|
|
@ -1,46 +1,59 @@
|
||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<android.support.v4.widget.DrawerLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
xmlns:tools="http://schemas.android.com/tools"
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
android:id="@+id/drawer_layout"
|
android:orientation="vertical"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="match_parent">
|
android:layout_height="match_parent"
|
||||||
|
android:clipChildren="false"
|
||||||
|
android:clipToPadding="false">
|
||||||
|
|
||||||
<RelativeLayout
|
<android.support.v7.widget.Toolbar
|
||||||
|
android:id="@+id/toolbar"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="match_parent">
|
android:layout_height="?attr/actionBarSize"
|
||||||
|
android:background="@color/colorPrimary"
|
||||||
|
app:contentInsetEndWithActions="48dp"
|
||||||
|
app:contentInsetStartWithNavigation="48dp"
|
||||||
|
android:elevation="4dp">
|
||||||
|
|
||||||
<LinearLayout
|
<android.support.design.widget.TabLayout
|
||||||
|
android:id="@+id/tabs_normal"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="match_parent"
|
android:layout_height="match_parent"
|
||||||
android:orientation="vertical">
|
android:background="@android:color/transparent"
|
||||||
|
app:tabIndicatorColor="#FFFFFF"
|
||||||
|
app:tabMode="scrollable" />
|
||||||
|
|
||||||
|
<android.support.design.widget.TabLayout
|
||||||
|
android:id="@+id/tabs_compact"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
android:background="@android:color/transparent"
|
||||||
|
app:tabMinWidth="0dp"
|
||||||
|
app:tabPadding="0dp"
|
||||||
|
app:tabIndicatorColor="#FFFFFF"
|
||||||
|
app:tabMode="scrollable" />
|
||||||
|
</android.support.v7.widget.Toolbar>
|
||||||
|
|
||||||
<info.nightscout.androidaps.tabs.SlidingTabLayout
|
<android.support.v4.widget.DrawerLayout
|
||||||
android:id="@+id/tabs"
|
android:id="@+id/drawer_layout"
|
||||||
android:layout_width="match_parent"
|
android:layout_weight="1"
|
||||||
android:layout_height="wrap_content"
|
android:layout_width="match_parent"
|
||||||
android:layout_marginBottom="10dp"
|
android:layout_height="0dp">
|
||||||
android:paddingEnd="30dp" />
|
|
||||||
|
|
||||||
<android.support.v4.view.ViewPager
|
<android.support.v4.view.ViewPager
|
||||||
android:id="@+id/pager"
|
android:id="@+id/pager"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="0dp"
|
android:layout_height="match_parent" />
|
||||||
android:layout_weight="1" />
|
|
||||||
|
|
||||||
</LinearLayout>
|
<android.support.design.widget.NavigationView
|
||||||
|
android:id="@+id/navigation_view"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
app:itemBackground="?selectableItemBackground"
|
||||||
|
app:itemTextColor="#FFFFFF"
|
||||||
|
android:layout_gravity="start" />
|
||||||
|
</android.support.v4.widget.DrawerLayout>
|
||||||
|
|
||||||
<ImageButton
|
</LinearLayout>
|
||||||
android:id="@+id/overview_menuButton"
|
|
||||||
android:layout_width="30dp"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:layout_alignParentEnd="true"
|
|
||||||
android:layout_alignParentTop="true"
|
|
||||||
android:background="@color/tabBgColor"
|
|
||||||
android:paddingTop="5dp"
|
|
||||||
app:srcCompat="@drawable/ic_more_vert_black_24dp" />
|
|
||||||
|
|
||||||
</RelativeLayout>
|
|
||||||
|
|
||||||
</android.support.v4.widget.DrawerLayout>
|
|
7
app/src/main/res/layout/activity_single_fragment.xml
Normal file
7
app/src/main/res/layout/activity_single_fragment.xml
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:id="@+id/frame_layout"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="match_parent">
|
||||||
|
|
||||||
|
</FrameLayout>
|
|
@ -1,31 +1,45 @@
|
||||||
<menu xmlns:android="http://schemas.android.com/apk/res/android"
|
<menu xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:tools="http://schemas.android.com/tools"
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
tools:context=".MainActivity">
|
tools:context=".MainActivity">
|
||||||
<item
|
<item
|
||||||
android:id="@+id/nav_preferences"
|
android:id="@+id/nav_preferences"
|
||||||
|
app:showAsAction="never"
|
||||||
android:title="@string/nav_preferences" />
|
android:title="@string/nav_preferences" />
|
||||||
|
<item
|
||||||
|
android:id="@+id/nav_plugin_preferences"
|
||||||
|
app:showAsAction="never"
|
||||||
|
android:title="@string/nav_plugin_preferences" />
|
||||||
<item
|
<item
|
||||||
android:id="@+id/nav_historybrowser"
|
android:id="@+id/nav_historybrowser"
|
||||||
|
app:showAsAction="never"
|
||||||
android:title="@string/nav_historybrowser" />
|
android:title="@string/nav_historybrowser" />
|
||||||
<item
|
<item
|
||||||
android:id="@+id/nav_setupwizard"
|
android:id="@+id/nav_setupwizard"
|
||||||
|
app:showAsAction="never"
|
||||||
android:title="@string/nav_setupwizard" />
|
android:title="@string/nav_setupwizard" />
|
||||||
<item
|
<item
|
||||||
android:id="@+id/nav_resetdb"
|
android:id="@+id/nav_resetdb"
|
||||||
|
app:showAsAction="never"
|
||||||
android:title="@string/nav_resetdb" />
|
android:title="@string/nav_resetdb" />
|
||||||
<item
|
<item
|
||||||
android:id="@+id/nav_export"
|
android:id="@+id/nav_export"
|
||||||
|
app:showAsAction="never"
|
||||||
android:title="@string/nav_export" />
|
android:title="@string/nav_export" />
|
||||||
<item
|
<item
|
||||||
android:id="@+id/nav_import"
|
android:id="@+id/nav_import"
|
||||||
|
app:showAsAction="never"
|
||||||
android:title="@string/nav_import" />
|
android:title="@string/nav_import" />
|
||||||
<item
|
<item
|
||||||
android:id="@+id/nav_show_logcat"
|
android:id="@+id/nav_show_logcat"
|
||||||
|
app:showAsAction="never"
|
||||||
android:title="@string/nav_show_logcat" />
|
android:title="@string/nav_show_logcat" />
|
||||||
<item
|
<item
|
||||||
android:id="@+id/nav_about"
|
android:id="@+id/nav_about"
|
||||||
|
app:showAsAction="never"
|
||||||
android:title="@string/nav_about" />
|
android:title="@string/nav_about" />
|
||||||
<item
|
<item
|
||||||
android:id="@+id/nav_exit"
|
android:id="@+id/nav_exit"
|
||||||
|
app:showAsAction="never"
|
||||||
android:title="@string/nav_exit" />
|
android:title="@string/nav_exit" />
|
||||||
</menu>
|
</menu>
|
||||||
|
|
10
app/src/main/res/menu/menu_single_fragment.xml
Normal file
10
app/src/main/res/menu/menu_single_fragment.xml
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
<menu xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
|
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||||
|
tools:context=".MainActivity">
|
||||||
|
<item
|
||||||
|
android:id="@+id/nav_plugin_preferences"
|
||||||
|
app:showAsAction="ifRoom"
|
||||||
|
android:icon="@drawable/ic_settings"
|
||||||
|
android:title="@string/nav_plugin_preferences" />
|
||||||
|
</menu>
|
|
@ -1,7 +0,0 @@
|
||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<resources>
|
|
||||||
|
|
||||||
<style name="AppTheme" parent="AppTheme.base">
|
|
||||||
|
|
||||||
</style>
|
|
||||||
</resources>
|
|
|
@ -33,9 +33,9 @@
|
||||||
<color name="tempTargetDisabledBackground">#303F9F</color>
|
<color name="tempTargetDisabledBackground">#303F9F</color>
|
||||||
|
|
||||||
|
|
||||||
<color name="colorPrimary">#3F51B5</color>
|
<color name="colorPrimary">#212121</color>
|
||||||
<color name="colorPrimaryDark">#303F9F</color>
|
<color name="colorPrimaryDark">#000000</color>
|
||||||
<color name="colorAccent">#FF4081</color>
|
<color name="colorAccent">#009688</color>
|
||||||
<color name="colorInitializingBorder">#00695c</color>
|
<color name="colorInitializingBorder">#00695c</color>
|
||||||
|
|
||||||
<color name="cardColorBackground">#121212</color>
|
<color name="cardColorBackground">#121212</color>
|
||||||
|
|
|
@ -2,4 +2,5 @@
|
||||||
<!-- Default screen margins, per the Android Design guidelines. -->
|
<!-- Default screen margins, per the Android Design guidelines. -->
|
||||||
<dimen name="activity_horizontal_margin">16dp</dimen>
|
<dimen name="activity_horizontal_margin">16dp</dimen>
|
||||||
<dimen name="activity_vertical_margin">16dp</dimen>
|
<dimen name="activity_vertical_margin">16dp</dimen>
|
||||||
|
<dimen name="compact_height">30dp</dimen>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -766,6 +766,7 @@
|
||||||
<string name="dexcomg5_xdripupload_summary">In xDrip+ select 640g/Eversense data source</string>
|
<string name="dexcomg5_xdripupload_summary">In xDrip+ select 640g/Eversense data source</string>
|
||||||
<string name="nsclientbg">NSClient BG</string>
|
<string name="nsclientbg">NSClient BG</string>
|
||||||
<string name="minimalbasalvaluereplaced">Basal value replaced by minimal supported value</string>
|
<string name="minimalbasalvaluereplaced">Basal value replaced by minimal supported value</string>
|
||||||
|
<string name="maximumbasalvaluereplaced">Basal value replaced by maximum supported value</string>
|
||||||
<string name="overview_editquickwizard_usebg">BG calculation</string>
|
<string name="overview_editquickwizard_usebg">BG calculation</string>
|
||||||
<string name="overview_editquickwizard_usebolusiob">Bolus IOB calculation</string>
|
<string name="overview_editquickwizard_usebolusiob">Bolus IOB calculation</string>
|
||||||
<string name="overview_editquickwizard_usebasaliob">Basal IOB calculation</string>
|
<string name="overview_editquickwizard_usebasaliob">Basal IOB calculation</string>
|
||||||
|
@ -1104,4 +1105,7 @@
|
||||||
<string name="danar_saveuseroptions">Save options to pump</string>
|
<string name="danar_saveuseroptions">Save options to pump</string>
|
||||||
<string name="option_on">On</string>
|
<string name="option_on">On</string>
|
||||||
<string name="option_off">Off</string>
|
<string name="option_off">Off</string>
|
||||||
|
<string name="open_navigation">Open navigation</string>
|
||||||
|
<string name="close_navigation">Close navigation</string>
|
||||||
|
<string name="nav_plugin_preferences">Plugin preferences</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -1,8 +1,12 @@
|
||||||
<resources>
|
<resources>
|
||||||
|
|
||||||
<!-- Base application theme. -->
|
<style name="AppTheme" parent="Theme.AppCompat">
|
||||||
<style name="AppTheme" parent="android:Theme.Material">
|
<item name="colorPrimary">@color/colorPrimary</item>
|
||||||
<!-- Customize your theme here. -->
|
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
|
||||||
|
<item name="colorAccent">@color/colorAccent</item>
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<style name="AppTheme.NoActionBar" parent="Theme.AppCompat.NoActionBar">
|
||||||
<item name="colorPrimary">@color/colorPrimary</item>
|
<item name="colorPrimary">@color/colorPrimary</item>
|
||||||
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
|
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
|
||||||
<item name="colorAccent">@color/colorAccent</item>
|
<item name="colorAccent">@color/colorAccent</item>
|
||||||
|
|
|
@ -16,12 +16,6 @@
|
||||||
android:summary="@string/automatically_connect_when"
|
android:summary="@string/automatically_connect_when"
|
||||||
android:title="@string/connect_preemptively" />
|
android:title="@string/connect_preemptively" />
|
||||||
|
|
||||||
<SwitchPreference
|
|
||||||
android:defaultValue="false"
|
|
||||||
android:key="insight_real_tbr_cancel"
|
|
||||||
android:summary="@string/insight_actually_cancel_tbr_summary"
|
|
||||||
android:title="@string/insight_use_real_tbr_cancels" />
|
|
||||||
|
|
||||||
<SwitchPreference
|
<SwitchPreference
|
||||||
android:defaultValue="false"
|
android:defaultValue="false"
|
||||||
android:key="insight_automatic_careportal_events"
|
android:key="insight_automatic_careportal_events"
|
||||||
|
|
|
@ -156,7 +156,7 @@ public class ConstraintsCheckerTest {
|
||||||
// Apply all limits
|
// Apply all limits
|
||||||
Constraint<Double> d = constraintChecker.getMaxBasalAllowed(AAPSMocker.getValidProfile());
|
Constraint<Double> d = constraintChecker.getMaxBasalAllowed(AAPSMocker.getValidProfile());
|
||||||
Assert.assertEquals(0.8d, d.value());
|
Assert.assertEquals(0.8d, d.value());
|
||||||
Assert.assertEquals(true, d.getReasonList().size() == 7); // 4x Safety & RS & R & Insight
|
Assert.assertEquals(true, d.getReasonList().size() == 6);
|
||||||
Assert.assertEquals("DanaR: Limiting basal rate to 0.80 U/h because of pump limit", d.getMostLimitedReasons());
|
Assert.assertEquals("DanaR: Limiting basal rate to 0.80 U/h because of pump limit", d.getMostLimitedReasons());
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,57 +0,0 @@
|
||||||
package info.nightscout.androidaps.plugins.PumpInsight;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import junit.framework.Assert;
|
|
||||||
|
|
||||||
import org.junit.Before;
|
|
||||||
import org.junit.Test;
|
|
||||||
import org.junit.runner.RunWith;
|
|
||||||
import org.powermock.core.classloader.annotations.PrepareForTest;
|
|
||||||
import org.powermock.modules.junit4.PowerMockRunner;
|
|
||||||
|
|
||||||
import info.AAPSMocker;
|
|
||||||
import info.nightscout.androidaps.Constants;
|
|
||||||
import info.nightscout.androidaps.MainApp;
|
|
||||||
import info.nightscout.androidaps.interfaces.Constraint;
|
|
||||||
import info.nightscout.androidaps.interfaces.PluginType;
|
|
||||||
import info.nightscout.androidaps.plugins.ConfigBuilder.ConfigBuilderPlugin;
|
|
||||||
import info.nightscout.androidaps.plugins.PumpInsight.connector.StatusTaskRunner;
|
|
||||||
import info.nightscout.utils.ToastUtils;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Created by mike on 23.03.2018.
|
|
||||||
*/
|
|
||||||
|
|
||||||
@RunWith(PowerMockRunner.class)
|
|
||||||
@PrepareForTest({MainApp.class, ConfigBuilderPlugin.class, ToastUtils.class, Context.class})
|
|
||||||
public class InsightPluginTest {
|
|
||||||
|
|
||||||
InsightPlugin insightPlugin;
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void basalRateShouldBeLimited() throws Exception {
|
|
||||||
insightPlugin.setPluginEnabled(PluginType.PUMP, true);
|
|
||||||
StatusTaskRunner.Result result = new StatusTaskRunner.Result();
|
|
||||||
result.maximumBasalAmount = 1.1d;
|
|
||||||
insightPlugin.setStatusResult(result);
|
|
||||||
|
|
||||||
Constraint<Double> c = new Constraint<>(Constants.REALLYHIGHBASALRATE);
|
|
||||||
insightPlugin.applyBasalConstraints(c, AAPSMocker.getValidProfile());
|
|
||||||
Assert.assertEquals(1.1d, c.value());
|
|
||||||
Assert.assertEquals("Insight: Limiting basal rate to 1.10 U/h because of pump limit", c.getReasons());
|
|
||||||
Assert.assertEquals("Insight: Limiting basal rate to 1.10 U/h because of pump limit", c.getMostLimitedReasons());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Before
|
|
||||||
public void prepareMocks() throws Exception {
|
|
||||||
AAPSMocker.mockMainApp();
|
|
||||||
AAPSMocker.mockConfigBuilder();
|
|
||||||
AAPSMocker.mockBus();
|
|
||||||
AAPSMocker.mockStrings();
|
|
||||||
AAPSMocker.mockCommandQueue();
|
|
||||||
|
|
||||||
insightPlugin = InsightPlugin.getPlugin();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -15,6 +15,6 @@ public class SourceNSClientPluginTest {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void advancedFilteringSupported() {
|
public void advancedFilteringSupported() {
|
||||||
Assert.assertEquals(true, SourceNSClientPlugin.getPlugin().advancedFilteringSupported());
|
Assert.assertEquals(false, SourceNSClientPlugin.getPlugin().advancedFilteringSupported());
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Reference in a new issue