Added actions to notifications (this required a slight detour)
This commit is contained in:
		| @@ -47,6 +47,10 @@ | |||||||
|                 android:name="android.support.PARENT_ACTIVITY" |                 android:name="android.support.PARENT_ACTIVITY" | ||||||
|                 android:value=".MainActivity"/> |                 android:value=".MainActivity"/> | ||||||
|         </activity> |         </activity> | ||||||
|  |         <activity | ||||||
|  |             android:name=".dialog.FullNodeDialogActivity" | ||||||
|  |             android:label="@string/full_node" | ||||||
|  |             android:theme="@style/Theme.AppCompat.Light.Dialog"/> | ||||||
|         <activity |         <activity | ||||||
|             android:name=".ComposeMessageActivity" |             android:name=".ComposeMessageActivity" | ||||||
|             android:label="@string/compose_message" |             android:label="@string/compose_message" | ||||||
| @@ -125,8 +129,12 @@ | |||||||
|             </intent-filter> |             </intent-filter> | ||||||
|         </activity> |         </activity> | ||||||
|  |  | ||||||
|         <service android:name=".service.BitmessageService"/> |         <service | ||||||
|         <service android:name=".service.ProofOfWorkService"/> |             android:name=".service.BitmessageService" | ||||||
|  |             android:exported="false"/> | ||||||
|  |         <service | ||||||
|  |             android:name=".service.ProofOfWorkService" | ||||||
|  |             android:exported="false"/> | ||||||
|  |  | ||||||
|         <!-- Synchronization --> |         <!-- Synchronization --> | ||||||
|         <provider |         <provider | ||||||
| @@ -137,6 +145,7 @@ | |||||||
|  |  | ||||||
|         <service |         <service | ||||||
|             android:name=".synchronization.AuthenticatorService" |             android:name=".synchronization.AuthenticatorService" | ||||||
|  |             android:exported="true" | ||||||
|             tools:ignore="ExportedService"> |             tools:ignore="ExportedService"> | ||||||
|             <intent-filter> |             <intent-filter> | ||||||
|                 <action android:name="android.accounts.AccountAuthenticator"/> |                 <action android:name="android.accounts.AccountAuthenticator"/> | ||||||
| @@ -158,6 +167,9 @@ | |||||||
|                 android:name="android.content.SyncAdapter" |                 android:name="android.content.SyncAdapter" | ||||||
|                 android:resource="@xml/syncadapter"/> |                 android:resource="@xml/syncadapter"/> | ||||||
|         </service> |         </service> | ||||||
|  |         <service | ||||||
|  |             android:name=".service.BitmessageIntentService" | ||||||
|  |             android:exported="false"/> | ||||||
|  |  | ||||||
|         <!-- Receive Wi-Fi connection state changes --> |         <!-- Receive Wi-Fi connection state changes --> | ||||||
|         <receiver android:name=".listener.WifiReceiver"> |         <receiver android:name=".listener.WifiReceiver"> | ||||||
|   | |||||||
| @@ -16,10 +16,16 @@ | |||||||
|  |  | ||||||
| package ch.dissem.apps.abit; | package ch.dissem.apps.abit; | ||||||
|  |  | ||||||
|  | import android.app.Activity; | ||||||
|  | import android.content.Context; | ||||||
|  | import android.content.Intent; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
|  | import android.support.v4.app.Fragment; | ||||||
| import android.support.v7.app.AppCompatActivity; | import android.support.v7.app.AppCompatActivity; | ||||||
| import android.support.v7.widget.Toolbar; | import android.support.v7.widget.Toolbar; | ||||||
|  |  | ||||||
|  | import ch.dissem.bitmessage.entity.Plaintext; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Compose a new message. |  * Compose a new message. | ||||||
|  */ |  */ | ||||||
| @@ -49,4 +55,30 @@ public class ComposeMessageActivity extends AppCompatActivity { | |||||||
|             .replace(R.id.content, fragment) |             .replace(R.id.content, fragment) | ||||||
|             .commit(); |             .commit(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public static void launchReplyTo(Fragment fragment, Plaintext item) { | ||||||
|  |         fragment.startActivity(getReplyIntent(fragment.getActivity(), item)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static void launchReplyTo(Activity activity, Plaintext item) { | ||||||
|  |         activity.startActivity(getReplyIntent(activity, item)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static Intent getReplyIntent(Context ctx, Plaintext item) { | ||||||
|  |         Intent replyIntent = new Intent(ctx, ComposeMessageActivity.class); | ||||||
|  |         replyIntent.putExtra(EXTRA_RECIPIENT, item.getFrom()); | ||||||
|  |         replyIntent.putExtra(EXTRA_IDENTITY, item.getTo()); | ||||||
|  |         String prefix; | ||||||
|  |         if (item.getSubject().length() >= 3 && item.getSubject().substring(0, 3) | ||||||
|  |             .equalsIgnoreCase("RE:")) { | ||||||
|  |             prefix = ""; | ||||||
|  |         } else { | ||||||
|  |             prefix = "RE: "; | ||||||
|  |         } | ||||||
|  |         replyIntent.putExtra(EXTRA_SUBJECT, prefix + item.getSubject()); | ||||||
|  |         replyIntent.putExtra(EXTRA_CONTENT, | ||||||
|  |             "\n\n------------------------------------------------------\n" | ||||||
|  |                 + item.getText()); | ||||||
|  |         return replyIntent; | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -17,19 +17,14 @@ | |||||||
| package ch.dissem.apps.abit; | package ch.dissem.apps.abit; | ||||||
|  |  | ||||||
| import android.app.AlertDialog; | import android.app.AlertDialog; | ||||||
| import android.content.ComponentName; |  | ||||||
| import android.content.Context; |  | ||||||
| import android.content.Intent; | import android.content.Intent; | ||||||
| import android.content.ServiceConnection; |  | ||||||
| import android.graphics.Point; | import android.graphics.Point; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.os.IBinder; |  | ||||||
| import android.support.v4.app.Fragment; | import android.support.v4.app.Fragment; | ||||||
| import android.support.v7.app.AppCompatActivity; | import android.support.v7.app.AppCompatActivity; | ||||||
| import android.support.v7.widget.Toolbar; | import android.support.v7.widget.Toolbar; | ||||||
| import android.view.View; | import android.view.View; | ||||||
| import android.view.ViewGroup; | import android.view.ViewGroup; | ||||||
| import android.widget.CompoundButton; |  | ||||||
| import android.widget.RelativeLayout; | import android.widget.RelativeLayout; | ||||||
|  |  | ||||||
| import com.github.amlcurran.showcaseview.ShowcaseView; | import com.github.amlcurran.showcaseview.ShowcaseView; | ||||||
| @@ -59,10 +54,10 @@ import java.util.Collection; | |||||||
| import java.util.List; | import java.util.List; | ||||||
|  |  | ||||||
| import ch.dissem.apps.abit.dialog.AddIdentityDialogFragment; | import ch.dissem.apps.abit.dialog.AddIdentityDialogFragment; | ||||||
|  | import ch.dissem.apps.abit.dialog.FullNodeDialogActivity; | ||||||
| import ch.dissem.apps.abit.listener.ActionBarListener; | import ch.dissem.apps.abit.listener.ActionBarListener; | ||||||
| import ch.dissem.apps.abit.listener.ListSelectionListener; | import ch.dissem.apps.abit.listener.ListSelectionListener; | ||||||
| import ch.dissem.apps.abit.service.BitmessageService; | import ch.dissem.apps.abit.service.BitmessageService; | ||||||
| import ch.dissem.apps.abit.service.BitmessageService.BitmessageBinder; |  | ||||||
| import ch.dissem.apps.abit.service.Singleton; | import ch.dissem.apps.abit.service.Singleton; | ||||||
| import ch.dissem.apps.abit.synchronization.SyncAdapter; | import ch.dissem.apps.abit.synchronization.SyncAdapter; | ||||||
| import ch.dissem.apps.abit.util.Preferences; | import ch.dissem.apps.abit.util.Preferences; | ||||||
| @@ -71,6 +66,7 @@ import ch.dissem.bitmessage.entity.BitmessageAddress; | |||||||
| import ch.dissem.bitmessage.entity.Plaintext; | import ch.dissem.bitmessage.entity.Plaintext; | ||||||
| import ch.dissem.bitmessage.entity.valueobject.Label; | import ch.dissem.bitmessage.entity.valueobject.Label; | ||||||
|  |  | ||||||
|  | import static ch.dissem.apps.abit.ComposeMessageActivity.launchReplyTo; | ||||||
| import static ch.dissem.apps.abit.service.BitmessageService.isRunning; | import static ch.dissem.apps.abit.service.BitmessageService.isRunning; | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -94,6 +90,7 @@ import static ch.dissem.apps.abit.service.BitmessageService.isRunning; | |||||||
| public class MainActivity extends AppCompatActivity | public class MainActivity extends AppCompatActivity | ||||||
|     implements ListSelectionListener<Serializable>, ActionBarListener { |     implements ListSelectionListener<Serializable>, ActionBarListener { | ||||||
|     public static final String EXTRA_SHOW_MESSAGE = "ch.dissem.abit.ShowMessage"; |     public static final String EXTRA_SHOW_MESSAGE = "ch.dissem.abit.ShowMessage"; | ||||||
|  |     public static final String EXTRA_REPLY_TO_MESSAGE = "ch.dissem.abit.ReplyToMessage"; | ||||||
|     public static final String ACTION_SHOW_INBOX = "ch.dissem.abit.ShowInbox"; |     public static final String ACTION_SHOW_INBOX = "ch.dissem.abit.ShowInbox"; | ||||||
|  |  | ||||||
|     private static final Logger LOG = LoggerFactory.getLogger(MainActivity.class); |     private static final Logger LOG = LoggerFactory.getLogger(MainActivity.class); | ||||||
| @@ -110,22 +107,6 @@ public class MainActivity extends AppCompatActivity | |||||||
|      */ |      */ | ||||||
|     private boolean twoPane; |     private boolean twoPane; | ||||||
|  |  | ||||||
|     private static BitmessageBinder service; |  | ||||||
|     private static boolean bound; |  | ||||||
|     private static ServiceConnection connection = new ServiceConnection() { |  | ||||||
|         @Override |  | ||||||
|         public void onServiceConnected(ComponentName name, IBinder service) { |  | ||||||
|             MainActivity.service = (BitmessageBinder) service; |  | ||||||
|             MainActivity.bound = true; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public void onServiceDisconnected(ComponentName name) { |  | ||||||
|             service = null; |  | ||||||
|             bound = false; |  | ||||||
|         } |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     private Label selectedLabel; |     private Label selectedLabel; | ||||||
|  |  | ||||||
|     private BitmessageContext bmc; |     private BitmessageContext bmc; | ||||||
| @@ -172,8 +153,13 @@ public class MainActivity extends AppCompatActivity | |||||||
|         Singleton.getMessageListener(this).resetNotification(); |         Singleton.getMessageListener(this).resetNotification(); | ||||||
|  |  | ||||||
|         // handle intents |         // handle intents | ||||||
|         if (getIntent().hasExtra(EXTRA_SHOW_MESSAGE)) { |         Intent intent = getIntent(); | ||||||
|             onItemSelected(getIntent().getSerializableExtra(EXTRA_SHOW_MESSAGE)); |         if (intent.hasExtra(EXTRA_SHOW_MESSAGE)) { | ||||||
|  |             onItemSelected(intent.getSerializableExtra(EXTRA_SHOW_MESSAGE)); | ||||||
|  |         } | ||||||
|  |         if (intent.hasExtra(EXTRA_REPLY_TO_MESSAGE)) { | ||||||
|  |             Plaintext item = (Plaintext) intent.getSerializableExtra(EXTRA_REPLY_TO_MESSAGE); | ||||||
|  |             launchReplyTo(this, item); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (Preferences.useTrustedNode(this)) { |         if (Preferences.useTrustedNode(this)) { | ||||||
| @@ -345,9 +331,9 @@ public class MainActivity extends AppCompatActivity | |||||||
|             .withChecked(isRunning()) |             .withChecked(isRunning()) | ||||||
|             .withOnCheckedChangeListener((drawerItem, buttonView, isChecked) -> { |             .withOnCheckedChangeListener((drawerItem, buttonView, isChecked) -> { | ||||||
|                 if (isChecked) { |                 if (isChecked) { | ||||||
|                     checkAndStartNode(buttonView); |                     checkAndStartNode(); | ||||||
|                 } else { |                 } else { | ||||||
|                     service.shutdownNode(); |                     stopService(new Intent(this, BitmessageService.class)); | ||||||
|                 } |                 } | ||||||
|             }); |             }); | ||||||
|  |  | ||||||
| @@ -448,23 +434,11 @@ public class MainActivity extends AppCompatActivity | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void checkAndStartNode(final CompoundButton buttonView) { |     private void checkAndStartNode() { | ||||||
|         if (service == null) return; |  | ||||||
|  |  | ||||||
|         if (Preferences.isConnectionAllowed(MainActivity.this)) { |         if (Preferences.isConnectionAllowed(MainActivity.this)) { | ||||||
|             service.startupNode(); |             startService(new Intent(this, BitmessageService.class)); | ||||||
|         } else { |         } else { | ||||||
|             new AlertDialog.Builder(MainActivity.this) |             startActivity(new Intent(this, FullNodeDialogActivity.class)); | ||||||
|                 .setMessage(R.string.full_node_warning) |  | ||||||
|                 .setPositiveButton( |  | ||||||
|                     android.R.string.yes, |  | ||||||
|                     (dialog, which) -> service.startupNode() |  | ||||||
|                 ) |  | ||||||
|                 .setNegativeButton( |  | ||||||
|                     android.R.string.no, |  | ||||||
|                     (dialog, which) -> updateNodeSwitch() |  | ||||||
|                 ) |  | ||||||
|                 .show(); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -483,12 +457,15 @@ public class MainActivity extends AppCompatActivity | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void updateNodeSwitch() { |     public static void updateNodeSwitch() { | ||||||
|         runOnUiThread(() -> { |         MainActivity i = getInstance(); | ||||||
|             nodeSwitch.withChecked(bmc.isRunning()); |         if (i != null) { | ||||||
|             drawer.updateStickyFooterItem(nodeSwitch); |             i.runOnUiThread(() -> { | ||||||
|  |                 i.nodeSwitch.withChecked(i.bmc.isRunning()); | ||||||
|  |                 i.drawer.updateStickyFooterItem(i.nodeSwitch); | ||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private void showSelectedLabel() { |     private void showSelectedLabel() { | ||||||
|         if (getSupportFragmentManager().findFragmentById(R.id.item_list) instanceof |         if (getSupportFragmentManager().findFragmentById(R.id.item_list) instanceof | ||||||
| @@ -556,22 +533,6 @@ public class MainActivity extends AppCompatActivity | |||||||
|         return selectedLabel; |         return selectedLabel; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     protected void onStart() { |  | ||||||
|         super.onStart(); |  | ||||||
|         bindService(new Intent(this, BitmessageService.class), connection, Context |  | ||||||
|             .BIND_AUTO_CREATE); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     protected void onStop() { |  | ||||||
|         if (bound) { |  | ||||||
|             unbindService(connection); |  | ||||||
|             bound = false; |  | ||||||
|         } |  | ||||||
|         super.onStop(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static MainActivity getInstance() { |     public static MainActivity getInstance() { | ||||||
|         if (instance == null) return null; |         if (instance == null) return null; | ||||||
|         return instance.get(); |         return instance.get(); | ||||||
|   | |||||||
| @@ -16,11 +16,9 @@ | |||||||
|  |  | ||||||
| package ch.dissem.apps.abit; | package ch.dissem.apps.abit; | ||||||
|  |  | ||||||
| import android.content.Intent; |  | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.support.v4.app.Fragment; | import android.support.v4.app.Fragment; | ||||||
| import android.text.util.Linkify; | import android.text.util.Linkify; | ||||||
| import android.text.util.Linkify.TransformFilter; |  | ||||||
| import android.view.LayoutInflater; | import android.view.LayoutInflater; | ||||||
| import android.view.Menu; | import android.view.Menu; | ||||||
| import android.view.MenuInflater; | import android.view.MenuInflater; | ||||||
| @@ -33,7 +31,6 @@ import android.widget.TextView; | |||||||
| import com.mikepenz.google_material_typeface_library.GoogleMaterial; | import com.mikepenz.google_material_typeface_library.GoogleMaterial; | ||||||
|  |  | ||||||
| import java.util.Iterator; | import java.util.Iterator; | ||||||
| import java.util.regex.Matcher; |  | ||||||
|  |  | ||||||
| import ch.dissem.apps.abit.listener.ActionBarListener; | import ch.dissem.apps.abit.listener.ActionBarListener; | ||||||
| import ch.dissem.apps.abit.service.Singleton; | import ch.dissem.apps.abit.service.Singleton; | ||||||
| @@ -44,10 +41,6 @@ import ch.dissem.bitmessage.entity.valueobject.Label; | |||||||
| import ch.dissem.bitmessage.ports.MessageRepository; | import ch.dissem.bitmessage.ports.MessageRepository; | ||||||
|  |  | ||||||
| import static android.text.util.Linkify.WEB_URLS; | import static android.text.util.Linkify.WEB_URLS; | ||||||
| import static ch.dissem.apps.abit.ComposeMessageActivity.EXTRA_CONTENT; |  | ||||||
| import static ch.dissem.apps.abit.ComposeMessageActivity.EXTRA_IDENTITY; |  | ||||||
| import static ch.dissem.apps.abit.ComposeMessageActivity.EXTRA_RECIPIENT; |  | ||||||
| import static ch.dissem.apps.abit.ComposeMessageActivity.EXTRA_SUBJECT; |  | ||||||
| import static ch.dissem.apps.abit.util.Constants.BITMESSAGE_ADDRESS_PATTERN; | import static ch.dissem.apps.abit.util.Constants.BITMESSAGE_ADDRESS_PATTERN; | ||||||
| import static ch.dissem.apps.abit.util.Constants.BITMESSAGE_URL_SCHEMA; | import static ch.dissem.apps.abit.util.Constants.BITMESSAGE_URL_SCHEMA; | ||||||
|  |  | ||||||
| @@ -113,11 +106,8 @@ public class MessageDetailFragment extends Fragment { | |||||||
|  |  | ||||||
|             Linkify.addLinks(messageBody, WEB_URLS); |             Linkify.addLinks(messageBody, WEB_URLS); | ||||||
|             Linkify.addLinks(messageBody, BITMESSAGE_ADDRESS_PATTERN, BITMESSAGE_URL_SCHEMA, null, |             Linkify.addLinks(messageBody, BITMESSAGE_ADDRESS_PATTERN, BITMESSAGE_URL_SCHEMA, null, | ||||||
|                 new TransformFilter() { |                 (match, url) -> match.group() | ||||||
|                     public final String transformUrl(final Matcher match, String url) { |             ); | ||||||
|                         return match.group(); |  | ||||||
|                     } |  | ||||||
|                 }); |  | ||||||
|  |  | ||||||
|             messageBody.setLinksClickable(true); |             messageBody.setLinksClickable(true); | ||||||
|             messageBody.setTextIsSelectable(true); |             messageBody.setTextIsSelectable(true); | ||||||
| @@ -158,22 +148,7 @@ public class MessageDetailFragment extends Fragment { | |||||||
|         MessageRepository messageRepo = Singleton.getMessageRepository(getContext()); |         MessageRepository messageRepo = Singleton.getMessageRepository(getContext()); | ||||||
|         switch (menuItem.getItemId()) { |         switch (menuItem.getItemId()) { | ||||||
|             case R.id.reply: |             case R.id.reply: | ||||||
|                 Intent replyIntent = new Intent(getActivity().getApplicationContext(), |                 ComposeMessageActivity.launchReplyTo(this, item); | ||||||
|                     ComposeMessageActivity.class); |  | ||||||
|                 replyIntent.putExtra(EXTRA_RECIPIENT, item.getFrom()); |  | ||||||
|                 replyIntent.putExtra(EXTRA_IDENTITY, item.getTo()); |  | ||||||
|                 String prefix; |  | ||||||
|                 if (item.getSubject().length() >= 3 && item.getSubject().substring(0, 3) |  | ||||||
|                     .equalsIgnoreCase("RE:")) { |  | ||||||
|                     prefix = ""; |  | ||||||
|                 } else { |  | ||||||
|                     prefix = "RE: "; |  | ||||||
|                 } |  | ||||||
|                 replyIntent.putExtra(EXTRA_SUBJECT, prefix + item.getSubject()); |  | ||||||
|                 replyIntent.putExtra(EXTRA_CONTENT, |  | ||||||
|                     "\n\n------------------------------------------------------\n" |  | ||||||
|                         + item.getText()); |  | ||||||
|                 startActivity(replyIntent); |  | ||||||
|                 return true; |                 return true; | ||||||
|             case R.id.delete: |             case R.id.delete: | ||||||
|                 if (isInTrash(item)) { |                 if (isInTrash(item)) { | ||||||
|   | |||||||
| @@ -0,0 +1,44 @@ | |||||||
|  | /* | ||||||
|  |  * Copyright 2016 Christian Basler | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package ch.dissem.apps.abit.dialog; | ||||||
|  |  | ||||||
|  | import android.app.Activity; | ||||||
|  | import android.content.Intent; | ||||||
|  | import android.os.Bundle; | ||||||
|  |  | ||||||
|  | import ch.dissem.apps.abit.R; | ||||||
|  | import ch.dissem.apps.abit.service.BitmessageService; | ||||||
|  |  | ||||||
|  | import static ch.dissem.apps.abit.MainActivity.updateNodeSwitch; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @author Christian Basler | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | public class FullNodeDialogActivity extends Activity { | ||||||
|  |     @Override | ||||||
|  |     protected void onCreate(Bundle savedInstanceState) { | ||||||
|  |         super.onCreate(savedInstanceState); | ||||||
|  |         setContentView(R.layout.dialog_full_node); | ||||||
|  |         findViewById(R.id.ok).setOnClickListener(v -> { | ||||||
|  |             startService(new Intent(this, BitmessageService.class)); | ||||||
|  |             updateNodeSwitch(); | ||||||
|  |             finish(); | ||||||
|  |         }); | ||||||
|  |         findViewById(R.id.dismiss).setOnClickListener(v -> finish()); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -27,21 +27,26 @@ import java.util.TimerTask; | |||||||
|  |  | ||||||
| import ch.dissem.apps.abit.MainActivity; | import ch.dissem.apps.abit.MainActivity; | ||||||
| import ch.dissem.apps.abit.R; | import ch.dissem.apps.abit.R; | ||||||
|  | import ch.dissem.apps.abit.service.BitmessageIntentService; | ||||||
| import ch.dissem.apps.abit.service.BitmessageService; | import ch.dissem.apps.abit.service.BitmessageService; | ||||||
| import ch.dissem.bitmessage.utils.Property; | import ch.dissem.bitmessage.utils.Property; | ||||||
|  |  | ||||||
|  | import static android.app.PendingIntent.FLAG_UPDATE_CURRENT; | ||||||
|  | import static ch.dissem.apps.abit.MainActivity.updateNodeSwitch; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Shows the network status (as long as the client is connected as a full node) |  * Shows the network status (as long as the client is connected as a full node) | ||||||
|  */ |  */ | ||||||
| public class NetworkNotification extends AbstractNotification { | public class NetworkNotification extends AbstractNotification { | ||||||
|     public static final int ONGOING_NOTIFICATION_ID = 2; |     public static final int NETWORK_NOTIFICATION_ID = 2; | ||||||
|  |  | ||||||
|     private NotificationCompat.Builder builder; |     private final NotificationCompat.Builder builder; | ||||||
|  |     private Timer timer; | ||||||
|  |  | ||||||
|     public NetworkNotification(Context ctx) { |     public NetworkNotification(Context ctx) { | ||||||
|         super(ctx); |         super(ctx); | ||||||
|         Intent showMessageIntent = new Intent(ctx, MainActivity.class); |         Intent showAppIntent = new Intent(ctx, MainActivity.class); | ||||||
|         PendingIntent pendingIntent = PendingIntent.getActivity(ctx, 1, showMessageIntent, 0); |         PendingIntent pendingIntent = PendingIntent.getActivity(ctx, 1, showAppIntent, 0); | ||||||
|         builder = new NotificationCompat.Builder(ctx); |         builder = new NotificationCompat.Builder(ctx); | ||||||
|         builder.setSmallIcon(R.drawable.ic_notification_full_node) |         builder.setSmallIcon(R.drawable.ic_notification_full_node) | ||||||
|             .setContentTitle(ctx.getString(R.string.bitmessage_full_node)) |             .setContentTitle(ctx.getString(R.string.bitmessage_full_node)) | ||||||
| @@ -58,10 +63,7 @@ public class NetworkNotification extends AbstractNotification { | |||||||
|         Property connections = BitmessageService.getStatus().getProperty("network", "connections"); |         Property connections = BitmessageService.getStatus().getProperty("network", "connections"); | ||||||
|         if (!running) { |         if (!running) { | ||||||
|             builder.setContentText(ctx.getString(R.string.connection_info_disconnected)); |             builder.setContentText(ctx.getString(R.string.connection_info_disconnected)); | ||||||
|             MainActivity mainActivity = MainActivity.getInstance(); |             updateNodeSwitch(); | ||||||
|             if (mainActivity != null) { |  | ||||||
|                 mainActivity.updateNodeSwitch(); |  | ||||||
|             } |  | ||||||
|         } else if (connections.getProperties().length == 0) { |         } else if (connections.getProperties().length == 0) { | ||||||
|             builder.setContentText(ctx.getString(R.string.connection_info_pending)); |             builder.setContentText(ctx.getString(R.string.connection_info_pending)); | ||||||
|         } else { |         } else { | ||||||
| @@ -80,6 +82,19 @@ public class NetworkNotification extends AbstractNotification { | |||||||
|             } |             } | ||||||
|             builder.setContentText(info); |             builder.setContentText(info); | ||||||
|         } |         } | ||||||
|  |         builder.mActions.clear(); | ||||||
|  |         Intent intent = new Intent(ctx, BitmessageIntentService.class); | ||||||
|  |         if (running) { | ||||||
|  |             intent.putExtra(BitmessageIntentService.EXTRA_SHUTDOWN_NODE, true); | ||||||
|  |             builder.addAction(R.drawable.ic_notification_node_stop, | ||||||
|  |                 ctx.getString(R.string.full_node_stop), | ||||||
|  |                 PendingIntent.getService(ctx, 0, intent, FLAG_UPDATE_CURRENT)); | ||||||
|  |         } else { | ||||||
|  |             intent.putExtra(BitmessageIntentService.EXTRA_STARTUP_NODE, true); | ||||||
|  |             builder.addAction(R.drawable.ic_notification_node_start, | ||||||
|  |                 ctx.getString(R.string.full_node_restart), | ||||||
|  |                 PendingIntent.getService(ctx, 1, intent, FLAG_UPDATE_CURRENT)); | ||||||
|  |         } | ||||||
|         notification = builder.build(); |         notification = builder.build(); | ||||||
|         return running; |         return running; | ||||||
|     } |     } | ||||||
| @@ -88,7 +103,8 @@ public class NetworkNotification extends AbstractNotification { | |||||||
|     public void show() { |     public void show() { | ||||||
|         super.show(); |         super.show(); | ||||||
|  |  | ||||||
|         new Timer().schedule(new TimerTask() { |         timer = new Timer(); | ||||||
|  |         timer.schedule(new TimerTask() { | ||||||
|             @Override |             @Override | ||||||
|             public void run() { |             public void run() { | ||||||
|                 if (!update()) { |                 if (!update()) { | ||||||
| @@ -99,14 +115,28 @@ public class NetworkNotification extends AbstractNotification { | |||||||
|         }, 10_000, 10_000); |         }, 10_000, 10_000); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     public void showShutdown() { | ||||||
|  |         if (timer != null) { | ||||||
|  |             timer.cancel(); | ||||||
|  |         } | ||||||
|  |         update(); | ||||||
|  |         super.show(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected int getNotificationId() { |     protected int getNotificationId() { | ||||||
|         return ONGOING_NOTIFICATION_ID; |         return NETWORK_NOTIFICATION_ID; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void connecting() { |     public void connecting() { | ||||||
|         builder.setOngoing(true); |         builder.setOngoing(true); | ||||||
|         builder.setContentText(ctx.getString(R.string.connection_info_pending)); |         builder.setContentText(ctx.getString(R.string.connection_info_pending)); | ||||||
|  |         Intent intent = new Intent(ctx, BitmessageIntentService.class); | ||||||
|  |         intent.putExtra(BitmessageIntentService.EXTRA_SHUTDOWN_NODE, true); | ||||||
|  |         builder.mActions.clear(); | ||||||
|  |         builder.addAction(R.drawable.ic_notification_node_stop, | ||||||
|  |             ctx.getString(R.string.full_node_stop), | ||||||
|  |             PendingIntent.getService(ctx, 0, intent, FLAG_UPDATE_CURRENT)); | ||||||
|         notification = builder.build(); |         notification = builder.build(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -31,8 +31,13 @@ import java.util.Collection; | |||||||
| import ch.dissem.apps.abit.Identicon; | import ch.dissem.apps.abit.Identicon; | ||||||
| import ch.dissem.apps.abit.MainActivity; | import ch.dissem.apps.abit.MainActivity; | ||||||
| import ch.dissem.apps.abit.R; | import ch.dissem.apps.abit.R; | ||||||
|  | import ch.dissem.apps.abit.service.BitmessageIntentService; | ||||||
| import ch.dissem.bitmessage.entity.Plaintext; | import ch.dissem.bitmessage.entity.Plaintext; | ||||||
|  |  | ||||||
|  | import static android.app.PendingIntent.FLAG_UPDATE_CURRENT; | ||||||
|  | import static ch.dissem.apps.abit.MainActivity.EXTRA_REPLY_TO_MESSAGE; | ||||||
|  | import static ch.dissem.apps.abit.MainActivity.EXTRA_SHOW_MESSAGE; | ||||||
|  | import static ch.dissem.apps.abit.service.BitmessageIntentService.EXTRA_DELETE_MESSAGE; | ||||||
| import static ch.dissem.apps.abit.util.Drawables.toBitmap; | import static ch.dissem.apps.abit.util.Drawables.toBitmap; | ||||||
|  |  | ||||||
| public class NewMessageNotification extends AbstractNotification { | public class NewMessageNotification extends AbstractNotification { | ||||||
| @@ -56,20 +61,28 @@ public class NewMessageNotification extends AbstractNotification { | |||||||
|             .setStyle(new NotificationCompat.BigTextStyle().bigText(bigText)) |             .setStyle(new NotificationCompat.BigTextStyle().bigText(bigText)) | ||||||
|             .setContentInfo("Info"); |             .setContentInfo("Info"); | ||||||
|  |  | ||||||
|         Intent showMessageIntent = new Intent(ctx, MainActivity.class); |         builder.setContentIntent( | ||||||
|         showMessageIntent.putExtra(MainActivity.EXTRA_SHOW_MESSAGE, plaintext); |             createActivityIntent(EXTRA_SHOW_MESSAGE, plaintext)); | ||||||
|         PendingIntent pendingIntent = PendingIntent.getActivity(ctx, 0, showMessageIntent, |         builder.addAction(R.drawable.ic_action_reply, ctx.getString(R.string.reply), | ||||||
|                 PendingIntent.FLAG_UPDATE_CURRENT); |             createActivityIntent(EXTRA_REPLY_TO_MESSAGE, plaintext)); | ||||||
|         builder.setContentIntent(pendingIntent); |  | ||||||
|  |  | ||||||
|         // TODO: add proper intents to reply/delete |  | ||||||
|         builder.addAction(R.drawable.ic_action_reply, ctx.getString(R.string.reply), pendingIntent); |  | ||||||
|         builder.addAction(R.drawable.ic_action_delete, ctx.getString(R.string.delete), |         builder.addAction(R.drawable.ic_action_delete, ctx.getString(R.string.delete), | ||||||
|                 pendingIntent); |             createServiceIntent(ctx, EXTRA_DELETE_MESSAGE, plaintext)); | ||||||
|         notification = builder.build(); |         notification = builder.build(); | ||||||
|         return this; |         return this; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private PendingIntent createActivityIntent(String action, Plaintext message) { | ||||||
|  |         Intent intent = new Intent(ctx, MainActivity.class); | ||||||
|  |         intent.putExtra(action, message); | ||||||
|  |         return PendingIntent.getActivity(ctx, action.hashCode(), intent, FLAG_UPDATE_CURRENT); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private PendingIntent createServiceIntent(Context ctx, String action, Plaintext message) { | ||||||
|  |         Intent intent = new Intent(ctx, BitmessageIntentService.class); | ||||||
|  |         intent.putExtra(action, message); | ||||||
|  |         return PendingIntent.getService(ctx, action.hashCode(), intent, FLAG_UPDATE_CURRENT); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @param unacknowledged will be accessed from different threads, so make sure wherever it's |      * @param unacknowledged will be accessed from different threads, so make sure wherever it's | ||||||
|      *                       accessed it will be in a <code>synchronized(unacknowledged) |      *                       accessed it will be in a <code>synchronized(unacknowledged) | ||||||
|   | |||||||
| @@ -0,0 +1,72 @@ | |||||||
|  | /* | ||||||
|  |  * Copyright 2016 Christian Basler | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package ch.dissem.apps.abit.service; | ||||||
|  |  | ||||||
|  | import android.app.IntentService; | ||||||
|  | import android.content.Intent; | ||||||
|  |  | ||||||
|  | import ch.dissem.apps.abit.dialog.FullNodeDialogActivity; | ||||||
|  | import ch.dissem.apps.abit.util.Preferences; | ||||||
|  | import ch.dissem.bitmessage.BitmessageContext; | ||||||
|  | import ch.dissem.bitmessage.entity.Plaintext; | ||||||
|  |  | ||||||
|  | import static ch.dissem.apps.abit.MainActivity.updateNodeSwitch; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @author Christian Basler | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | public class BitmessageIntentService extends IntentService { | ||||||
|  |     public static final String EXTRA_DELETE_MESSAGE = "ch.dissem.abit.DeleteMessage"; | ||||||
|  |     public static final String EXTRA_STARTUP_NODE = "ch.dissem.abit.StartFullNode"; | ||||||
|  |     public static final String EXTRA_SHUTDOWN_NODE = "ch.dissem.abit.StopFullNode"; | ||||||
|  |  | ||||||
|  |     private BitmessageContext bmc; | ||||||
|  |  | ||||||
|  |     public BitmessageIntentService() { | ||||||
|  |         super("BitmessageIntentService"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void onCreate() { | ||||||
|  |         super.onCreate(); | ||||||
|  |         bmc = Singleton.getBitmessageContext(this); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     protected void onHandleIntent(Intent intent) { | ||||||
|  |         if (intent.hasExtra(EXTRA_DELETE_MESSAGE)) { | ||||||
|  |             Plaintext item = (Plaintext) intent.getSerializableExtra(EXTRA_DELETE_MESSAGE); | ||||||
|  |             bmc.labeler().delete(item); | ||||||
|  |             bmc.messages().save(item); | ||||||
|  |         } | ||||||
|  |         if (intent.hasExtra(EXTRA_STARTUP_NODE)) { | ||||||
|  |             if (Preferences.isConnectionAllowed(this)) { | ||||||
|  |                 startService(new Intent(this, BitmessageService.class)); | ||||||
|  |                 updateNodeSwitch(); | ||||||
|  |             } else { | ||||||
|  |                 Intent dialogIntent = new Intent(this, FullNodeDialogActivity.class); | ||||||
|  |                 dialogIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); | ||||||
|  |                 startActivity(dialogIntent); | ||||||
|  |                 sendBroadcast(new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS)); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (intent.hasExtra(EXTRA_SHUTDOWN_NODE)) { | ||||||
|  |             stopService(new Intent(this, BitmessageService.class)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -18,14 +18,14 @@ package ch.dissem.apps.abit.service; | |||||||
|  |  | ||||||
| import android.app.Service; | import android.app.Service; | ||||||
| import android.content.Intent; | import android.content.Intent; | ||||||
| import android.os.Binder; |  | ||||||
| import android.os.IBinder; | import android.os.IBinder; | ||||||
|  | import android.support.annotation.Nullable; | ||||||
|  |  | ||||||
| import ch.dissem.apps.abit.notification.NetworkNotification; | import ch.dissem.apps.abit.notification.NetworkNotification; | ||||||
| import ch.dissem.bitmessage.BitmessageContext; | import ch.dissem.bitmessage.BitmessageContext; | ||||||
| import ch.dissem.bitmessage.utils.Property; | import ch.dissem.bitmessage.utils.Property; | ||||||
|  |  | ||||||
| import static ch.dissem.apps.abit.notification.NetworkNotification.ONGOING_NOTIFICATION_ID; | import static ch.dissem.apps.abit.notification.NetworkNotification.NETWORK_NOTIFICATION_ID; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Define a Service that returns an IBinder for the |  * Define a Service that returns an IBinder for the | ||||||
| @@ -44,55 +44,41 @@ public class BitmessageService extends Service { | |||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onCreate() { |     public void onCreate() { | ||||||
|         synchronized (BitmessageService.class) { |  | ||||||
|         if (bmc == null) { |         if (bmc == null) { | ||||||
|             bmc = Singleton.getBitmessageContext(this); |             bmc = Singleton.getBitmessageContext(this); | ||||||
|         } |         } | ||||||
|         notification = new NetworkNotification(this); |         notification = new NetworkNotification(this); | ||||||
|         } |         running = false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public int onStartCommand(Intent intent, int flags, int startId) { |     public int onStartCommand(Intent intent, int flags, int startId) { | ||||||
|         return Service.START_STICKY; |         if (!isRunning()) { | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void onDestroy() { |  | ||||||
|         if (bmc.isRunning()) bmc.shutdown(); |  | ||||||
|         running = false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Return an object that allows the system to invoke |  | ||||||
|      * the sync adapter. |  | ||||||
|      */ |  | ||||||
|     @Override |  | ||||||
|     public IBinder onBind(Intent intent) { |  | ||||||
|         return new BitmessageBinder(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public class BitmessageBinder extends Binder { |  | ||||||
|         public void startupNode() { |  | ||||||
|             startService(new Intent(BitmessageService.this, BitmessageService.class)); |  | ||||||
|             running = true; |             running = true; | ||||||
|             notification.connecting(); |             notification.connecting(); | ||||||
|             startForeground(ONGOING_NOTIFICATION_ID, notification.getNotification()); |             startForeground(NETWORK_NOTIFICATION_ID, notification.getNotification()); | ||||||
|             if (!bmc.isRunning()) { |             if (!bmc.isRunning()) { | ||||||
|                 bmc.startup(); |                 bmc.startup(); | ||||||
|             } |             } | ||||||
|             notification.show(); |             notification.show(); | ||||||
|         } |         } | ||||||
|  |         return Service.START_STICKY; | ||||||
|  |     } | ||||||
|  |  | ||||||
|         public void shutdownNode() { |     @Override | ||||||
|  |     public void onDestroy() { | ||||||
|         if (bmc.isRunning()) { |         if (bmc.isRunning()) { | ||||||
|             bmc.shutdown(); |             bmc.shutdown(); | ||||||
|         } |         } | ||||||
|         running = false; |         running = false; | ||||||
|             stopForeground(true); |         notification.showShutdown(); | ||||||
|             notification.show(); |  | ||||||
|         stopSelf(); |         stopSelf(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Nullable | ||||||
|  |     @Override | ||||||
|  |     public IBinder onBind(Intent intent) { | ||||||
|  |         return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static Property getStatus() { |     public static Property getStatus() { | ||||||
|   | |||||||
							
								
								
									
										25
									
								
								app/src/main/res/drawable/ic_notification_node_start.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								app/src/main/res/drawable/ic_notification_node_start.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | <!-- | ||||||
|  |   ~ Copyright 2016 Christian Basler | ||||||
|  |   ~ | ||||||
|  |   ~ 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. | ||||||
|  |   --> | ||||||
|  |  | ||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |         android:width="24dp" | ||||||
|  |         android:height="24dp" | ||||||
|  |         android:viewportWidth="24.0" | ||||||
|  |         android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FFFFFFFF" | ||||||
|  |         android:pathData="M8,5v14l11,-7z"/> | ||||||
|  | </vector> | ||||||
							
								
								
									
										25
									
								
								app/src/main/res/drawable/ic_notification_node_stop.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								app/src/main/res/drawable/ic_notification_node_stop.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | <!-- | ||||||
|  |   ~ Copyright 2016 Christian Basler | ||||||
|  |   ~ | ||||||
|  |   ~ 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. | ||||||
|  |   --> | ||||||
|  |  | ||||||
|  | <vector xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |         android:width="24dp" | ||||||
|  |         android:height="24dp" | ||||||
|  |         android:viewportWidth="24.0" | ||||||
|  |         android:viewportHeight="24.0"> | ||||||
|  |     <path | ||||||
|  |         android:fillColor="#FFFFFFFF" | ||||||
|  |         android:pathData="M6,19h4L10,5L6,5v14zM14,5v14h4L18,5h-4z"/> | ||||||
|  | </vector> | ||||||
| @@ -28,7 +28,7 @@ | |||||||
|  |  | ||||||
|     <TextView |     <TextView | ||||||
|         android:id="@+id/description" |         android:id="@+id/description" | ||||||
|         android:layout_width="match_parent" |         android:layout_width="0dp" | ||||||
|         android:layout_height="wrap_content" |         android:layout_height="wrap_content" | ||||||
|         android:text="@string/add_identity_warning" |         android:text="@string/add_identity_warning" | ||||||
|         app:layout_constraintLeft_toLeftOf="parent" |         app:layout_constraintLeft_toLeftOf="parent" | ||||||
| @@ -39,7 +39,7 @@ | |||||||
|  |  | ||||||
|     <RadioGroup |     <RadioGroup | ||||||
|         android:id="@+id/radioGroup" |         android:id="@+id/radioGroup" | ||||||
|         android:layout_width="match_parent" |         android:layout_width="0dp" | ||||||
|         android:layout_height="wrap_content" |         android:layout_height="wrap_content" | ||||||
|         android:paddingBottom="24dp" |         android:paddingBottom="24dp" | ||||||
|         android:paddingTop="24dp" |         android:paddingTop="24dp" | ||||||
| @@ -86,7 +86,6 @@ | |||||||
|         app:layout_constraintHorizontal_bias="1.0" |         app:layout_constraintHorizontal_bias="1.0" | ||||||
|         app:layout_constraintRight_toRightOf="@+id/radioGroup" |         app:layout_constraintRight_toRightOf="@+id/radioGroup" | ||||||
|         app:layout_constraintTop_toBottomOf="@+id/radioGroup" |         app:layout_constraintTop_toBottomOf="@+id/radioGroup" | ||||||
|         tools:layout_constraintLeft_creator="1" |  | ||||||
|         tools:layout_constraintRight_creator="1" |         tools:layout_constraintRight_creator="1" | ||||||
|         tools:layout_constraintTop_creator="1"/> |         tools:layout_constraintTop_creator="1"/> | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										60
									
								
								app/src/main/res/layout/dialog_full_node.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								app/src/main/res/layout/dialog_full_node.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <!-- | ||||||
|  |   ~ Copyright 2016 Christian Basler | ||||||
|  |   ~ | ||||||
|  |   ~ 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. | ||||||
|  |   --> | ||||||
|  |  | ||||||
|  | <android.support.constraint.ConstraintLayout | ||||||
|  |     xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     xmlns:app="http://schemas.android.com/apk/res-auto" | ||||||
|  |     xmlns:tools="http://schemas.android.com/tools" | ||||||
|  |     android:layout_width="336dp" | ||||||
|  |     android:layout_height="wrap_content"> | ||||||
|  |  | ||||||
|  |     <TextView | ||||||
|  |         android:id="@+id/description" | ||||||
|  |         android:layout_width="320dp" | ||||||
|  |         android:layout_height="wrap_content" | ||||||
|  |         android:padding="24dp" | ||||||
|  |         android:text="@string/full_node_warning" | ||||||
|  |         app:layout_constraintStart_toStartOf="parent" | ||||||
|  |         app:layout_constraintTop_toTopOf="parent" | ||||||
|  |         tools:layout_editor_absoluteX="0dp"/> | ||||||
|  |  | ||||||
|  |     <Button | ||||||
|  |         android:id="@+id/ok" | ||||||
|  |         style="?android:attr/borderlessButtonStyle" | ||||||
|  |         android:layout_width="wrap_content" | ||||||
|  |         android:layout_height="wrap_content" | ||||||
|  |         android:padding="8dp" | ||||||
|  |         android:text="@string/startup_node" | ||||||
|  |         android:textColor="@color/colorAccent" | ||||||
|  |         app:layout_constraintEnd_toEndOf="@+id/description" | ||||||
|  |         app:layout_constraintHorizontal_bias="1.0" | ||||||
|  |         app:layout_constraintTop_toBottomOf="@+id/description" | ||||||
|  |         tools:layout_editor_absoluteX="184dp"/> | ||||||
|  |  | ||||||
|  |     <Button | ||||||
|  |         android:id="@+id/dismiss" | ||||||
|  |         style="?android:attr/borderlessButtonStyle" | ||||||
|  |         android:layout_width="wrap_content" | ||||||
|  |         android:layout_height="wrap_content" | ||||||
|  |         android:padding="8dp" | ||||||
|  |         android:text="@string/cancel" | ||||||
|  |         android:textColor="@color/colorAccent" | ||||||
|  |         app:layout_constraintEnd_toStartOf="@+id/ok" | ||||||
|  |         app:layout_constraintTop_toBottomOf="@+id/description" | ||||||
|  |         tools:ignore="RtlSymmetry" | ||||||
|  |         tools:layout_editor_absoluteX="87dp"/> | ||||||
|  | </android.support.constraint.ConstraintLayout> | ||||||
| @@ -96,4 +96,7 @@ Als Alternative kann in den Einstellungen ein vertrauenswürdiger Knoten konfigu | |||||||
|     <string name="select_file_title">Datei auswählen</string> |     <string name="select_file_title">Datei auswählen</string> | ||||||
|     <string name="select_identities_to_import">Bitte wähle die zu importierenden Identitäten:</string> |     <string name="select_identities_to_import">Bitte wähle die zu importierenden Identitäten:</string> | ||||||
|     <string name="import_input_description">Du kannst einfach den Inhalt eines Exports oder einer ‘keys.dat’-Datei einfügen</string> |     <string name="import_input_description">Du kannst einfach den Inhalt eines Exports oder einer ‘keys.dat’-Datei einfügen</string> | ||||||
|  |     <string name="full_node_restart">Knoten starten</string> | ||||||
|  |     <string name="full_node_stop">Knoten beenden</string> | ||||||
|  |     <string name="startup_node">Knoten starten</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
| @@ -95,4 +95,7 @@ As an alternative you could configure a trusted node in the settings, but as of | |||||||
|     <string name="select_file_title">Select a File</string> |     <string name="select_file_title">Select a File</string> | ||||||
|     <string name="select_identities_to_import">Please select the identities you want to import:</string> |     <string name="select_identities_to_import">Please select the identities you want to import:</string> | ||||||
|     <string name="import_input_description">You can just paste the contents of an export or a ‘keys.dat’ file</string> |     <string name="import_input_description">You can just paste the contents of an export or a ‘keys.dat’ file</string> | ||||||
|  |     <string name="full_node_stop">shutdown node</string> | ||||||
|  |     <string name="full_node_restart">restart node</string> | ||||||
|  |     <string name="startup_node">Startup node</string> | ||||||
| </resources> | </resources> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user