there seem to be two definitions for full-screen
android full screen = no taskbar, no status bar or buttons, simply fill the screen
android plot full-screen = no legends left and right and on the bottom, simply a full-screen graph widget
this question is related to the Android full-screen with the swipe action to make the taskbar and button-bar hid (full-screen) or displayed ... it does not extend (fill) the graph portion down to the legend at the bottom
the following is based on Android plot 1.4.3
creating a new android studio 2.3.2 project with an empty activity and copying the Android plot demo app's "simple xy plot" java code into onCreate and the xml portion into the layout xml file works fine and the graph portion is expanded all the way to the bottom legend
creating a new android studio 2.3.2 project with a full-screen activity (with swipe action to hide/show taskbar and button-bar) setup fails to expand the graph portion down to the bottom legend
the android studio generated default activity code framework is now more complex as compared to the first example with an empty activity
lines are shown with "// zzz ..." indicate deviations from android studio generated code
full-screen activity java code start
package com.efilabs.plotxynew;
import android.annotation.SuppressLint;
import android.support.v7.app.ActionBar;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.os.Handler;
import android.view.MotionEvent;
import android.view.View;
import com.androidplot.xy.XYPlot;
/**
* An example full-screen activity that shows and hides the system UI (i.e.
* status bar and navigation/system bar) with user interaction.
*/
public class ActPlotNew0 extends AppCompatActivity
{
/**
* Whether or not the system UI should be auto-hidden after
* {@link #AUTO_HIDE_DELAY_MILLIS} milliseconds.
*/
private static final boolean AUTO_HIDE = true;
/**
* If {@link #AUTO_HIDE} is set, the number of milliseconds to wait after
* user interaction before hiding the system UI.
*/
private static final int AUTO_HIDE_DELAY_MILLIS = 3000;
/**
* Some older devices needs a small delay between UI widget updates
* and a change of the status and navigation bar.
*/
private static final int UI_ANIMATION_DELAY = 300;
private final Handler mHideHandler = new Handler ();
private View mContentView; // zzz
//private XYPlot mContentView; // zzz
private final Runnable mHidePart2Runnable = new Runnable ()
{
@SuppressLint ("InlinedApi")
@Override
public void run ()
{
// Delayed removal of status and navigation bar
// Note that some of these constants are new as of API 16 (Jelly Bean)
// and API 19 (KitKat). It is safe to use them, as they are inclined
// at compile-time and do nothing on earlier devices.
mContentView.setSystemUiVisibility (View.SYSTEM_UI_FLAG_LOW_PROFILE
| View.SYSTEM_UI_FLAG_FULLSCREEN
| View.SYSTEM_UI_FLAG_LAYOUT_STABLE
| View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY
| View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
| View.SYSTEM_UI_FLAG_HIDE_NAVIGATION);
}
};
private View mControlsView;
private final Runnable mShowPart2Runnable = new Runnable ()
{
@Override
public void run ()
{
// Delayed display of UI elements
ActionBar actionBar = getSupportActionBar ();
if (actionBar != null) {
actionBar.show ();
}
mControlsView.setVisibility (View.VISIBLE);
}
};
private boolean mVisible;
private final Runnable mHideRunnable = new Runnable ()
{
@Override
public void run ()
{
hide ();
}
};
/**
* Touch listener to use for in-layout UI controls to delay hiding the
* system UI. This is to prevent the jarring behavior of controls going away
* while interacting with activity UI.
*/
private final View.OnTouchListener mDelayHideTouchListener = new View.OnTouchListener ()
{
@Override
public boolean onTouch (View view, MotionEvent motionEvent)
{
if (AUTO_HIDE) {
delayedHide (AUTO_HIDE_DELAY_MILLIS);
}
return false;
}
};
@Override
protected void onCreate (Bundle savedInstanceState)
{
super.onCreate (savedInstanceState);
setContentView (R.layout.act_plotnew0);
mVisible = true;
mControlsView = findViewById (R.id.fullscreen_content_controls);
mContentView = findViewById (R.id.fullscreen_content); // zzz
// mContentView = (XYPlot) findViewById (R.id.fullscreen_content); // zzz
// Set up the user interaction to manually show or hide the system UI.
mContentView.setOnClickListener (new View.OnClickListener ()
{
@Override
public void onClick (View view)
{
toggle ();
}
});
// Upon interacting with UI controls, delay any scheduled hide()
// operations to prevent the jarring behavior of controls going away
// while interacting with the UI.
findViewById (R.id.dummy_button).setOnTouchListener (mDelayHideTouchListener);
}
@Override
protected void onPostCreate (Bundle savedInstanceState)
{
super.onPostCreate (savedInstanceState);
// Trigger the initial hide() shortly after the activity has been
// created, to briefly hint to the user that UI controls
// are available.
delayedHide (100);
}
private void toggle ()
{
if (mVisible) {
hide ();
} else {
show ();
}
}
private void hide ()
{
// Hide UI first
ActionBar actionBar = getSupportActionBar ();
if (actionBar != null) {
actionBar.hide ();
}
mControlsView.setVisibility (View.GONE);
mVisible = false;
// Schedule a runnable to remove the status and navigation bar after a delay
mHideHandler.removeCallbacks (mShowPart2Runnable);
mHideHandler.postDelayed (mHidePart2Runnable, UI_ANIMATION_DELAY);
}
@SuppressLint ("InlinedApi")
private void show ()
{
// Show the system bar
mContentView.setSystemUiVisibility (View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
| View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION);
mVisible = true;
// Schedule a runnable to display UI elements after a delay
mHideHandler.removeCallbacks (mHidePart2Runnable);
mHideHandler.postDelayed (mShowPart2Runnable, UI_ANIMATION_DELAY);
}
/**
* Schedules a call to hide() in [delay] milliseconds, canceling any
* previously scheduled calls.
*/
private void delayedHide (int delayMillis)
{
mHideHandler.removeCallbacks (mHideRunnable);
mHideHandler.postDelayed (mHideRunnable, delayMillis);
}
}
full-screen activity java code end
XML layout contains inserted android plot view portion
full-screen activity layout XML start
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
xmlns:ap="http://schemas.android.com/apk/res-auto"
android:background="#0099cc"
tools:context="com.efilabs.plotxynew.ActPlotNew0">
<!-- The primary full-screen view. This can be replaced with whatever view
is needed to present your content, e.g. VideoView, SurfaceView,
TextureView, etc. -->
<!--TextView
android:id="@+id/fullscreen_content"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center"
android:keepScreenOn="true"
android:text="@string/dummy_content"
android:textColor="#33b5e5"
android:textSize="50sp"
android:textStyle="bold"/-->
<com.androidplot.xy.XYPlot
style="@style/APDefacto.Light"
android:id="@+id/fullscreen_content"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
ap:title="A Simple XY Plot"
ap:rangeTitle="range"
ap:domainTitle="domain"
ap:lineLabels="left|bottom"
ap:lineLabelRotationBottom="-45"/>
<!-- This FrameLayout insets its children based on system windows using
android:fitsSystemWindows. -->
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fitsSystemWindows="true">
<LinearLayout
android:id="@+id/fullscreen_content_controls"
style="?metaButtonBarStyle"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="bottom|center_horizontal"
android:background="@color/black_overlay"
android:orientation="horizontal"
tools:ignore="UselessParent">
<Button
android:id="@+id/dummy_button"
style="?metaButtonBarButtonStyle"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="@string/dummy_button"/>
</LinearLayout>
</FrameLayout>
</FrameLayout>
full-screen activity layout XML end
To not mess with the Android generated java code, I created a 2nd java activity class extending the Android generated java activity code and placed the "simple XY plot" into it
extended activity code start
package com.efilabs.plotxynew;
import android.graphics.DashPathEffect;
import android.os.Bundle;
import android.support.v7.app.ActionBar;
import com.androidplot.util.PixelUtils;
import com.androidplot.xy.CatmullRomInterpolator;
import com.androidplot.xy.LineAndPointFormatter;
import com.androidplot.xy.SimpleXYSeries;
import com.androidplot.xy.XYGraphWidget;
import com.androidplot.xy.XYPlot;
import com.androidplot.xy.XYSeries;
import java.text.FieldPosition;
import java.text.Format;
import java.text.ParsePosition;
import java.util.Arrays;
public class ActPlotNew0a extends ActPlotNew0
{
private XYPlot plot;
@Override
protected void onCreate (Bundle savedInstanceState)
{
super.onCreate (savedInstanceState);
// setContentView (R.layout.act_plotnew0); // xxx provides full screen but swipe doesn't work anylonger
ActionBar bar = getSupportActionBar () ;
if (bar != null) bar.hide () ;
// initialize our XYPlot reference:
plot = (XYPlot) findViewById(R.id.fullscreen_content);
// create a couple arrays of y-values to plot:
final Number[] domainLabels = {1, 2, 3, 6, 7, 8, 9, 10, 13, 14};
Number[] series1Numbers = {1, 4, 2, 8, 4, 16, 8, 32, 16, 64};
Number[] series2Numbers = {5, 2, 10, 5, 20, 10, 40, 20, 80, 40};
// turn the above arrays into XYSeries':
// (Y_VALS_ONLY means use the element index as the x value)
XYSeries series1 = new SimpleXYSeries (
Arrays.asList(series1Numbers), SimpleXYSeries.ArrayFormat.Y_VALS_ONLY, "Series1");
XYSeries series2 = new SimpleXYSeries(
Arrays.asList(series2Numbers), SimpleXYSeries.ArrayFormat.Y_VALS_ONLY, "Series2");
// create formatters to use for drawing a series using LineAndPointRenderer
// and configure them from xml:
LineAndPointFormatter series1Format =
new LineAndPointFormatter(this, R.xml.line_point_formatter_with_labels);
LineAndPointFormatter series2Format =
new LineAndPointFormatter(this, R.xml.line_point_formatter_with_labels_2);
// add an "dash" effect to the series2 line:
series2Format.getLinePaint().setPathEffect(new DashPathEffect (new float[]
{
// always use DP when specifying pixel sizes, to keep things consistent across devices:
PixelUtils.dpToPix(20),
PixelUtils.dpToPix(15)
}, 0));
// (optional) add some smoothing to the lines:
// see: http://androidplot.com/smooth-curves-and-androidplot/
series1Format.setInterpolationParams(
new CatmullRomInterpolator.Params(10, CatmullRomInterpolator.Type.Centripetal));
series2Format.setInterpolationParams(
new CatmullRomInterpolator.Params(10, CatmullRomInterpolator.Type.Centripetal));
// add a new series' to the xyplot:
plot.addSeries(series1, series1Format);
plot.addSeries(series2, series2Format);
plot.getGraph().getLineLabelStyle(XYGraphWidget.Edge.BOTTOM).setFormat(new Format ()
{
@Override
public StringBuffer format(Object obj, StringBuffer toAppendTo, FieldPosition pos)
{
int i = Math.round(((Number) obj).floatValue());
return toAppendTo.append(domainLabels[i]);
}
@Override
public Object parseObject(String source, ParsePosition pos)
{
return null;
}
});
}
}
extended activity code end
the funny part is, un-commenting line 28 containing "... // xxx ..." makes the full screen work, but now the swipe gesture to switch between hiding/show taskbar and button-bar" does not work anymore
I'm not experienced enough to resolve this issue and would need some serious help
Note: the above scenario would work fine with Android plot 0.9.8 and would provide a full-screen Android plot graph portion filling down to the bottom legend
The root cause of the issue is the call to setSystemUiVisibility(...)
after the plot has been laid out and drawn. This was not a problem in 0.9.8 because XYGraphWidget
would manually re-calculate it's dimensions on every draw. This is an expensive and (usually) unnecessary operation though and was removed in later releases for performance reasons.
I know this doesn't address your use case (which is a valid one) but if you get rid of the postDelayed and move your full screen adjustment code into onCreate, then fullscreen mode will display correctly.
I'm working on an alternate implementation of the graph render logic that will hopefully avoid the performance issue while supporting your use case, for the 1.4.5 release. It should not require any code changes on your end.
In the mean time, an easy fix is to do the following immediately after setSystemUiVisibility
:
plot.getGraph().setGridRect(null);
plot.redraw();