org.jitsi.service.osgi.OSGiActivity.java Source code

Java tutorial

Introduction

Here is the source code for org.jitsi.service.osgi.OSGiActivity.java

Source

/*
 * Jitsi, the OpenSource Java VoIP and Instant Messaging client.
 *
 * Distributable under LGPL license.
 * See terms of license at gnu.org.
 */
package org.jitsi.service.osgi;

import android.annotation.*;
import android.app.*;
import android.content.*;
import android.os.*;
import android.os.Bundle; // disambiguation

import android.support.v4.app.*;
import android.view.*;
import net.java.sip.communicator.util.*;
import org.jitsi.*;
import org.jitsi.android.*;
import org.jitsi.android.gui.*;
import org.jitsi.android.gui.LauncherActivity;
import org.jitsi.android.gui.util.*;
import org.jitsi.android.plugin.errorhandler.*;
import org.osgi.framework.*;

import java.util.*;

/**
 * Implements a base <tt>Activity</tt> which employs OSGi.
 *
 * @author Lyubomir Marinov
 * @author Pawel Domas
 */
public class OSGiActivity extends FragmentActivity {
    /**
     * The logger
     */
    private final static Logger logger = Logger.getLogger(OSGiActivity.class);

    /**
     * UI thread handler
     */
    public final static Handler uiHandler = new Handler(Looper.getMainLooper());

    private BundleActivator bundleActivator;

    private BundleContext bundleContext;

    private BundleContextHolder service;

    private ServiceConnection serviceConnection;

    /**
     * EXIT action listener that triggers closes the <tt>Activity</tt>
     */
    private ExitActionListener exitListener = new ExitActionListener();

    /**
     * List of attached {@link OSGiUiPart}.
     */
    private List<OSGiUiPart> osgiFrgaments = new ArrayList<OSGiUiPart>();

    /**
     * Starts this osgi activity.
     *
     * @param bundleContext the osgi <tt>BundleContext</tt>
     * @throws Exception
     */
    private void internalStart(BundleContext bundleContext) throws Exception {
        this.bundleContext = bundleContext;

        boolean start = false;

        try {
            start(bundleContext);
            start = true;
        } finally {
            if (!start && (this.bundleContext == bundleContext))
                this.bundleContext = null;
        }
    }

    /**
     * Stops this osgi activity.
     *
     * @param bundleContext the osgi <tt>BundleContext</tt>
     * @throws Exception
     */
    private void internalStop(BundleContext bundleContext) throws Exception {
        if (this.bundleContext != null) {
            if (bundleContext == null)
                bundleContext = this.bundleContext;
            if (this.bundleContext == bundleContext)
                this.bundleContext = null;
            stop(bundleContext);
        }
    }

    /**
     * Called when the activity is starting. Initializes the corresponding
     * call interface.
     *
     * @param savedInstanceState If the activity is being re-initialized after
     * previously being shut down then this Bundle contains the data it most
     * recently supplied in onSaveInstanceState(Bundle).
     * Note: Otherwise it is null.
     */
    @TargetApi(Build.VERSION_CODES.ICE_CREAM_SANDWICH)
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // Hooks the exception handler to the UI thread
        ExceptionHandler.checkAndAttachExceptionHandler();

        if (AndroidUtils.hasAPI(11)) {
            ActionBar actionBar = getActionBar();
            if (actionBar != null) {

                // Disable up arrow on home activity
                Class<?> homeActivity = JitsiApplication.getHomeScreenActivityClass();
                if (this.getClass().equals(homeActivity)) {
                    actionBar.setDisplayHomeAsUpEnabled(false);

                    if (AndroidUtils.hasAPI(14)) {
                        actionBar.setHomeButtonEnabled(false);
                    }
                }

                ActionBarUtil.setTitle(this, getTitle());
            }
        }

        super.onCreate(savedInstanceState);

