Snap for 4957512 from 1aabc2cc4e to qt-release
Change-Id: I7d10e174dfee3fc2f41c2d4b024a091b95ee2dfd
This commit is contained in:
@@ -1892,7 +1892,8 @@
|
||||
android:name=".development.DevelopmentSettingsDisabledActivity"
|
||||
android:icon="@drawable/ic_settings_development"
|
||||
android:label="@string/development_settings_title"
|
||||
android:theme="@android:style/Theme.NoDisplay">
|
||||
android:excludeFromRecents="true"
|
||||
android:theme="@style/Transparent">
|
||||
<intent-filter android:priority="-1">
|
||||
<action android:name="android.settings.APPLICATION_DEVELOPMENT_SETTINGS" />
|
||||
<action android:name="com.android.settings.APPLICATION_DEVELOPMENT_SETTINGS" />
|
||||
@@ -3069,10 +3070,12 @@
|
||||
</intent-filter>
|
||||
</service>
|
||||
|
||||
<activity android:name=".HelpTrampoline"
|
||||
<activity
|
||||
android:name=".HelpTrampoline"
|
||||
android:exported="true"
|
||||
android:theme="@android:style/Theme.NoDisplay"
|
||||
android:theme="@style/Transparent"
|
||||
android:permission="android.permission.DUMP"
|
||||
android:excludeFromRecents="true"
|
||||
android:enabled="@bool/config_has_help" />
|
||||
|
||||
<activity android:name=".applications.autofill.AutofillPickerActivity"
|
||||
|
||||
@@ -136,5 +136,11 @@
|
||||
<color name="battery_maybe_color_dark">#fdd835</color> <!-- Material Yellow 600 -->
|
||||
<color name="battery_bad_color_dark">#f44336</color> <!-- Material Red 500 -->
|
||||
|
||||
<!-- TODO: Figure out colors -->
|
||||
<color name="face_anim_particle_color_1">#ff00bcd4</color> <!-- Material Cyan 500 -->
|
||||
<color name="face_anim_particle_color_2">#ffef6c00</color> <!-- Material Orange 800 -->
|
||||
<color name="face_anim_particle_color_3">#ff4caf50</color> <!-- Material Green 500 -->
|
||||
<color name="face_anim_particle_color_4">#fffdd835</color> <!-- Material Yellow 600 -->
|
||||
<color name="face_anim_particle_error">#ff9e9e9e</color> <!-- Material Gray 500 -->
|
||||
</resources>
|
||||
|
||||
|
||||
@@ -15,7 +15,9 @@
|
||||
limitations under the License.
|
||||
-->
|
||||
|
||||
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<PreferenceScreen
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:settings="http://schemas.android.com/apk/res-auto">
|
||||
|
||||
<com.android.settings.applications.LayoutPreference
|
||||
android:key="header_view"
|
||||
@@ -30,10 +32,11 @@
|
||||
<PreferenceCategory
|
||||
android:title="@string/battery_detail_manage_title">
|
||||
|
||||
<Preference
|
||||
<com.android.settingslib.RestrictedPreference
|
||||
android:key="background_activity"
|
||||
android:title="@string/background_activity_title"
|
||||
android:selectable="true"/>
|
||||
android:selectable="true"
|
||||
settings:userRestriction="no_control_apps"/>
|
||||
|
||||
<Preference
|
||||
android:key="battery_optimization"
|
||||
|
||||
@@ -37,7 +37,7 @@ import java.util.ArrayList;
|
||||
public abstract class BiometricEnrollSidecar extends InstrumentedFragment {
|
||||
|
||||
public interface Listener {
|
||||
void onEnrollmentHelp(CharSequence helpString);
|
||||
void onEnrollmentHelp(int helpMsgId, CharSequence helpString);
|
||||
void onEnrollmentError(int errMsgId, CharSequence errString);
|
||||
void onEnrollmentProgressChange(int steps, int remaining);
|
||||
}
|
||||
@@ -82,7 +82,7 @@ public abstract class BiometricEnrollSidecar extends InstrumentedFragment {
|
||||
|
||||
@Override
|
||||
public void send(Listener listener) {
|
||||
listener.onEnrollmentHelp(helpString);
|
||||
listener.onEnrollmentHelp(helpMsgId, helpString);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -174,7 +174,7 @@ public abstract class BiometricEnrollSidecar extends InstrumentedFragment {
|
||||
|
||||
protected void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
|
||||
if (mListener != null) {
|
||||
mListener.onEnrollmentHelp(helpString);
|
||||
mListener.onEnrollmentHelp(helpMsgId, helpString);
|
||||
} else {
|
||||
mQueuedEvents.add(new QueuedEnrollmentHelp(helpMsgId, helpString));
|
||||
}
|
||||
|
||||
236
src/com/android/settings/biometrics/face/AnimationParticle.java
Normal file
236
src/com/android/settings/biometrics/face/AnimationParticle.java
Normal file
@@ -0,0 +1,236 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.biometrics.face;
|
||||
|
||||
import android.animation.ArgbEvaluator;
|
||||
import android.content.Context;
|
||||
import android.graphics.Canvas;
|
||||
import android.graphics.Paint;
|
||||
import android.graphics.Path;
|
||||
import android.graphics.Rect;
|
||||
import android.graphics.RectF;
|
||||
import android.util.Log;
|
||||
|
||||
import com.android.settings.R;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Class containing the state for an individual feedback dot / path. The dots are assigned colors
|
||||
* based on their index.
|
||||
*/
|
||||
public class AnimationParticle {
|
||||
|
||||
private static final String TAG = "AnimationParticle";
|
||||
|
||||
private static final int MIN_STROKE_WIDTH = 10;
|
||||
private static final int MAX_STROKE_WIDTH = 20; // Be careful that this doesn't get clipped
|
||||
private static final int FINAL_RING_STROKE_WIDTH = 15;
|
||||
|
||||
private static final float ROTATION_SPEED_NORMAL = 0.8f; // radians per second, 1 = ~57 degrees
|
||||
private static final float ROTATION_ACCELERATION_SPEED = 2.0f;
|
||||
private static final float PULSE_SPEED_NORMAL = 1 * 2 * (float) Math.PI; // 1 cycle per second
|
||||
private static final float RING_SWEEP_GROW_RATE_PRIMARY = 480; // degrees per second
|
||||
private static final float RING_SWEEP_GROW_RATE_SECONDARY = 240; // degrees per second
|
||||
private static final float RING_SIZE_FINALIZATION_TIME = 0.1f; // seconds
|
||||
|
||||
private final Rect mBounds; // bounds for the canvas
|
||||
private final int mBorderWidth; // amount of padding from the edges
|
||||
private final ArgbEvaluator mEvaluator;
|
||||
private final int mErrorColor;
|
||||
private final int mIndex;
|
||||
private final Listener mListener;
|
||||
|
||||
private final Paint mPaint;
|
||||
private final int mAssignedColor;
|
||||
private final float mOffsetTimeSec; // stagger particle size to make a wave effect
|
||||
|
||||
private int mLastAnimationState;
|
||||
private int mAnimationState;
|
||||
private float mCurrentSize = MIN_STROKE_WIDTH;
|
||||
private float mCurrentAngle; // 0 is to the right, in radians
|
||||
private float mRotationSpeed = ROTATION_SPEED_NORMAL; // speed of dot rotation
|
||||
private float mSweepAngle = 0; // ring sweep, degrees per second
|
||||
private float mSweepRate = RING_SWEEP_GROW_RATE_SECONDARY; // acceleration
|
||||
private float mRingAdjustRate; // rate at which ring should grow/shrink to final size
|
||||
private float mRingCompletionTime; // time at which ring should be completed
|
||||
|
||||
public interface Listener {
|
||||
void onRingCompleted(int index);
|
||||
}
|
||||
|
||||
public AnimationParticle(Context context, Listener listener, Rect bounds, int borderWidth,
|
||||
int index, int totalParticles, List<Integer> colors) {
|
||||
mBounds = bounds;
|
||||
mBorderWidth = borderWidth;
|
||||
mEvaluator = new ArgbEvaluator();
|
||||
mErrorColor = context.getResources()
|
||||
.getColor(R.color.face_anim_particle_error, context.getTheme());
|
||||
mIndex = index;
|
||||
mListener = listener;
|
||||
|
||||
mCurrentAngle = (float) index / totalParticles * 2 * (float) Math.PI;
|
||||
mOffsetTimeSec = (float) index / totalParticles
|
||||
* (1 / ROTATION_SPEED_NORMAL) * 2 * (float) Math.PI;
|
||||
|
||||
mPaint = new Paint();
|
||||
mAssignedColor = colors.get(index % colors.size());
|
||||
mPaint.setColor(mAssignedColor);
|
||||
mPaint.setAntiAlias(true);
|
||||
mPaint.setStrokeWidth(mCurrentSize);
|
||||
mPaint.setStyle(Paint.Style.FILL);
|
||||
mPaint.setStrokeCap(Paint.Cap.ROUND);
|
||||
}
|
||||
|
||||
public void updateState(int animationState) {
|
||||
if (mAnimationState == animationState) {
|
||||
Log.w(TAG, "Already in state " + animationState);
|
||||
return;
|
||||
}
|
||||
if (animationState == ParticleCollection.STATE_COMPLETE) {
|
||||
mPaint.setStyle(Paint.Style.STROKE);
|
||||
}
|
||||
mLastAnimationState = mAnimationState;
|
||||
mAnimationState = animationState;
|
||||
}
|
||||
|
||||
// There are two types of particles, secondary and primary. Primary particles accelerate faster
|
||||
// during the "completed" animation. Particles are secondary by default.
|
||||
public void setAsPrimary() {
|
||||
mSweepRate = RING_SWEEP_GROW_RATE_PRIMARY;
|
||||
}
|
||||
|
||||
public void update(long t, long dt) {
|
||||
if (mAnimationState != ParticleCollection.STATE_COMPLETE) {
|
||||
updateDot(t, dt);
|
||||
} else {
|
||||
updateRing(t, dt);
|
||||
}
|
||||
}
|
||||
|
||||
private void updateDot(long t, long dt) {
|
||||
final float dtSec = 0.001f * dt;
|
||||
final float tSec = 0.001f * t;
|
||||
|
||||
final float multiplier = mRotationSpeed / ROTATION_SPEED_NORMAL;
|
||||
|
||||
// Calculate rotation speed / angle
|
||||
if ((mAnimationState == ParticleCollection.STATE_STOPPED_COLORFUL
|
||||
|| mAnimationState == ParticleCollection.STATE_STOPPED_GRAY)
|
||||
&& mRotationSpeed > 0) {
|
||||
// Linear slow down for now
|
||||
mRotationSpeed = Math.max(mRotationSpeed - ROTATION_ACCELERATION_SPEED * dtSec, 0);
|
||||
} else if (mAnimationState == ParticleCollection.STATE_STARTED
|
||||
&& mRotationSpeed < ROTATION_SPEED_NORMAL) {
|
||||
// Linear speed up for now
|
||||
mRotationSpeed += ROTATION_ACCELERATION_SPEED * dtSec;
|
||||
}
|
||||
|
||||
mCurrentAngle += dtSec * mRotationSpeed;
|
||||
|
||||
// Calculate dot / ring size; linearly proportional with rotation speed
|
||||
mCurrentSize =
|
||||
(MAX_STROKE_WIDTH - MIN_STROKE_WIDTH) / 2
|
||||
* (float) Math.sin(tSec * PULSE_SPEED_NORMAL + mOffsetTimeSec)
|
||||
+ (MAX_STROKE_WIDTH + MIN_STROKE_WIDTH) / 2;
|
||||
mCurrentSize = (mCurrentSize - MIN_STROKE_WIDTH) * multiplier + MIN_STROKE_WIDTH;
|
||||
|
||||
// Calculate paint color; linearly proportional to rotation speed
|
||||
int color = mAssignedColor;
|
||||
if (mAnimationState == ParticleCollection.STATE_STOPPED_GRAY) {
|
||||
color = (int) mEvaluator.evaluate(1 - multiplier, mAssignedColor, mErrorColor);
|
||||
} else if (mLastAnimationState == ParticleCollection.STATE_STOPPED_GRAY) {
|
||||
color = (int) mEvaluator.evaluate(1 - multiplier, mAssignedColor, mErrorColor);
|
||||
}
|
||||
|
||||
mPaint.setColor(color);
|
||||
mPaint.setStrokeWidth(mCurrentSize);
|
||||
}
|
||||
|
||||
private void updateRing(long t, long dt) {
|
||||
final float dtSec = 0.001f * dt;
|
||||
final float tSec = 0.001f * t;
|
||||
|
||||
// Store the start time, since we need to guarantee all rings reach final size at same time
|
||||
// independent of current size. The magic 0 check is safe.
|
||||
if (mRingAdjustRate == 0) {
|
||||
mRingAdjustRate =
|
||||
(FINAL_RING_STROKE_WIDTH - mCurrentSize) / RING_SIZE_FINALIZATION_TIME;
|
||||
if (mRingCompletionTime == 0) {
|
||||
mRingCompletionTime = tSec + RING_SIZE_FINALIZATION_TIME;
|
||||
}
|
||||
}
|
||||
|
||||
// Accelerate to attack speed.. jk, back to normal speed
|
||||
if (mRotationSpeed < ROTATION_SPEED_NORMAL) {
|
||||
mRotationSpeed += ROTATION_ACCELERATION_SPEED * dtSec;
|
||||
}
|
||||
|
||||
// For arcs, this is the "start"
|
||||
mCurrentAngle += dtSec * mRotationSpeed;
|
||||
|
||||
// Update the sweep angle until it fills entire circle
|
||||
if (mSweepAngle < 360) {
|
||||
final float sweepGrowth = mSweepRate * dtSec;
|
||||
mSweepAngle = mSweepAngle + sweepGrowth;
|
||||
mSweepRate = mSweepRate + sweepGrowth;
|
||||
}
|
||||
if (mSweepAngle > 360) {
|
||||
mSweepAngle = 360;
|
||||
mListener.onRingCompleted(mIndex);
|
||||
}
|
||||
|
||||
// Animate stroke width to final size.
|
||||
if (tSec < RING_SIZE_FINALIZATION_TIME) {
|
||||
mCurrentSize = mCurrentSize + mRingAdjustRate * dtSec;
|
||||
mPaint.setStrokeWidth(mCurrentSize);
|
||||
} else {
|
||||
// There should be small to no discontinuity in this if/else
|
||||
mCurrentSize = FINAL_RING_STROKE_WIDTH;
|
||||
mPaint.setStrokeWidth(mCurrentSize);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public void draw(Canvas canvas) {
|
||||
if (mAnimationState != ParticleCollection.STATE_COMPLETE) {
|
||||
drawDot(canvas);
|
||||
} else {
|
||||
drawRing(canvas);
|
||||
}
|
||||
}
|
||||
|
||||
// Draws a dot at the current position on the circumference of the path.
|
||||
private void drawDot(Canvas canvas) {
|
||||
final float w = mBounds.right - mBounds.exactCenterX() - mBorderWidth;
|
||||
final float h = mBounds.bottom - mBounds.exactCenterY() - mBorderWidth;
|
||||
canvas.drawCircle(
|
||||
mBounds.exactCenterX() + w * (float) Math.cos(mCurrentAngle),
|
||||
mBounds.exactCenterY() + h * (float) Math.sin(mCurrentAngle),
|
||||
mCurrentSize,
|
||||
mPaint);
|
||||
}
|
||||
|
||||
private void drawRing(Canvas canvas) {
|
||||
RectF arc = new RectF(
|
||||
mBorderWidth, mBorderWidth,
|
||||
mBounds.width() - mBorderWidth, mBounds.height() - mBorderWidth);
|
||||
Path path = new Path();
|
||||
path.arcTo(arc, (float) Math.toDegrees(mCurrentAngle), mSweepAngle);
|
||||
canvas.drawPath(path, mPaint);
|
||||
}
|
||||
}
|
||||
@@ -16,6 +16,8 @@
|
||||
|
||||
package com.android.settings.biometrics.face;
|
||||
|
||||
import android.animation.TimeAnimator;
|
||||
import android.content.Context;
|
||||
import android.graphics.Canvas;
|
||||
import android.graphics.Color;
|
||||
import android.graphics.ColorFilter;
|
||||
@@ -26,16 +28,43 @@ import android.graphics.PorterDuffXfermode;
|
||||
import android.graphics.Rect;
|
||||
import android.graphics.drawable.Drawable;
|
||||
|
||||
/**
|
||||
* A drawable containing the circle cutout.
|
||||
*/
|
||||
public class FaceEnrollAnimationDrawable extends Drawable {
|
||||
import com.android.settings.biometrics.BiometricEnrollSidecar;
|
||||
|
||||
/**
|
||||
* A drawable containing the circle cutout as well as the animations.
|
||||
*/
|
||||
public class FaceEnrollAnimationDrawable extends Drawable
|
||||
implements BiometricEnrollSidecar.Listener {
|
||||
|
||||
// Tune this parameter so the UI looks nice - and so that we don't have to draw the animations
|
||||
// outside our bounds. A fraction of each rotating dot should be overlapping the camera preview.
|
||||
private static final int BORDER_BOUNDS = 20;
|
||||
|
||||
private final Context mContext;
|
||||
private final ParticleCollection.Listener mListener;
|
||||
private Rect mBounds;
|
||||
private final Paint mSquarePaint;
|
||||
private final Paint mCircleCutoutPaint;
|
||||
|
||||
public FaceEnrollAnimationDrawable() {
|
||||
private ParticleCollection mParticleCollection;
|
||||
|
||||
private TimeAnimator mTimeAnimator;
|
||||
|
||||
private final ParticleCollection.Listener mAnimationListener
|
||||
= new ParticleCollection.Listener() {
|
||||
@Override
|
||||
public void onEnrolled() {
|
||||
if (mTimeAnimator != null && mTimeAnimator.isStarted()) {
|
||||
mTimeAnimator.end();
|
||||
mListener.onEnrolled();
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
public FaceEnrollAnimationDrawable(Context context, ParticleCollection.Listener listener) {
|
||||
mContext = context;
|
||||
mListener = listener;
|
||||
|
||||
mSquarePaint = new Paint();
|
||||
mSquarePaint.setColor(Color.WHITE);
|
||||
mSquarePaint.setAntiAlias(true);
|
||||
@@ -46,9 +75,35 @@ public class FaceEnrollAnimationDrawable extends Drawable {
|
||||
mCircleCutoutPaint.setAntiAlias(true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
|
||||
mParticleCollection.onEnrollmentHelp(helpMsgId, helpString);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentError(int errMsgId, CharSequence errString) {
|
||||
mParticleCollection.onEnrollmentError(errMsgId, errString);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentProgressChange(int steps, int remaining) {
|
||||
mParticleCollection.onEnrollmentProgressChange(steps, remaining);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onBoundsChange(Rect bounds) {
|
||||
mBounds = bounds;
|
||||
mParticleCollection =
|
||||
new ParticleCollection(mContext, mAnimationListener, bounds, BORDER_BOUNDS);
|
||||
|
||||
if (mTimeAnimator == null) {
|
||||
mTimeAnimator = new TimeAnimator();
|
||||
mTimeAnimator.setTimeListener((animation, totalTimeMs, deltaTimeMs) -> {
|
||||
mParticleCollection.update(totalTimeMs, deltaTimeMs);
|
||||
FaceEnrollAnimationDrawable.this.invalidateSelf();
|
||||
});
|
||||
mTimeAnimator.start();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -63,7 +118,10 @@ public class FaceEnrollAnimationDrawable extends Drawable {
|
||||
|
||||
// Clear a circle in the middle for the camera preview
|
||||
canvas.drawCircle(mBounds.exactCenterX(), mBounds.exactCenterY(),
|
||||
mBounds.height() / 2, mCircleCutoutPaint);
|
||||
mBounds.height() / 2 - BORDER_BOUNDS, mCircleCutoutPaint);
|
||||
|
||||
// Draw the animation
|
||||
mParticleCollection.draw(canvas);
|
||||
|
||||
canvas.restore();
|
||||
}
|
||||
|
||||
@@ -45,7 +45,14 @@ public class FaceEnrollEnrolling extends BiometricsEnrollEnrolling {
|
||||
private TextView mErrorText;
|
||||
private Interpolator mLinearOutSlowInInterpolator;
|
||||
private boolean mShouldFinishOnStop = true;
|
||||
private FaceEnrollPreviewFragment mFaceCameraPreview;
|
||||
private FaceEnrollPreviewFragment mPreviewFragment;
|
||||
|
||||
private ParticleCollection.Listener mListener = new ParticleCollection.Listener() {
|
||||
@Override
|
||||
public void onEnrolled() {
|
||||
FaceEnrollEnrolling.this.launchFinish(mToken);
|
||||
}
|
||||
};
|
||||
|
||||
public static class FaceErrorDialog extends BiometricErrorDialog {
|
||||
static FaceErrorDialog newInstance(CharSequence msg, int msgId) {
|
||||
@@ -87,7 +94,7 @@ public class FaceEnrollEnrolling extends BiometricsEnrollEnrolling {
|
||||
|
||||
if (shouldLaunchConfirmLock()) {
|
||||
launchConfirmLock(R.string.security_settings_face_preference_title,
|
||||
Utils.getFaceManagerOrNull(this).preEnroll());
|
||||
Utils.getFingerprintManagerOrNull(this).preEnroll());
|
||||
mShouldFinishOnStop = false;
|
||||
} else {
|
||||
startEnrollment();
|
||||
@@ -97,13 +104,14 @@ public class FaceEnrollEnrolling extends BiometricsEnrollEnrolling {
|
||||
@Override
|
||||
public void startEnrollment() {
|
||||
super.startEnrollment();
|
||||
mFaceCameraPreview = (FaceEnrollPreviewFragment) getSupportFragmentManager()
|
||||
mPreviewFragment = (FaceEnrollPreviewFragment) getSupportFragmentManager()
|
||||
.findFragmentByTag(TAG_FACE_PREVIEW);
|
||||
if (mFaceCameraPreview == null) {
|
||||
mFaceCameraPreview = new FaceEnrollPreviewFragment();
|
||||
getSupportFragmentManager().beginTransaction().add(mFaceCameraPreview, TAG_FACE_PREVIEW)
|
||||
if (mPreviewFragment == null) {
|
||||
mPreviewFragment = new FaceEnrollPreviewFragment();
|
||||
getSupportFragmentManager().beginTransaction().add(mPreviewFragment, TAG_FACE_PREVIEW)
|
||||
.commitAllowingStateLoss();
|
||||
}
|
||||
mPreviewFragment.setListener(mListener);
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -132,10 +140,11 @@ public class FaceEnrollEnrolling extends BiometricsEnrollEnrolling {
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentHelp(CharSequence helpString) {
|
||||
public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
|
||||
if (!TextUtils.isEmpty(helpString)) {
|
||||
showError(helpString);
|
||||
}
|
||||
mPreviewFragment.onEnrollmentHelp(helpMsgId, helpString);
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -149,6 +158,7 @@ public class FaceEnrollEnrolling extends BiometricsEnrollEnrolling {
|
||||
msgId = R.string.security_settings_face_enroll_error_generic_dialog_message;
|
||||
break;
|
||||
}
|
||||
mPreviewFragment.onEnrollmentError(errMsgId, errString);
|
||||
showErrorDialog(getText(msgId), errMsgId);
|
||||
}
|
||||
|
||||
@@ -157,6 +167,8 @@ public class FaceEnrollEnrolling extends BiometricsEnrollEnrolling {
|
||||
if (DEBUG) {
|
||||
Log.v(TAG, "Steps: " + steps + " Remaining: " + remaining);
|
||||
}
|
||||
mPreviewFragment.onEnrollmentProgressChange(steps, remaining);
|
||||
|
||||
// TODO: Update the actual animation
|
||||
showError("Steps: " + steps + " Remaining: " + remaining);
|
||||
}
|
||||
|
||||
@@ -38,6 +38,7 @@ import android.widget.ImageView;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.biometrics.BiometricEnrollSidecar;
|
||||
import com.android.settings.core.InstrumentedPreferenceFragment;
|
||||
|
||||
import java.util.ArrayList;
|
||||
@@ -50,7 +51,8 @@ import java.util.List;
|
||||
* Fragment that contains the logic for showing and controlling the camera preview, circular
|
||||
* overlay, as well as the enrollment animations.
|
||||
*/
|
||||
public class FaceEnrollPreviewFragment extends InstrumentedPreferenceFragment {
|
||||
public class FaceEnrollPreviewFragment extends InstrumentedPreferenceFragment
|
||||
implements BiometricEnrollSidecar.Listener {
|
||||
|
||||
private static final String TAG = "FaceEnrollPreviewFragment";
|
||||
|
||||
@@ -65,6 +67,7 @@ public class FaceEnrollPreviewFragment extends InstrumentedPreferenceFragment {
|
||||
private CameraCaptureSession mCaptureSession;
|
||||
private CaptureRequest mPreviewRequest;
|
||||
private Size mPreviewSize;
|
||||
private ParticleCollection.Listener mListener;
|
||||
|
||||
// View used to contain the circular cutout and enrollment animation drawable
|
||||
private ImageView mCircleView;
|
||||
@@ -75,6 +78,15 @@ public class FaceEnrollPreviewFragment extends InstrumentedPreferenceFragment {
|
||||
// Texture used for showing the camera preview
|
||||
private FaceSquareTextureView mTextureView;
|
||||
|
||||
// Listener sent to the animation drawable
|
||||
private final ParticleCollection.Listener mAnimationListener
|
||||
= new ParticleCollection.Listener() {
|
||||
@Override
|
||||
public void onEnrolled() {
|
||||
mListener.onEnrolled();
|
||||
}
|
||||
};
|
||||
|
||||
private final TextureView.SurfaceTextureListener mSurfaceTextureListener =
|
||||
new TextureView.SurfaceTextureListener() {
|
||||
|
||||
@@ -185,7 +197,7 @@ public class FaceEnrollPreviewFragment extends InstrumentedPreferenceFragment {
|
||||
// Must disable hardware acceleration for this view, otherwise transparency breaks
|
||||
mCircleView.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
|
||||
|
||||
mAnimationDrawable = new FaceEnrollAnimationDrawable();
|
||||
mAnimationDrawable = new FaceEnrollAnimationDrawable(getContext(), mAnimationListener);
|
||||
mCircleView.setImageDrawable(mAnimationDrawable);
|
||||
|
||||
mCameraManager = (CameraManager) getContext().getSystemService(Context.CAMERA_SERVICE);
|
||||
@@ -212,6 +224,25 @@ public class FaceEnrollPreviewFragment extends InstrumentedPreferenceFragment {
|
||||
closeCamera();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentError(int errMsgId, CharSequence errString) {
|
||||
mAnimationDrawable.onEnrollmentError(errMsgId, errString);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
|
||||
mAnimationDrawable.onEnrollmentHelp(helpMsgId, helpString);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentProgressChange(int steps, int remaining) {
|
||||
mAnimationDrawable.onEnrollmentProgressChange(steps, remaining);
|
||||
}
|
||||
|
||||
public void setListener(ParticleCollection.Listener listener) {
|
||||
mListener = listener;
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets up member variables related to camera.
|
||||
*
|
||||
|
||||
143
src/com/android/settings/biometrics/face/ParticleCollection.java
Normal file
143
src/com/android/settings/biometrics/face/ParticleCollection.java
Normal file
@@ -0,0 +1,143 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.biometrics.face;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.res.Resources;
|
||||
import android.graphics.Canvas;
|
||||
import android.graphics.Rect;
|
||||
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.biometrics.BiometricEnrollSidecar;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Class that's used to create, maintain, and update the state of each animation particle. Particles
|
||||
* should have their colors assigned based on their index. Particles are split into primary and
|
||||
* secondary types - primary types animate twice as fast during the completion effect. The particles
|
||||
* are updated/drawn in a special order so that the overlap is correct during the final completion
|
||||
* effect.
|
||||
*/
|
||||
public class ParticleCollection implements BiometricEnrollSidecar.Listener {
|
||||
|
||||
private static final String TAG = "AnimationController";
|
||||
|
||||
private static final int NUM_PARTICLES = 12;
|
||||
|
||||
public static final int STATE_STARTED = 1; // dots are rotating
|
||||
public static final int STATE_STOPPED_COLORFUL = 2; // dots are not rotating but colorful
|
||||
public static final int STATE_STOPPED_GRAY = 3; // dots are not rotating and also gray (error)
|
||||
public static final int STATE_COMPLETE = 4; // face is enrolled
|
||||
|
||||
private final List<AnimationParticle> mParticleList;
|
||||
private final List<Integer> mPrimariesInProgress; // primary particles not done animating yet
|
||||
private int mState;
|
||||
private Listener mListener;
|
||||
|
||||
public interface Listener {
|
||||
void onEnrolled();
|
||||
}
|
||||
|
||||
private final AnimationParticle.Listener mParticleListener = new AnimationParticle.Listener() {
|
||||
@Override
|
||||
public void onRingCompleted(int index) {
|
||||
final boolean wasEmpty = mPrimariesInProgress.isEmpty();
|
||||
// We can stop the time animator once the three primary particles have finished
|
||||
for (int i = 0; i < mPrimariesInProgress.size(); i++) {
|
||||
if (mPrimariesInProgress.get(i).intValue() == index) {
|
||||
mPrimariesInProgress.remove(i);
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (mPrimariesInProgress.isEmpty() && !wasEmpty) {
|
||||
mListener.onEnrolled();
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
public ParticleCollection(Context context, Listener listener, Rect bounds, int borderWidth) {
|
||||
mParticleList = new ArrayList<>();
|
||||
mListener = listener;
|
||||
|
||||
final List<Integer> colors = new ArrayList<>();
|
||||
final Resources.Theme theme = context.getTheme();
|
||||
final Resources resources = context.getResources();
|
||||
colors.add(resources.getColor(R.color.face_anim_particle_color_1, theme));
|
||||
colors.add(resources.getColor(R.color.face_anim_particle_color_2, theme));
|
||||
colors.add(resources.getColor(R.color.face_anim_particle_color_3, theme));
|
||||
colors.add(resources.getColor(R.color.face_anim_particle_color_4, theme));
|
||||
|
||||
// Primary particles expand faster during the completion animation
|
||||
mPrimariesInProgress = new ArrayList<>(Arrays.asList(0, 4, 8));
|
||||
|
||||
// Order in which to draw the particles. This is so the final "completion" animation has
|
||||
// the correct behavior.
|
||||
final int[] order = {3, 7, 11, 2, 6, 10, 1, 5, 9, 0, 4, 8};
|
||||
|
||||
for (int i = 0; i < NUM_PARTICLES; i++) {
|
||||
AnimationParticle particle = new AnimationParticle(context, mParticleListener, bounds,
|
||||
borderWidth, order[i], NUM_PARTICLES, colors);
|
||||
if (mPrimariesInProgress.contains(order[i])) {
|
||||
particle.setAsPrimary();
|
||||
}
|
||||
mParticleList.add(particle);
|
||||
}
|
||||
|
||||
updateState(STATE_STARTED);
|
||||
}
|
||||
|
||||
public void update(long t, long dt) {
|
||||
for (int i = 0; i < mParticleList.size(); i++) {
|
||||
mParticleList.get(i).update(t, dt);
|
||||
}
|
||||
}
|
||||
|
||||
public void draw(Canvas canvas) {
|
||||
for (int i = 0; i < mParticleList.size(); i++) {
|
||||
mParticleList.get(i).draw(canvas);
|
||||
}
|
||||
}
|
||||
|
||||
private void updateState(int state) {
|
||||
if (mState != state) {
|
||||
for (int i = 0; i < mParticleList.size(); i++) {
|
||||
mParticleList.get(i).updateState(state);
|
||||
}
|
||||
mState = state;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentError(int errMsgId, CharSequence errString) {
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentProgressChange(int steps, int remaining) {
|
||||
if (remaining == 0) {
|
||||
updateState(STATE_COMPLETE);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -245,7 +245,7 @@ public class FingerprintEnrollEnrolling extends BiometricsEnrollEnrolling {
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentHelp(CharSequence helpString) {
|
||||
public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
|
||||
if (!TextUtils.isEmpty(helpString)) {
|
||||
mErrorText.removeCallbacks(mTouchAgainRunnable);
|
||||
showError(helpString);
|
||||
|
||||
@@ -94,7 +94,7 @@ public class FingerprintEnrollFindSensor extends BiometricEnrollBase {
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentHelp(CharSequence helpString) {
|
||||
public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
|
||||
}
|
||||
|
||||
@Override
|
||||
|
||||
@@ -123,7 +123,6 @@ public class DashboardSummary extends InstrumentedFragment
|
||||
|
||||
mSummaryLoader = new SummaryLoader(activity, CategoryKey.CATEGORY_HOMEPAGE);
|
||||
|
||||
mConditionManager = ConditionManager.get(activity, false);
|
||||
if (com.android.settings.homepage.conditional.v2.ConditionManager.isEnabled(activity)) {
|
||||
mConditionManager = null;
|
||||
mConditionManager2 =
|
||||
|
||||
@@ -28,6 +28,7 @@ import com.android.settings.fuelgauge.batterytip.BatteryTipDialogFragment;
|
||||
import com.android.settings.fuelgauge.batterytip.tips.BatteryTip;
|
||||
import com.android.settings.fuelgauge.batterytip.tips.RestrictAppTip;
|
||||
import com.android.settings.fuelgauge.batterytip.tips.UnrestrictAppTip;
|
||||
import com.android.settingslib.RestrictedPreference;
|
||||
import com.android.settingslib.core.AbstractPreferenceController;
|
||||
import com.android.settingslib.fuelgauge.PowerWhitelistBackend;
|
||||
|
||||
@@ -76,6 +77,11 @@ public class BackgroundActivityPreferenceController extends AbstractPreferenceCo
|
||||
|
||||
@Override
|
||||
public void updateState(Preference preference) {
|
||||
final RestrictedPreference restrictedPreference = (RestrictedPreference) preference;
|
||||
if (restrictedPreference.isDisabledByAdmin()) {
|
||||
// If disabled, let RestrictedPreference handle it and do nothing here
|
||||
return;
|
||||
}
|
||||
final int mode = mAppOpsManager
|
||||
.checkOpNoThrow(AppOpsManager.OP_RUN_ANY_IN_BACKGROUND, mUid, mTargetPackage);
|
||||
final boolean whitelisted = mPowerWhitelistBackend.isWhitelisted(mTargetPackage);
|
||||
|
||||
@@ -25,6 +25,7 @@ import android.provider.Settings;
|
||||
|
||||
import com.android.settings.R;
|
||||
|
||||
@Deprecated
|
||||
public abstract class AbnormalRingerConditionBase extends Condition {
|
||||
|
||||
private final IntentFilter mFilter;
|
||||
@@ -45,7 +46,7 @@ public abstract class AbnormalRingerConditionBase extends Condition {
|
||||
|
||||
@Override
|
||||
public CharSequence[] getActions() {
|
||||
return new CharSequence[] {
|
||||
return new CharSequence[]{
|
||||
mManager.getContext().getText(R.string.condition_device_muted_action_turn_on_sound)
|
||||
};
|
||||
}
|
||||
|
||||
@@ -28,6 +28,10 @@ import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
|
||||
import com.android.settings.R;
|
||||
import com.android.settingslib.WirelessUtils;
|
||||
|
||||
/**
|
||||
* @deprecated in favor of {@link AirplaneModeConditionCard}.
|
||||
*/
|
||||
@Deprecated
|
||||
public class AirplaneModeCondition extends Condition {
|
||||
public static String TAG = "APM_Condition";
|
||||
|
||||
|
||||
@@ -23,6 +23,7 @@ import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.Settings;
|
||||
|
||||
@Deprecated
|
||||
public class BackgroundDataCondition extends Condition {
|
||||
|
||||
public BackgroundDataCondition(ConditionManager manager) {
|
||||
|
||||
@@ -26,6 +26,10 @@ import com.android.settings.fuelgauge.BatterySaverReceiver;
|
||||
import com.android.settings.fuelgauge.batterysaver.BatterySaverSettings;
|
||||
import com.android.settingslib.fuelgauge.BatterySaverUtils;
|
||||
|
||||
/**
|
||||
* @deprecated in favor of {@link BatterySaverConditionCard}
|
||||
*/
|
||||
@Deprecated
|
||||
public class BatterySaverCondition extends Condition implements
|
||||
BatterySaverReceiver.BatterySaverListener {
|
||||
|
||||
|
||||
@@ -23,6 +23,7 @@ import com.android.internal.telephony.TelephonyIntents;
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.Settings;
|
||||
|
||||
@Deprecated
|
||||
public class CellularDataCondition extends Condition {
|
||||
|
||||
private final Receiver mReceiver;
|
||||
|
||||
@@ -33,6 +33,10 @@ import com.android.settings.R;
|
||||
import com.android.settings.core.SubSettingLauncher;
|
||||
import com.android.settings.notification.ZenModeSettings;
|
||||
|
||||
/**
|
||||
* @deprecated in favor of {@link com.android.settings.homepage.conditional.v2.DndConditionCard}
|
||||
*/
|
||||
@Deprecated
|
||||
public class DndCondition extends Condition {
|
||||
|
||||
private static final String TAG = "DndCondition";
|
||||
|
||||
@@ -33,6 +33,7 @@ import com.android.settings.core.SubSettingLauncher;
|
||||
import com.android.settingslib.RestrictedLockUtils;
|
||||
import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin;
|
||||
|
||||
@Deprecated
|
||||
public class HotspotCondition extends Condition {
|
||||
|
||||
private final WifiManager mWifiManager;
|
||||
@@ -95,7 +96,7 @@ public class HotspotCondition extends Condition {
|
||||
UserManager.DISALLOW_CONFIG_TETHERING, UserHandle.myUserId())) {
|
||||
return new CharSequence[0];
|
||||
}
|
||||
return new CharSequence[] {context.getString(R.string.condition_turn_off)};
|
||||
return new CharSequence[]{context.getString(R.string.condition_turn_off)};
|
||||
}
|
||||
|
||||
@Override
|
||||
|
||||
@@ -25,6 +25,10 @@ import com.android.settings.R;
|
||||
import com.android.settings.core.SubSettingLauncher;
|
||||
import com.android.settings.display.NightDisplaySettings;
|
||||
|
||||
/**
|
||||
* @deprecated in favor of {@link NightDisplayConditionController}
|
||||
*/
|
||||
@Deprecated
|
||||
public final class NightDisplayCondition extends Condition
|
||||
implements ColorDisplayController.Callback {
|
||||
|
||||
|
||||
@@ -26,6 +26,7 @@ import android.provider.Settings;
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
@Deprecated
|
||||
public class RingerMutedCondition extends AbnormalRingerConditionBase {
|
||||
|
||||
private final NotificationManager mNotificationManager;
|
||||
|
||||
@@ -22,6 +22,7 @@ import android.media.AudioManager;
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
@Deprecated
|
||||
public class RingerVibrateCondition extends AbnormalRingerConditionBase {
|
||||
|
||||
RingerVibrateCondition(ConditionManager manager) {
|
||||
|
||||
@@ -29,6 +29,7 @@ import com.android.settings.Settings;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
@Deprecated
|
||||
public class WorkModeCondition extends Condition {
|
||||
|
||||
private UserManager mUm;
|
||||
|
||||
@@ -0,0 +1,76 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.IntentFilter;
|
||||
import android.media.AudioManager;
|
||||
import android.provider.Settings;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public abstract class AbnormalRingerConditionController implements ConditionalCardController {
|
||||
|
||||
private static final IntentFilter FILTER =
|
||||
new IntentFilter(AudioManager.INTERNAL_RINGER_MODE_CHANGED_ACTION);
|
||||
|
||||
protected final AudioManager mAudioManager;
|
||||
private final Context mAppContext;
|
||||
private final ConditionManager mConditionManager;
|
||||
private final RingerModeChangeReceiver mReceiver;
|
||||
|
||||
public AbnormalRingerConditionController(Context appContext, ConditionManager manager) {
|
||||
mAppContext = appContext;
|
||||
mConditionManager = manager;
|
||||
mAudioManager = (AudioManager) appContext.getSystemService(Context.AUDIO_SERVICE);
|
||||
mReceiver = new RingerModeChangeReceiver();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPrimaryClick(Context context) {
|
||||
context.startActivity(new Intent(Settings.ACTION_SOUND_SETTINGS));
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActionClick() {
|
||||
mAudioManager.setRingerModeInternal(AudioManager.RINGER_MODE_NORMAL);
|
||||
mAudioManager.setStreamVolume(AudioManager.STREAM_RING, 1, 0 /* flags */);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void startMonitoringStateChange() {
|
||||
mAppContext.registerReceiver(mReceiver, FILTER);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void stopMonitoringStateChange() {
|
||||
mAppContext.unregisterReceiver(mReceiver);
|
||||
}
|
||||
|
||||
class RingerModeChangeReceiver extends BroadcastReceiver {
|
||||
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
final String action = intent.getAction();
|
||||
if (AudioManager.INTERNAL_RINGER_MODE_CHANGED_ACTION.equals(action)) {
|
||||
mConditionManager.onConditionChanged();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,62 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
public class AirplaneModeConditionCard implements ConditionalCard {
|
||||
|
||||
private final Context mAppContext;
|
||||
|
||||
public AirplaneModeConditionCard(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return AirplaneModeConditionController.ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getActionText() {
|
||||
return mAppContext.getText(R.string.condition_turn_off);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
return MetricsProto.MetricsEvent.SETTINGS_CONDITION_AIRPLANE_MODE;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return mAppContext.getDrawable(R.drawable.ic_airplane);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return mAppContext.getText(R.string.condition_airplane_title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
return mAppContext.getText(R.string.condition_airplane_summary);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,86 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.IntentFilter;
|
||||
import android.net.ConnectivityManager;
|
||||
import android.provider.Settings;
|
||||
|
||||
import com.android.settingslib.WirelessUtils;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public class AirplaneModeConditionController implements ConditionalCardController {
|
||||
|
||||
static final int ID = Objects.hash("AirplaneModeConditionController");
|
||||
|
||||
private static final IntentFilter AIRPLANE_MODE_FILTER =
|
||||
new IntentFilter(Intent.ACTION_AIRPLANE_MODE_CHANGED);
|
||||
|
||||
private final ConditionManager mConditionManager;
|
||||
private final Context mAppContext;
|
||||
private final Receiver mReceiver;
|
||||
|
||||
public AirplaneModeConditionController(Context appContext, ConditionManager conditionManager) {
|
||||
mAppContext = appContext;
|
||||
mConditionManager = conditionManager;
|
||||
mReceiver = new Receiver();
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isDisplayable() {
|
||||
return WirelessUtils.isAirplaneModeOn(mAppContext);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPrimaryClick(Context context) {
|
||||
context.startActivity(
|
||||
new Intent(Settings.ACTION_WIRELESS_SETTINGS));
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActionClick() {
|
||||
ConnectivityManager.from(mAppContext).setAirplaneMode(false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void startMonitoringStateChange() {
|
||||
mAppContext.registerReceiver(mReceiver, AIRPLANE_MODE_FILTER);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void stopMonitoringStateChange() {
|
||||
mAppContext.unregisterReceiver(mReceiver);
|
||||
}
|
||||
|
||||
public class Receiver extends BroadcastReceiver {
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
if (Intent.ACTION_AIRPLANE_MODE_CHANGED.equals(intent.getAction())) {
|
||||
mConditionManager.onConditionChanged();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,62 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
public class BackgroundDataConditionCard implements ConditionalCard {
|
||||
|
||||
private final Context mAppContext;
|
||||
|
||||
public BackgroundDataConditionCard(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return BackgroundDataConditionController.ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getActionText() {
|
||||
return mAppContext.getText(R.string.condition_turn_off);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
return MetricsProto.MetricsEvent.SETTINGS_CONDITION_BACKGROUND_DATA;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return mAppContext.getDrawable(R.drawable.ic_data_saver);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return mAppContext.getText(R.string.condition_bg_data_title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
return mAppContext.getText(R.string.condition_bg_data_summary);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,65 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.net.NetworkPolicyManager;
|
||||
|
||||
import com.android.settings.Settings;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public class BackgroundDataConditionController implements ConditionalCardController {
|
||||
static final int ID = Objects.hash("BackgroundDataConditionController");
|
||||
|
||||
private final Context mAppContext;
|
||||
|
||||
public BackgroundDataConditionController(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isDisplayable() {
|
||||
return NetworkPolicyManager.from(mAppContext).getRestrictBackground();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPrimaryClick(Context context) {
|
||||
context.startActivity(new Intent(context, Settings.DataUsageSummaryActivity.class));
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActionClick() {
|
||||
NetworkPolicyManager.from(mAppContext).setRestrictBackground(false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void startMonitoringStateChange() {
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public void stopMonitoringStateChange() {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,62 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
public class BatterySaverConditionCard implements ConditionalCard {
|
||||
|
||||
private final Context mAppContext;
|
||||
|
||||
public BatterySaverConditionCard(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return BatterySaverConditionController.ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getActionText() {
|
||||
return mAppContext.getText(R.string.condition_turn_off);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
return MetricsProto.MetricsEvent.SETTINGS_CONDITION_BATTERY_SAVER;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return mAppContext.getDrawable(R.drawable.ic_battery_saver_accent_24dp);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return mAppContext.getText(R.string.condition_battery_title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
return mAppContext.getText(R.string.condition_battery_summary);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,92 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.os.PowerManager;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.core.SubSettingLauncher;
|
||||
import com.android.settings.fuelgauge.BatterySaverReceiver;
|
||||
import com.android.settings.fuelgauge.batterysaver.BatterySaverSettings;
|
||||
import com.android.settingslib.fuelgauge.BatterySaverUtils;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public class BatterySaverConditionController implements ConditionalCardController,
|
||||
BatterySaverReceiver.BatterySaverListener {
|
||||
static final int ID = Objects.hash("BatterySaverConditionController");
|
||||
|
||||
private final Context mAppContext;
|
||||
private final ConditionManager mConditionManager;
|
||||
private final BatterySaverReceiver mReceiver;
|
||||
private final PowerManager mPowerManager;
|
||||
|
||||
public BatterySaverConditionController(Context appContext, ConditionManager conditionManager) {
|
||||
mAppContext = appContext;
|
||||
mConditionManager = conditionManager;
|
||||
mPowerManager = appContext.getSystemService(PowerManager.class);
|
||||
mReceiver = new BatterySaverReceiver(appContext);
|
||||
mReceiver.setBatterySaverListener(this);
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isDisplayable() {
|
||||
return mPowerManager.isPowerSaveMode();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPrimaryClick(Context context) {
|
||||
new SubSettingLauncher(context)
|
||||
.setDestination(BatterySaverSettings.class.getName())
|
||||
.setSourceMetricsCategory(MetricsProto.MetricsEvent.DASHBOARD_SUMMARY)
|
||||
.setTitleRes(R.string.battery_saver)
|
||||
.launch();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActionClick() {
|
||||
BatterySaverUtils.setPowerSaveMode(mAppContext, false,
|
||||
/*needFirstTimeWarning*/ false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void startMonitoringStateChange() {
|
||||
mReceiver.setListening(true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void stopMonitoringStateChange() {
|
||||
mReceiver.setListening(false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPowerSaveModeChanged() {
|
||||
mConditionManager.onConditionChanged();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBatteryChanged(boolean pluggedIn) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,62 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
public class CellularDataConditionCard implements ConditionalCard {
|
||||
|
||||
private final Context mAppContext;
|
||||
|
||||
public CellularDataConditionCard(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return CellularDataConditionController.ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getActionText() {
|
||||
return mAppContext.getText(R.string.condition_turn_on);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
return MetricsProto.MetricsEvent.SETTINGS_CONDITION_CELLULAR_DATA;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return mAppContext.getDrawable(R.drawable.ic_cellular_off);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return mAppContext.getText(R.string.condition_cellular_title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
return mAppContext.getText(R.string.condition_cellular_summary);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,97 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.IntentFilter;
|
||||
import android.net.ConnectivityManager;
|
||||
import android.telephony.TelephonyManager;
|
||||
|
||||
import com.android.internal.telephony.TelephonyIntents;
|
||||
import com.android.settings.Settings;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public class CellularDataConditionController implements ConditionalCardController {
|
||||
|
||||
static final int ID = Objects.hash("CellularDataConditionController");
|
||||
|
||||
private static final IntentFilter DATA_CONNECTION_FILTER =
|
||||
new IntentFilter(TelephonyIntents.ACTION_ANY_DATA_CONNECTION_STATE_CHANGED);
|
||||
|
||||
private final Context mAppContext;
|
||||
private final ConditionManager mConditionManager;
|
||||
private final Receiver mReceiver;
|
||||
private final TelephonyManager mTelephonyManager;
|
||||
private final ConnectivityManager mConnectivityManager;
|
||||
|
||||
public CellularDataConditionController(Context appContext, ConditionManager conditionManager) {
|
||||
mAppContext = appContext;
|
||||
mConditionManager = conditionManager;
|
||||
mReceiver = new Receiver();
|
||||
mConnectivityManager = appContext.getSystemService(
|
||||
ConnectivityManager.class);
|
||||
mTelephonyManager = appContext.getSystemService(TelephonyManager.class);
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isDisplayable() {
|
||||
if (!mConnectivityManager.isNetworkSupported(ConnectivityManager.TYPE_MOBILE)
|
||||
|| mTelephonyManager.getSimState() != TelephonyManager.SIM_STATE_READY) {
|
||||
return false;
|
||||
}
|
||||
return !mTelephonyManager.isDataEnabled();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPrimaryClick(Context context) {
|
||||
context.startActivity(new Intent(context,
|
||||
Settings.DataUsageSummaryActivity.class));
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActionClick() {
|
||||
mTelephonyManager.setDataEnabled(true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void startMonitoringStateChange() {
|
||||
mAppContext.registerReceiver(mReceiver, DATA_CONNECTION_FILTER);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void stopMonitoringStateChange() {
|
||||
mAppContext.unregisterReceiver(mReceiver);
|
||||
}
|
||||
|
||||
public class Receiver extends BroadcastReceiver {
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
if (TelephonyIntents.ACTION_ANY_DATA_CONNECTION_STATE_CHANGED.equals(
|
||||
intent.getAction())) {
|
||||
mConditionManager.onConditionChanged();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -141,9 +141,28 @@ public class ConditionManager {
|
||||
|
||||
private void initCandidates() {
|
||||
// Initialize controllers first.
|
||||
mCardControllers.add(new AirplaneModeConditionController(mAppContext, this /* manager */));
|
||||
mCardControllers.add(new BackgroundDataConditionController(mAppContext));
|
||||
mCardControllers.add(new BatterySaverConditionController(mAppContext, this /* manager */));
|
||||
mCardControllers.add(new CellularDataConditionController(mAppContext, this /* manager */));
|
||||
mCardControllers.add(new DndConditionCardController(mAppContext, this /* manager */));
|
||||
mCardControllers.add(new HotspotConditionController(mAppContext, this /* manager */));
|
||||
mCardControllers.add(new NightDisplayConditionController(mAppContext));
|
||||
mCardControllers.add(new RingerVibrateConditionController(mAppContext, this /* manager */));
|
||||
mCardControllers.add(new RingerMutedConditionController(mAppContext, this /* manager */));
|
||||
mCardControllers.add(new WorkModeConditionController(mAppContext));
|
||||
|
||||
// Initialize ui model later. UI model depends on controller.
|
||||
mCandidates.add(new AirplaneModeConditionCard(mAppContext));
|
||||
mCandidates.add(new BackgroundDataConditionCard(mAppContext));
|
||||
mCandidates.add(new BatterySaverConditionCard(mAppContext));
|
||||
mCandidates.add(new CellularDataConditionCard(mAppContext));
|
||||
mCandidates.add(new DndConditionCard(mAppContext, this /* manager */));
|
||||
mCandidates.add(new HotspotConditionCard(mAppContext, this /* manager */));
|
||||
mCandidates.add(new NightDisplayConditionCard(mAppContext));
|
||||
mCandidates.add(new RingerMutedConditionCard(mAppContext));
|
||||
mCandidates.add(new RingerVibrateConditionCard(mAppContext));
|
||||
mCandidates.add(new WorkModeConditionCard(mAppContext));
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
@@ -78,7 +78,7 @@ public class DndConditionCardController implements ConditionalCardController {
|
||||
public void onPrimaryClick(Context context) {
|
||||
new SubSettingLauncher(context)
|
||||
.setDestination(ZenModeSettings.class.getName())
|
||||
.setSourceMetricsCategory(MetricsProto.MetricsEvent.DASHBOARD_SUMMARY)
|
||||
.setSourceMetricsCategory(MetricsProto.MetricsEvent.SETTINGS_HOMEPAGE)
|
||||
.setTitleRes(R.string.zen_mode_settings_title)
|
||||
.launch();
|
||||
}
|
||||
|
||||
@@ -0,0 +1,72 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
import android.os.UserHandle;
|
||||
import android.os.UserManager;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
import com.android.settingslib.RestrictedLockUtils;
|
||||
|
||||
public class HotspotConditionCard implements ConditionalCard {
|
||||
|
||||
private final Context mAppContext;
|
||||
private final ConditionManager mConditionManager;
|
||||
|
||||
public HotspotConditionCard(Context appContext, ConditionManager manager) {
|
||||
mAppContext = appContext;
|
||||
mConditionManager = manager;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return HotspotConditionController.ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getActionText() {
|
||||
if (RestrictedLockUtils.hasBaseUserRestriction(mAppContext,
|
||||
UserManager.DISALLOW_CONFIG_TETHERING, UserHandle.myUserId())) {
|
||||
return null;
|
||||
}
|
||||
return mAppContext.getText(R.string.condition_turn_off);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
return MetricsProto.MetricsEvent.SETTINGS_CONDITION_HOTSPOT;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return mAppContext.getDrawable(R.drawable.ic_hotspot);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return mAppContext.getText(R.string.condition_hotspot_title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
final HotspotConditionController controller = mConditionManager.getController(getId());
|
||||
return controller.getSummary();
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,121 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.BroadcastReceiver;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.IntentFilter;
|
||||
import android.net.ConnectivityManager;
|
||||
import android.net.wifi.WifiConfiguration;
|
||||
import android.net.wifi.WifiManager;
|
||||
import android.os.UserHandle;
|
||||
import android.os.UserManager;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.TetherSettings;
|
||||
import com.android.settings.core.SubSettingLauncher;
|
||||
import com.android.settingslib.RestrictedLockUtils;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public class HotspotConditionController implements ConditionalCardController {
|
||||
static final int ID = Objects.hash("HotspotConditionController");
|
||||
|
||||
private static final IntentFilter WIFI_AP_STATE_FILTER =
|
||||
new IntentFilter(WifiManager.WIFI_AP_STATE_CHANGED_ACTION);
|
||||
|
||||
private final Context mAppContext;
|
||||
private final ConditionManager mConditionManager;
|
||||
private final WifiManager mWifiManager;
|
||||
private final Receiver mReceiver;
|
||||
|
||||
|
||||
public HotspotConditionController(Context appContext, ConditionManager conditionManager) {
|
||||
mAppContext = appContext;
|
||||
mConditionManager = conditionManager;
|
||||
mWifiManager = appContext.getSystemService(WifiManager.class);
|
||||
mReceiver = new Receiver();
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isDisplayable() {
|
||||
return mWifiManager.isWifiApEnabled();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPrimaryClick(Context context) {
|
||||
new SubSettingLauncher(context)
|
||||
.setDestination(TetherSettings.class.getName())
|
||||
.setSourceMetricsCategory(MetricsProto.MetricsEvent.DASHBOARD_SUMMARY)
|
||||
.setTitleRes(R.string.tether_settings_title_all)
|
||||
.launch();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActionClick() {
|
||||
final RestrictedLockUtils.EnforcedAdmin admin =
|
||||
RestrictedLockUtils.checkIfRestrictionEnforced(
|
||||
mAppContext, UserManager.DISALLOW_CONFIG_TETHERING, UserHandle.myUserId());
|
||||
if (admin != null) {
|
||||
RestrictedLockUtils.sendShowAdminSupportDetailsIntent(mAppContext, admin);
|
||||
} else {
|
||||
ConnectivityManager cm = (ConnectivityManager) mAppContext.getSystemService(
|
||||
Context.CONNECTIVITY_SERVICE);
|
||||
cm.stopTethering(ConnectivityManager.TETHERING_WIFI);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void startMonitoringStateChange() {
|
||||
mAppContext.registerReceiver(mReceiver, WIFI_AP_STATE_FILTER);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void stopMonitoringStateChange() {
|
||||
mAppContext.unregisterReceiver(mReceiver);
|
||||
}
|
||||
|
||||
public CharSequence getSummary() {
|
||||
return mAppContext.getString(R.string.condition_hotspot_summary, getSsid());
|
||||
}
|
||||
|
||||
private CharSequence getSsid() {
|
||||
WifiConfiguration wifiConfig = mWifiManager.getWifiApConfiguration();
|
||||
if (wifiConfig == null) {
|
||||
return mAppContext.getText(
|
||||
com.android.internal.R.string.wifi_tether_configure_ssid_default);
|
||||
} else {
|
||||
return wifiConfig.SSID;
|
||||
}
|
||||
}
|
||||
|
||||
public class Receiver extends BroadcastReceiver {
|
||||
@Override
|
||||
public void onReceive(Context context, Intent intent) {
|
||||
if (WifiManager.WIFI_AP_STATE_CHANGED_ACTION.equals(intent.getAction())) {
|
||||
mConditionManager.onConditionChanged();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,62 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
public class NightDisplayConditionCard implements ConditionalCard {
|
||||
|
||||
private final Context mAppContext;
|
||||
|
||||
public NightDisplayConditionCard(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return NightDisplayConditionController.ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getActionText() {
|
||||
return mAppContext.getText(R.string.condition_turn_off);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
return MetricsProto.MetricsEvent.SETTINGS_CONDITION_NIGHT_DISPLAY;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return mAppContext.getDrawable(R.drawable.ic_settings_night_display);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return mAppContext.getText(R.string.condition_night_display_title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
return mAppContext.getText(R.string.condition_night_display_summary);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,72 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
|
||||
import com.android.internal.app.ColorDisplayController;
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.core.SubSettingLauncher;
|
||||
import com.android.settings.display.NightDisplaySettings;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public class NightDisplayConditionController implements ConditionalCardController,
|
||||
ColorDisplayController.Callback {
|
||||
static final int ID = Objects.hash("NightDisplayConditionController");
|
||||
|
||||
private final ColorDisplayController mController;
|
||||
|
||||
public NightDisplayConditionController(Context appContext) {
|
||||
mController = new ColorDisplayController(appContext);
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isDisplayable() {
|
||||
return mController.isActivated();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPrimaryClick(Context context) {
|
||||
new SubSettingLauncher(context)
|
||||
.setDestination(NightDisplaySettings.class.getName())
|
||||
.setSourceMetricsCategory(MetricsProto.MetricsEvent.SETTINGS_HOMEPAGE)
|
||||
.setTitleRes(R.string.night_display_title)
|
||||
.launch();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActionClick() {
|
||||
mController.setActivated(false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void startMonitoringStateChange() {
|
||||
mController.setListener(this);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void stopMonitoringStateChange() {
|
||||
mController.setListener(null);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,62 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
public class RingerMutedConditionCard implements ConditionalCard {
|
||||
|
||||
private final Context mAppContext;
|
||||
|
||||
public RingerMutedConditionCard(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return RingerMutedConditionController.ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getActionText() {
|
||||
return mAppContext.getText(R.string.condition_device_muted_action_turn_on_sound);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
return MetricsProto.MetricsEvent.SETTINGS_CONDITION_DEVICE_MUTED;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return mAppContext.getDrawable(R.drawable.ic_notifications_off_24dp);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return mAppContext.getText(R.string.condition_device_muted_title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
return mAppContext.getText(R.string.condition_device_muted_summary);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,55 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import static android.content.Context.NOTIFICATION_SERVICE;
|
||||
|
||||
import android.app.NotificationManager;
|
||||
import android.content.Context;
|
||||
import android.media.AudioManager;
|
||||
import android.provider.Settings;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public class RingerMutedConditionController extends AbnormalRingerConditionController {
|
||||
static final int ID = Objects.hash("RingerMutedConditionController");
|
||||
|
||||
private final NotificationManager mNotificationManager;
|
||||
|
||||
public RingerMutedConditionController(Context appContext, ConditionManager conditionManager) {
|
||||
super(appContext, conditionManager);
|
||||
mNotificationManager =
|
||||
(NotificationManager) appContext.getSystemService(NOTIFICATION_SERVICE);
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isDisplayable() {
|
||||
int zen = Settings.Global.ZEN_MODE_OFF;
|
||||
if (mNotificationManager != null) {
|
||||
zen = mNotificationManager.getZenMode();
|
||||
}
|
||||
final boolean zenModeEnabled = zen != Settings.Global.ZEN_MODE_OFF;
|
||||
final boolean isSilent =
|
||||
mAudioManager.getRingerModeInternal() == AudioManager.RINGER_MODE_SILENT;
|
||||
return isSilent && !zenModeEnabled;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,62 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
public class RingerVibrateConditionCard implements ConditionalCard {
|
||||
|
||||
private final Context mAppContext;
|
||||
|
||||
public RingerVibrateConditionCard(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return RingerVibrateConditionController.ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getActionText() {
|
||||
return mAppContext.getText(R.string.condition_device_muted_action_turn_on_sound);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
return MetricsProto.MetricsEvent.SETTINGS_CONDITION_DEVICE_VIBRATE;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return mAppContext.getDrawable(R.drawable.ic_volume_ringer_vibrate);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return mAppContext.getText(R.string.condition_device_vibrate_title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
return mAppContext.getText(R.string.condition_device_vibrate_summary);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,41 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.media.AudioManager;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public class RingerVibrateConditionController extends AbnormalRingerConditionController {
|
||||
static final int ID = Objects.hash("RingerVibrateConditionController");
|
||||
|
||||
public RingerVibrateConditionController(Context appContext, ConditionManager conditionManager) {
|
||||
super(appContext, conditionManager);
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isDisplayable() {
|
||||
return mAudioManager.getRingerModeInternal() == AudioManager.RINGER_MODE_VIBRATE;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,63 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.drawable.Drawable;
|
||||
|
||||
import com.android.internal.logging.nano.MetricsProto;
|
||||
import com.android.settings.R;
|
||||
|
||||
public class WorkModeConditionCard implements ConditionalCard {
|
||||
|
||||
private final Context mAppContext;
|
||||
|
||||
public WorkModeConditionCard(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return WorkModeConditionController.ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getActionText() {
|
||||
return mAppContext.getText(R.string.condition_turn_on);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
return MetricsProto.MetricsEvent.SETTINGS_CONDITION_WORK_MODE;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return mAppContext.getDrawable(R.drawable.ic_signal_workmode_enable);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return mAppContext.getText(R.string.condition_work_title);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
return mAppContext.getText(R.string.condition_work_summary);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,90 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.pm.UserInfo;
|
||||
import android.os.UserHandle;
|
||||
import android.os.UserManager;
|
||||
|
||||
import com.android.settings.Settings;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.Objects;
|
||||
|
||||
public class WorkModeConditionController implements ConditionalCardController {
|
||||
|
||||
static final int ID = Objects.hash("WorkModeConditionController");
|
||||
|
||||
private final Context mAppContext;
|
||||
private final UserManager mUm;
|
||||
private UserHandle mUserHandle;
|
||||
|
||||
public WorkModeConditionController(Context appContext) {
|
||||
mAppContext = appContext;
|
||||
mUm = mAppContext.getSystemService(UserManager.class);
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getId() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isDisplayable() {
|
||||
updateUserHandle();
|
||||
return mUserHandle != null && mUm.isQuietModeEnabled(mUserHandle);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPrimaryClick(Context context) {
|
||||
context.startActivity(new Intent(context,
|
||||
Settings.AccountDashboardActivity.class));
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActionClick() {
|
||||
if (mUserHandle != null) {
|
||||
mUm.requestQuietModeEnabled(false, mUserHandle);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void startMonitoringStateChange() {
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
public void stopMonitoringStateChange() {
|
||||
|
||||
}
|
||||
|
||||
private void updateUserHandle() {
|
||||
List<UserInfo> profiles = mUm.getProfiles(UserHandle.myUserId());
|
||||
final int profilesCount = profiles.size();
|
||||
mUserHandle = null;
|
||||
for (int i = 0; i < profilesCount; i++) {
|
||||
UserInfo userInfo = profiles.get(i);
|
||||
if (userInfo.isManagedProfile()) {
|
||||
// We assume there's only one managed profile, otherwise UI needs to change.
|
||||
mUserHandle = userInfo.getUserHandle();
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -105,6 +105,7 @@ public class ApnSettings extends RestrictedSettingsFragment implements
|
||||
private RestoreApnProcessHandler mRestoreApnProcessHandler;
|
||||
private HandlerThread mRestoreDefaultApnThread;
|
||||
private SubscriptionInfo mSubscriptionInfo;
|
||||
private int mSubId;
|
||||
private UiccController mUiccController;
|
||||
private String mMvnoType;
|
||||
private String mMvnoMatchData;
|
||||
@@ -140,6 +141,13 @@ public class ApnSettings extends RestrictedSettingsFragment implements
|
||||
} else if(intent.getAction().equals(
|
||||
TelephonyManager.ACTION_SUBSCRIPTION_CARRIER_IDENTITY_CHANGED)) {
|
||||
if (!mRestoreDefaultApnMode) {
|
||||
int extraSubId = intent.getIntExtra(TelephonyManager.EXTRA_SUBSCRIPTION_ID,
|
||||
SubscriptionManager.INVALID_SUBSCRIPTION_ID);
|
||||
if (extraSubId != mSubId) {
|
||||
// subscription has changed
|
||||
mSubId = extraSubId;
|
||||
mSubscriptionInfo = getSubscriptionInfo(mSubId);
|
||||
}
|
||||
fillList();
|
||||
}
|
||||
}
|
||||
@@ -164,7 +172,7 @@ public class ApnSettings extends RestrictedSettingsFragment implements
|
||||
public void onCreate(Bundle icicle) {
|
||||
super.onCreate(icicle);
|
||||
final Activity activity = getActivity();
|
||||
final int subId = activity.getIntent().getIntExtra(SUB_ID,
|
||||
mSubId = activity.getIntent().getIntExtra(SUB_ID,
|
||||
SubscriptionManager.INVALID_SUBSCRIPTION_ID);
|
||||
|
||||
mIntentFilter = new IntentFilter(
|
||||
@@ -173,7 +181,7 @@ public class ApnSettings extends RestrictedSettingsFragment implements
|
||||
|
||||
setIfOnlyAvailableForAdmins(true);
|
||||
|
||||
mSubscriptionInfo = SubscriptionManager.from(activity).getActiveSubscriptionInfo(subId);
|
||||
mSubscriptionInfo = getSubscriptionInfo(mSubId);
|
||||
mUiccController = UiccController.getInstance();
|
||||
|
||||
CarrierConfigManager configManager = (CarrierConfigManager)
|
||||
@@ -254,6 +262,10 @@ public class ApnSettings extends RestrictedSettingsFragment implements
|
||||
return null;
|
||||
}
|
||||
|
||||
private SubscriptionInfo getSubscriptionInfo(int subId) {
|
||||
return SubscriptionManager.from(getActivity()).getActiveSubscriptionInfo(subId);
|
||||
}
|
||||
|
||||
private void fillList() {
|
||||
final TelephonyManager tm = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
|
||||
final int subId = mSubscriptionInfo != null ? mSubscriptionInfo.getSubscriptionId()
|
||||
|
||||
@@ -99,6 +99,11 @@ public class WifiTetherSettings extends RestrictedDashboardFragment
|
||||
super.onAttach(context);
|
||||
mWifiManager = (WifiManager) context.getSystemService(Context.WIFI_SERVICE);
|
||||
mTetherChangeReceiver = new TetherChangeReceiver();
|
||||
|
||||
mSSIDPreferenceController = use(WifiTetherSSIDPreferenceController.class);
|
||||
mSecurityPreferenceController = use(WifiTetherSecurityPreferenceController.class);
|
||||
mPasswordPreferenceController = use(WifiTetherPasswordPreferenceController.class);
|
||||
mApBandPreferenceController = use(WifiTetherApBandPreferenceController.class);
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -140,18 +145,17 @@ public class WifiTetherSettings extends RestrictedDashboardFragment
|
||||
|
||||
@Override
|
||||
protected List<AbstractPreferenceController> createPreferenceControllers(Context context) {
|
||||
final List<AbstractPreferenceController> controllers = new ArrayList<>();
|
||||
mSSIDPreferenceController = new WifiTetherSSIDPreferenceController(context, this);
|
||||
mSecurityPreferenceController = new WifiTetherSecurityPreferenceController(context, this);
|
||||
mPasswordPreferenceController = new WifiTetherPasswordPreferenceController(context, this);
|
||||
mApBandPreferenceController = new WifiTetherApBandPreferenceController(context, this);
|
||||
return buildPreferenceControllers(context, this::onTetherConfigUpdated);
|
||||
}
|
||||
|
||||
private static List<AbstractPreferenceController> buildPreferenceControllers(Context context,
|
||||
WifiTetherBasePreferenceController.OnTetherConfigUpdateListener listener) {
|
||||
final List<AbstractPreferenceController> controllers = new ArrayList<>();
|
||||
controllers.add(new WifiTetherSSIDPreferenceController(context, listener));
|
||||
controllers.add(new WifiTetherSecurityPreferenceController(context, listener));
|
||||
controllers.add(new WifiTetherPasswordPreferenceController(context, listener));
|
||||
controllers.add(new WifiTetherApBandPreferenceController(context, listener));
|
||||
|
||||
controllers.add(mSSIDPreferenceController);
|
||||
controllers.add(mSecurityPreferenceController);
|
||||
controllers.add(mPasswordPreferenceController);
|
||||
controllers.add(mApBandPreferenceController);
|
||||
controllers.add(
|
||||
new WifiTetherAutoOffPreferenceController(context, KEY_WIFI_TETHER_AUTO_OFF));
|
||||
return controllers;
|
||||
}
|
||||
|
||||
@@ -227,6 +231,12 @@ public class WifiTetherSettings extends RestrictedDashboardFragment
|
||||
keys.add(KEY_WIFI_TETHER_SCREEN);
|
||||
return keys;
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<AbstractPreferenceController> createPreferenceControllers(
|
||||
Context context) {
|
||||
return buildPreferenceControllers(context, null /* listener */);
|
||||
}
|
||||
};
|
||||
|
||||
@VisibleForTesting
|
||||
|
||||
@@ -17,10 +17,13 @@
|
||||
package com.android.settings.fuelgauge;
|
||||
|
||||
import static com.google.common.truth.Truth.assertThat;
|
||||
|
||||
import static org.mockito.Matchers.anyBoolean;
|
||||
import static org.mockito.Matchers.anyInt;
|
||||
import static org.mockito.Matchers.anyString;
|
||||
import static org.mockito.Mockito.doNothing;
|
||||
import static org.mockito.Mockito.doReturn;
|
||||
import static org.mockito.Mockito.never;
|
||||
import static org.mockito.Mockito.spy;
|
||||
import static org.mockito.Mockito.verify;
|
||||
import static org.mockito.Mockito.when;
|
||||
@@ -38,6 +41,7 @@ import com.android.settings.testutils.FakeFeatureFactory;
|
||||
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
||||
import com.android.settings.testutils.shadow.SettingsShadowResources;
|
||||
import com.android.settings.testutils.shadow.ShadowFragment;
|
||||
import com.android.settingslib.RestrictedPreference;
|
||||
import com.android.settingslib.fuelgauge.PowerWhitelistBackend;
|
||||
|
||||
import org.junit.Before;
|
||||
@@ -49,8 +53,6 @@ import org.mockito.MockitoAnnotations;
|
||||
import org.robolectric.RuntimeEnvironment;
|
||||
import org.robolectric.annotation.Config;
|
||||
|
||||
import androidx.preference.Preference;
|
||||
|
||||
@RunWith(SettingsRobolectricTestRunner.class)
|
||||
@Config(shadows = {SettingsShadowResources.SettingsShadowTheme.class, ShadowFragment.class})
|
||||
public class BackgroundActivityPreferenceControllerTest {
|
||||
@@ -80,7 +82,7 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
@Mock
|
||||
private PowerWhitelistBackend mPowerWhitelistBackend;
|
||||
private BackgroundActivityPreferenceController mController;
|
||||
private Preference mPreference;
|
||||
private RestrictedPreference mPreference;
|
||||
private Context mShadowContext;
|
||||
private BatteryUtils mBatteryUtils;
|
||||
|
||||
@@ -109,7 +111,7 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
mBatteryUtils = spy(new BatteryUtils(mShadowContext));
|
||||
doNothing().when(mBatteryUtils).setForceAppStandby(anyInt(), anyString(), anyInt());
|
||||
|
||||
mPreference = new Preference(mShadowContext);
|
||||
mPreference = spy(new RestrictedPreference(mShadowContext, null /* attrs */));
|
||||
mPreference.setKey(BackgroundActivityPreferenceController.KEY_BACKGROUND_ACTIVITY);
|
||||
mController = spy(new BackgroundActivityPreferenceController(
|
||||
mContext, mFragment, UID_LOW_SDK, LOW_SDK_PACKAGE, mPowerWhitelistBackend));
|
||||
@@ -118,7 +120,7 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testHandlePreferenceTreeClick_restrictApp_showDialog() {
|
||||
public void handlePreferenceTreeClick_restrictApp_showDialog() {
|
||||
doReturn(AppOpsManager.MODE_ALLOWED).when(mAppOpsManager)
|
||||
.checkOpNoThrow(anyInt(), anyInt(), anyString());
|
||||
|
||||
@@ -128,7 +130,7 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testHandlePreferenceTreeClick_unRestrictApp_showDialog() {
|
||||
public void handlePreferenceTreeClick_unRestrictApp_showDialog() {
|
||||
doReturn(AppOpsManager.MODE_IGNORED).when(mAppOpsManager)
|
||||
.checkOpNoThrow(anyInt(), anyInt(), anyString());
|
||||
|
||||
@@ -138,7 +140,7 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testUpdateState_noError_setEnabled() {
|
||||
public void updateState_noError_setEnabled() {
|
||||
when(mAppOpsManager.checkOpNoThrow(AppOpsManager.OP_RUN_ANY_IN_BACKGROUND, UID_LOW_SDK,
|
||||
LOW_SDK_PACKAGE)).thenReturn(AppOpsManager.MODE_ALLOWED);
|
||||
|
||||
@@ -149,7 +151,7 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testUpdateState_whitelisted() {
|
||||
public void updateState_whitelisted() {
|
||||
when(mPowerWhitelistBackend.isWhitelisted(LOW_SDK_PACKAGE)).thenReturn(true);
|
||||
mController.updateState(mPreference);
|
||||
assertThat(mPreference.isEnabled()).isFalse();
|
||||
@@ -158,7 +160,16 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testUpdateSummary_modeError_showSummaryDisabled() {
|
||||
public void updateState_disabledByAdmin_doNothing() {
|
||||
doReturn(true).when(mPreference).isDisabledByAdmin();
|
||||
|
||||
mController.updateState(mPreference);
|
||||
|
||||
verify(mPreference, never()).setEnabled(anyBoolean());
|
||||
}
|
||||
|
||||
@Test
|
||||
public void updateSummary_modeError_showSummaryDisabled() {
|
||||
when(mAppOpsManager.checkOpNoThrow(AppOpsManager.OP_RUN_ANY_IN_BACKGROUND, UID_LOW_SDK,
|
||||
LOW_SDK_PACKAGE)).thenReturn(AppOpsManager.MODE_ERRORED);
|
||||
final CharSequence expectedSummary = mShadowContext.getText(
|
||||
@@ -169,7 +180,7 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testUpdateSummary_modeDefault_showNotRestricted() {
|
||||
public void updateSummary_modeDefault_showNotRestricted() {
|
||||
when(mAppOpsManager.checkOpNoThrow(AppOpsManager.OP_RUN_ANY_IN_BACKGROUND, UID_LOW_SDK,
|
||||
LOW_SDK_PACKAGE)).thenReturn(AppOpsManager.MODE_DEFAULT);
|
||||
|
||||
@@ -179,7 +190,7 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testUpdateSummary_modeIgnored_showRestricted() {
|
||||
public void updateSummary_modeIgnored_showRestricted() {
|
||||
when(mAppOpsManager.checkOpNoThrow(AppOpsManager.OP_RUN_ANY_IN_BACKGROUND, UID_LOW_SDK,
|
||||
LOW_SDK_PACKAGE)).thenReturn(AppOpsManager.MODE_IGNORED);
|
||||
|
||||
@@ -189,7 +200,7 @@ public class BackgroundActivityPreferenceControllerTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testIsAvailable_ReturnTrue() {
|
||||
public void isAvailable_ReturnTrue() {
|
||||
assertThat(mController.isAvailable()).isTrue();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -14,15 +14,13 @@
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package com.android.settings.homepage.conditional;
|
||||
package com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import static com.google.common.truth.Truth.assertThat;
|
||||
|
||||
import static org.mockito.Mockito.when;
|
||||
import static org.mockito.Mockito.never;
|
||||
import static org.mockito.Mockito.verify;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.graphics.drawable.Drawable;
|
||||
import android.media.AudioManager;
|
||||
|
||||
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
||||
@@ -35,7 +33,7 @@ import org.mockito.MockitoAnnotations;
|
||||
import org.robolectric.RuntimeEnvironment;
|
||||
|
||||
@RunWith(SettingsRobolectricTestRunner.class)
|
||||
public class AbnormalRingerConditionBaseTest {
|
||||
public class AbnormalRingerConditionControllerBaseTest {
|
||||
|
||||
@Mock
|
||||
private ConditionManager mConditionManager;
|
||||
@@ -47,53 +45,39 @@ public class AbnormalRingerConditionBaseTest {
|
||||
public void setUp() {
|
||||
MockitoAnnotations.initMocks(this);
|
||||
mContext = RuntimeEnvironment.application;
|
||||
when(mConditionManager.getContext()).thenReturn(mContext);
|
||||
mCondition = new TestCondition(mConditionManager);
|
||||
|
||||
mCondition = new TestCondition(mContext, mConditionManager);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void newInstance_shouldMonitorRingerStateChangeBroadcast() {
|
||||
public void startMonitor_shouldMonitorRingerStateChangeBroadcast() {
|
||||
final Intent broadcast1 = new Intent("foo.bar.action");
|
||||
final Intent broadcast2 = new Intent(AudioManager.INTERNAL_RINGER_MODE_CHANGED_ACTION);
|
||||
|
||||
mCondition.startMonitoringStateChange();
|
||||
|
||||
mContext.sendBroadcast(broadcast1);
|
||||
assertThat(mCondition.mRefreshCalled).isFalse();
|
||||
verify(mConditionManager, never()).onConditionChanged();
|
||||
|
||||
mContext.sendBroadcast(broadcast2);
|
||||
assertThat(mCondition.mRefreshCalled).isTrue();
|
||||
verify(mConditionManager).onConditionChanged();
|
||||
}
|
||||
|
||||
private static class TestCondition extends AbnormalRingerConditionBase {
|
||||
private boolean mRefreshCalled;
|
||||
private static class TestCondition extends AbnormalRingerConditionController {
|
||||
|
||||
TestCondition(ConditionManager manager) {
|
||||
super(manager);
|
||||
public TestCondition(Context appContext, ConditionManager conditionManager) {
|
||||
super(appContext, conditionManager);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void refreshState() {
|
||||
mRefreshCalled = true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsConstant() {
|
||||
public long getId() {
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Drawable getIcon() {
|
||||
return null;
|
||||
public boolean isDisplayable() {
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getTitle() {
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CharSequence getSummary() {
|
||||
return null;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
@@ -13,13 +13,12 @@
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package com.android.settings.homepage.conditional;
|
||||
package com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import static com.google.common.truth.Truth.assertThat;
|
||||
|
||||
import static org.mockito.Mockito.spy;
|
||||
import static org.mockito.Mockito.verify;
|
||||
import static org.mockito.Mockito.when;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
@@ -31,30 +30,25 @@ import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
import org.junit.runner.RunWith;
|
||||
import org.mockito.ArgumentCaptor;
|
||||
import org.mockito.Mock;
|
||||
import org.mockito.MockitoAnnotations;
|
||||
import org.robolectric.RuntimeEnvironment;
|
||||
|
||||
@RunWith(SettingsRobolectricTestRunner.class)
|
||||
public class BackgroundDataConditionTest {
|
||||
@Mock
|
||||
private ConditionManager mConditionManager;
|
||||
|
||||
public class BackgroundDataConditionControllerTest {
|
||||
private Context mContext;
|
||||
private BackgroundDataConditionController mController;
|
||||
|
||||
@Before
|
||||
public void setUp() {
|
||||
MockitoAnnotations.initMocks(this);
|
||||
mContext = spy(RuntimeEnvironment.application);
|
||||
when(mConditionManager.getContext()).thenReturn(mContext);
|
||||
mController = new BackgroundDataConditionController(mContext);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void onPrimaryClick_shouldReturn2SummaryActivity() {
|
||||
final ArgumentCaptor<Intent> argumentCaptor = ArgumentCaptor.forClass(Intent.class);
|
||||
BackgroundDataCondition backgroundDataCondition
|
||||
= new BackgroundDataCondition(mConditionManager);
|
||||
backgroundDataCondition.onPrimaryClick();
|
||||
mController.onPrimaryClick(mContext);
|
||||
verify(mContext).startActivity(argumentCaptor.capture());
|
||||
Intent intent = argumentCaptor.getValue();
|
||||
|
||||
@@ -0,0 +1,87 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import static com.google.common.truth.Truth.assertThat;
|
||||
|
||||
import static org.mockito.ArgumentMatchers.any;
|
||||
import static org.mockito.Mockito.spy;
|
||||
import static org.mockito.Mockito.verify;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.IntentFilter;
|
||||
import android.os.PowerManager;
|
||||
|
||||
import com.android.settings.fuelgauge.BatterySaverReceiver;
|
||||
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
||||
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
import org.junit.runner.RunWith;
|
||||
import org.mockito.Mock;
|
||||
import org.mockito.MockitoAnnotations;
|
||||
import org.robolectric.RuntimeEnvironment;
|
||||
import org.robolectric.Shadows;
|
||||
import org.robolectric.shadows.ShadowPowerManager;
|
||||
|
||||
@RunWith(SettingsRobolectricTestRunner.class)
|
||||
public class BatterySaverConditionControllerTest {
|
||||
@Mock
|
||||
private ConditionManager mConditionManager;
|
||||
|
||||
private ShadowPowerManager mPowerManager;
|
||||
private Context mContext;
|
||||
private BatterySaverConditionController mController;
|
||||
|
||||
@Before
|
||||
public void setUp() {
|
||||
MockitoAnnotations.initMocks(this);
|
||||
mContext = spy(RuntimeEnvironment.application);
|
||||
mPowerManager = Shadows.shadowOf(mContext.getSystemService(PowerManager.class));
|
||||
mController = new BatterySaverConditionController(mContext, mConditionManager);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void startMonitor_shouldRegisterReceiver() {
|
||||
mController.startMonitoringStateChange();
|
||||
|
||||
verify(mContext).registerReceiver(any(BatterySaverReceiver.class), any(IntentFilter.class));
|
||||
}
|
||||
|
||||
@Test
|
||||
public void stopMonitor_shouldUnregisterReceiver() {
|
||||
mController.startMonitoringStateChange();
|
||||
mController.stopMonitoringStateChange();
|
||||
|
||||
verify(mContext).unregisterReceiver(any(BatterySaverReceiver.class));
|
||||
}
|
||||
|
||||
@Test
|
||||
public void isDisplayable_PowerSaverOn_true() {
|
||||
mPowerManager.setIsPowerSaveMode(true);
|
||||
|
||||
assertThat(mController.isDisplayable()).isTrue();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void isDisplayable_PowerSaverOff_false() {
|
||||
mPowerManager.setIsPowerSaveMode(false);
|
||||
|
||||
assertThat(mController.isDisplayable()).isFalse();
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,57 @@
|
||||
/*
|
||||
* Copyright (C) 2018 The Android Open Source Project
|
||||
*
|
||||
* 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 com.android.settings.homepage.conditional.v2;
|
||||
|
||||
import static org.mockito.ArgumentMatchers.argThat;
|
||||
import static org.mockito.Mockito.spy;
|
||||
import static org.mockito.Mockito.verify;
|
||||
|
||||
import android.content.ComponentName;
|
||||
import android.content.Context;
|
||||
|
||||
import com.android.settings.Settings;
|
||||
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
||||
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
import org.junit.runner.RunWith;
|
||||
import org.robolectric.RuntimeEnvironment;
|
||||
|
||||
@RunWith(SettingsRobolectricTestRunner.class)
|
||||
public class WorkModeConditionControllerTest {
|
||||
|
||||
private Context mContext;
|
||||
private WorkModeConditionController mController;
|
||||
|
||||
@Before
|
||||
public void setUp() {
|
||||
mContext = spy(RuntimeEnvironment.application);
|
||||
mController = new WorkModeConditionController(mContext);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void onPrimaryClick_shouldLaunchAccountsSetting() {
|
||||
final ComponentName componentName =
|
||||
new ComponentName(mContext, Settings.AccountDashboardActivity.class);
|
||||
|
||||
mController.onPrimaryClick(mContext);
|
||||
|
||||
verify(mContext).startActivity(
|
||||
argThat(intent -> intent.getComponent().equals(componentName)));
|
||||
}
|
||||
|
||||
}
|
||||
@@ -52,6 +52,11 @@ public class ShadowWifiManager extends org.robolectric.shadows.ShadowWifiManager
|
||||
return Collections.emptyList();
|
||||
}
|
||||
|
||||
@Implementation
|
||||
public boolean isDualModeSupported() {
|
||||
return false;
|
||||
}
|
||||
|
||||
public static ShadowWifiManager get() {
|
||||
return Shadow.extract(application.getSystemService(WifiManager.class));
|
||||
}
|
||||
|
||||
@@ -17,14 +17,13 @@
|
||||
package com.android.settings.wallpaper;
|
||||
|
||||
import static com.google.common.truth.Truth.assertThat;
|
||||
|
||||
import static org.mockito.Mockito.when;
|
||||
|
||||
import android.app.WallpaperManager;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.res.Resources;
|
||||
|
||||
import com.android.settings.SubSettings;
|
||||
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
||||
|
||||
import org.junit.After;
|
||||
@@ -35,14 +34,11 @@ import org.mockito.Mock;
|
||||
import org.mockito.MockitoAnnotations;
|
||||
import org.robolectric.Robolectric;
|
||||
import org.robolectric.RuntimeEnvironment;
|
||||
import org.robolectric.Shadows;
|
||||
import org.robolectric.android.controller.ActivityController;
|
||||
import org.robolectric.annotation.Config;
|
||||
import org.robolectric.annotation.Implementation;
|
||||
import org.robolectric.annotation.Implements;
|
||||
import org.robolectric.annotation.Resetter;
|
||||
import org.robolectric.shadows.ShadowActivity;
|
||||
import org.robolectric.shadows.ShadowPackageManager;
|
||||
|
||||
@RunWith(SettingsRobolectricTestRunner.class)
|
||||
public class WallpaperSuggestionActivityTest {
|
||||
@@ -65,20 +61,6 @@ public class WallpaperSuggestionActivityTest {
|
||||
ShadowWallpaperManager.reset();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void launch_primarySuggestionActivityDoesNotExist_shouldFallback() {
|
||||
ShadowPackageManager packageManager =
|
||||
Shadows.shadowOf(RuntimeEnvironment.application.getPackageManager());
|
||||
packageManager.removePackage("com.android.settings");
|
||||
|
||||
ShadowActivity activity = Shadows.shadowOf(mController.setup().get());
|
||||
final Intent intent = activity.getNextStartedActivity();
|
||||
|
||||
assertThat(intent.getComponent().getClassName()).isEqualTo(SubSettings.class.getName());
|
||||
assertThat(intent.getFlags()).isEqualTo(Intent.FLAG_ACTIVITY_FORWARD_RESULT);
|
||||
assertThat(activity.isFinishing()).isTrue();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void wallpaperServiceEnabled_no_shouldReturnTrue() {
|
||||
when(mContext.getResources()).thenReturn(mResources);
|
||||
|
||||
@@ -17,6 +17,7 @@
|
||||
package com.android.settings.wifi.tether;
|
||||
|
||||
import static com.google.common.truth.Truth.assertThat;
|
||||
|
||||
import static org.mockito.Mockito.doReturn;
|
||||
import static org.mockito.Mockito.spy;
|
||||
import static org.mockito.Mockito.when;
|
||||
@@ -27,6 +28,7 @@ import android.os.UserHandle;
|
||||
import android.os.UserManager;
|
||||
|
||||
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
||||
import com.android.settings.testutils.shadow.ShadowWifiManager;
|
||||
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
@@ -34,12 +36,15 @@ import org.junit.runner.RunWith;
|
||||
import org.mockito.Mock;
|
||||
import org.mockito.MockitoAnnotations;
|
||||
import org.robolectric.RuntimeEnvironment;
|
||||
import org.robolectric.annotation.Config;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
@RunWith(SettingsRobolectricTestRunner.class)
|
||||
@Config(shadows = {ShadowWifiManager.class})
|
||||
public class WifiTetherSettingsTest {
|
||||
private static final String[] WIFI_REGEXS = {"wifi_regexs"};
|
||||
|
||||
private Context mContext;
|
||||
|
||||
@@ -55,12 +60,12 @@ public class WifiTetherSettingsTest {
|
||||
MockitoAnnotations.initMocks(this);
|
||||
doReturn(mConnectivityManager)
|
||||
.when(mContext).getSystemService(Context.CONNECTIVITY_SERVICE);
|
||||
doReturn(mUserManager)
|
||||
.when(mContext).getSystemService(Context.USER_SERVICE);
|
||||
doReturn(WIFI_REGEXS).when(mConnectivityManager).getTetherableWifiRegexs();
|
||||
doReturn(mUserManager).when(mContext).getSystemService(Context.USER_SERVICE);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testWifiTetherNonIndexableKeys_tetherAvailable_keysNotReturned() {
|
||||
public void wifiTetherNonIndexableKeys_tetherAvailable_keysNotReturned() {
|
||||
// To let TetherUtil.isTetherAvailable return true, select one of the combinations
|
||||
setupIsTetherAvailable(true);
|
||||
|
||||
@@ -74,7 +79,7 @@ public class WifiTetherSettingsTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testWifiTetherNonIndexableKeys_tetherNotAvailable_keysReturned() {
|
||||
public void wifiTetherNonIndexableKeys_tetherNotAvailable_keysReturned() {
|
||||
// To let TetherUtil.isTetherAvailable return false, select one of the combinations
|
||||
setupIsTetherAvailable(false);
|
||||
|
||||
@@ -87,6 +92,12 @@ public class WifiTetherSettingsTest {
|
||||
assertThat(niks).contains(WifiTetherSettings.KEY_WIFI_TETHER_NETWORK_AP_BAND);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void createPreferenceControllers_notEmpty() {
|
||||
assertThat(WifiTetherSettings.SEARCH_INDEX_DATA_PROVIDER.getPreferenceControllers(mContext))
|
||||
.isNotEmpty();
|
||||
}
|
||||
|
||||
private void setupIsTetherAvailable(boolean returnValue) {
|
||||
when(mConnectivityManager.isTetheringSupported()).thenReturn(true);
|
||||
|
||||
|
||||
Reference in New Issue
Block a user