        ServiceConnection serviceConnection = new ServiceConnection() {
            public void onServiceConnected(ComponentName name, IBinder service) {
                if (this == OSGiActivity.this.serviceConnection)
                    setService((BundleContextHolder) service);
            }

            public void onServiceDisconnected(ComponentName name) {
                if (this == OSGiActivity.this.serviceConnection)
                    setService(null);
            }
        };

        this.serviceConnection = serviceConnection;

        boolean bindService = false;

        try {
            bindService = bindService(new Intent(this, OSGiService.class), serviceConnection, BIND_AUTO_CREATE);
        } finally {
            if (!bindService)
                this.serviceConnection = null;
        }

        // Registers exit action listener
        this.registerReceiver(exitListener, new IntentFilter(JitsiApplication.ACTION_EXIT));
    }

    /**
     * Called when an activity is destroyed.
     */
    @Override
    protected void onDestroy() {
        // Unregisters exit action listener
        unregisterReceiver(exitListener);

        ServiceConnection serviceConnection = this.serviceConnection;

        this.serviceConnection = null;
        try {
            setService(null);
        } finally {
            if (serviceConnection != null)
                unbindService(serviceConnection);
        }

        super.onDestroy();
    }

    protected void onPause() {
        // Clear the references to this activity.
        clearReferences();

        super.onPause();
    }

    protected void onResume() {
        super.onResume();

        JitsiApplication.setCurrentActivity(this);
        // If OSGi service is running check for send logs
        if (bundleContext != null) {
            checkForSendLogsDialog();
        }
    }

    /**
     * Checks if the crash has occurred since the Jitsi was last started.
     * If it's true asks the user about eventual logs report.
     */
    private void checkForSendLogsDialog() {
        // Checks if Jitsi has previously crashed and asks the user user
        // about log reporting
        if (!ExceptionHandler.hasCrashed()) {
            return;
        }
        // Clears the crash status
        ExceptionHandler.resetCrashedStatus();
        // Asks the user
        AlertDialog.Builder question = new AlertDialog.Builder(this);
        question.setTitle(R.string.service_gui_WARNING)
                .setMessage(getString(R.string.service_gui_SEND_LOGS_QUESTION))
                .setPositiveButton(R.string.service_gui_YES, new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        dialog.dismiss();
                        JitsiApplication.showSendLogsDialog();
                    }
                }).setNegativeButton(R.string.service_gui_NO, new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        dialog.dismiss();
                    }
                }).create().show();

    }

    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);

        JitsiApplication.setCurrentActivity(this);
    }

    private void setService(BundleContextHolder service) {
        if (this.service != service) {
            if ((this.service != null) && (bundleActivator != null)) {
                try {
                    this.service.removeBundleActivator(bundleActivator);
                    bundleActivator = null;
                } finally {
                    try {
                        internalStop(null);
                    } catch (Throwable t) {
                        if (t instanceof ThreadDeath)
                            throw (ThreadDeath) t;
                    }
                }
            }

            this.service = service;

            if (this.service != null) {
                if (bundleActivator == null) {
                    bundleActivator = new BundleActivator() {
                        public void start(BundleContext bundleContext) throws Exception {
                            internalStart(bundleContext);
                        }

                        public void stop(BundleContext bundleContext) throws Exception {
                            internalStop(bundleContext);
                        }
                    };
                }
                this.service.addBundleActivator(bundleActivator);
            }
        }
    }

    protected void start(BundleContext bundleContext) throws Exception {
        // Starts children OSGI fragments.
        for (OSGiUiPart osGiFragment : osgiFrgaments) {
            osGiFragment.start(bundleContext);
        }
        // If OSGi has just started and we're on UI thread check for crash event
        // We must be on UIThread to show the dialog and it makes no sense
        // to show it from the background, so it will be eventually displayed
        // from onResume()
        if (Looper.getMainLooper() == Looper.myLooper()) {
            checkForSendLogsDialog();
        }
    }

    protected void stop(BundleContext bundleContext) throws Exception {
        // Stops children OSGI fragments.
        for (OSGiUiPart osGiFragment : osgiFrgaments) {
            osGiFragment.stop(bundleContext);
        }
    }

    /**
     * Registers child <tt>OSGiUiPart</tt> to be notified on startup.
     * @param fragment child <tt>OSGiUiPart</tt> contained in this
     *        <tt>Activity</tt>.
     */
    public void registerOSGiFragment(OSGiUiPart fragment) {
        osgiFrgaments.add(fragment);

        if (bundleContext != null) {
            // If context exists it means we have started already,
            // so start the fragment immediately
            try {
                fragment.start(bundleContext);
            } catch (Exception e) {
                logger.error("Error starting OSGiFragment", e);
            }
        }
    }

    /**
     * Unregisters child <tt>OSGiUiPart</tt>.
     *
     * @param fragment the <tt>OSGiUiPart</tt> that will be unregistered.
     */
    public void unregisterOSGiFragment(OSGiUiPart fragment) {
        if (bundleContext != null) {
            try {
                fragment.stop(bundleContext);
            } catch (Exception e) {
                logger.error("Error while trying to stop OSGiFragment", e);
            }
        }
        osgiFrgaments.remove(fragment);
    }

    /**
     * Convenience method which starts a new activity
     * for given <tt>activityClass</tt> class
     *
     * @param activityClass the activity class
     */
    protected void startActivity(Class<?> activityClass) {
        Intent intent = new Intent(this, activityClass);
        startActivity(intent);
    }

    /**
     * Convenience method that switches from one activity to another.
     *
     * @param activityClass the activity class
     */
    protected void switchActivity(Class<?> activityClass) {
        startActivity(activityClass);
        finish();
    }

    /**
     * Convenience method that switches from one activity to another.
     *
     * @param activityIntent the next activity <tt>Intent</tt>
     */
    protected void switchActivity(Intent activityIntent) {
        startActivity(activityIntent);
        finish();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle home action
        switch (item.getItemId()) {
        case android.R.id.home:
            Class<?> homeActivity = JitsiApplication.getHomeScreenActivityClass();
            if (!this.getClass().equals(homeActivity)) {
                switchActivity(homeActivity);
            }
            return true;
        default:
            return super.onOptionsItemSelected(item);
        }
    }

    /**
     * Returns OSGI <tt>BundleContext</tt>.
     * @return OSGI <tt>BundleContext</tt>.
     */
    protected BundleContext getBundlecontext() {
        return bundleContext;
    }

    /**
     * Returns the content <tt>View</tt>.
     * @return the content <tt>View</tt>.
     */
    protected View getContentView() {
        return findViewById(android.R.id.content);
    }

    /**
     * Checks if the OSGi is started and if not eventually triggers
     * <tt>LauncherActivity</tt> that will restore current activity from it's
     * <tt>Intent</tt>.
     * @return <tt>true</tt> if restore <tt>Intent</tt> has been posted.
     */
    protected boolean postRestoreIntent() {
        // Restore after OSGi startup
        if (AndroidGUIActivator.bundleContext == null) {
            Intent intent = new Intent(JitsiApplication.getGlobalContext(), LauncherActivity.class);
            intent.putExtra(LauncherActivity.ARG_RESTORE_INTENT, getIntent());
            startActivity(intent);
            finish();
            return true;
        }
        return false;
    }

    /**
     * Broadcast listener that listens for {@link JitsiApplication#ACTION_EXIT}
     * and then finishes this <tt>Activity</tt>.
     * 
     */
    class ExitActionListener extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            finish();
        }
    }

    private void clearReferences() {
        Activity currentActivity = JitsiApplication.getCurrentActivity();
        if (currentActivity != null && currentActivity.equals(this))
            JitsiApplication.setCurrentActivity(null);
    }
}