-
-
Notifications
You must be signed in to change notification settings - Fork 12
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #22 from arch10/display-enhancement
Display enhancement
- Loading branch information
Showing
16 changed files
with
638 additions
and
208 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
201 changes: 201 additions & 0 deletions
201
app/src/main/java/com/example/arch1/testapplication/CalculatorEditText.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,201 @@ | ||
/* | ||
* Copyright (C) 2014 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.example.arch1.testapplication; | ||
|
||
import android.content.Context; | ||
import android.content.res.TypedArray; | ||
import android.graphics.Paint; | ||
import android.graphics.Paint.FontMetricsInt; | ||
import android.graphics.Rect; | ||
import android.os.Parcelable; | ||
import android.text.method.ScrollingMovementMethod; | ||
import android.text.TextPaint; | ||
import android.util.AttributeSet; | ||
import android.util.TypedValue; | ||
import android.view.ActionMode; | ||
import android.view.Menu; | ||
import android.view.MenuItem; | ||
import android.view.MotionEvent; | ||
import android.widget.TextView; | ||
import androidx.appcompat.widget.AppCompatEditText; | ||
|
||
public class CalculatorEditText extends AppCompatEditText { | ||
|
||
private final static ActionMode.Callback NO_SELECTION_ACTION_MODE_CALLBACK = | ||
new ActionMode.Callback() { | ||
@Override | ||
public boolean onActionItemClicked(ActionMode mode, MenuItem item) { | ||
return false; | ||
} | ||
|
||
@Override | ||
public boolean onCreateActionMode(ActionMode mode, Menu menu) { | ||
// Prevents the selection action mode on double tap. | ||
return false; | ||
} | ||
|
||
@Override | ||
public void onDestroyActionMode(ActionMode mode) { | ||
} | ||
|
||
@Override | ||
public boolean onPrepareActionMode(ActionMode mode, Menu menu) { | ||
return false; | ||
} | ||
}; | ||
|
||
private final float mMaximumTextSize; | ||
private final float mMinimumTextSize; | ||
private final float mStepTextSize; | ||
|
||
// Temporary objects for use in layout methods. | ||
private final Paint mTempPaint = new TextPaint(); | ||
private final Rect mTempRect = new Rect(); | ||
|
||
private int mWidthConstraint = -1; | ||
private OnTextSizeChangeListener mOnTextSizeChangeListener; | ||
|
||
public CalculatorEditText(Context context) { | ||
this(context, null); | ||
} | ||
|
||
public CalculatorEditText(Context context, AttributeSet attrs) { | ||
this(context, attrs, 0); | ||
} | ||
|
||
public CalculatorEditText(Context context, AttributeSet attrs, int defStyle) { | ||
super(context, attrs, defStyle); | ||
|
||
final TypedArray a = context.obtainStyledAttributes( | ||
attrs, R.styleable.CalculatorEditText, defStyle, 0); | ||
mMaximumTextSize = a.getDimension( | ||
R.styleable.CalculatorEditText_maxTextSize, getTextSize()); | ||
mMinimumTextSize = a.getDimension( | ||
R.styleable.CalculatorEditText_minTextSize, getTextSize()); | ||
mStepTextSize = a.getDimension(R.styleable.CalculatorEditText_stepTextSize, | ||
(mMaximumTextSize - mMinimumTextSize) / 3); | ||
|
||
a.recycle(); | ||
|
||
setCustomSelectionActionModeCallback(NO_SELECTION_ACTION_MODE_CALLBACK); | ||
if (isFocusable()) { | ||
setMovementMethod(ScrollingMovementMethod.getInstance()); | ||
} | ||
setTextSize(TypedValue.COMPLEX_UNIT_PX, mMaximumTextSize); | ||
setMinHeight(getLineHeight() + getCompoundPaddingBottom() + getCompoundPaddingTop()); | ||
} | ||
|
||
@Override | ||
public boolean onTouchEvent(MotionEvent event) { | ||
if (event.getActionMasked() == MotionEvent.ACTION_UP) { | ||
// Hack to prevent keyboard and insertion handle from showing. | ||
cancelLongPress(); | ||
} | ||
return super.onTouchEvent(event); | ||
} | ||
|
||
@Override | ||
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { | ||
super.onMeasure(widthMeasureSpec, heightMeasureSpec); | ||
|
||
mWidthConstraint = | ||
MeasureSpec.getSize(widthMeasureSpec) - getPaddingLeft() - getPaddingRight(); | ||
setTextSize(TypedValue.COMPLEX_UNIT_PX, getVariableTextSize(getText().toString())); | ||
} | ||
|
||
@Override | ||
public Parcelable onSaveInstanceState() { | ||
super.onSaveInstanceState(); | ||
|
||
// EditText will freeze any text with a selection regardless of getFreezesText() -> | ||
// return null to prevent any state from being preserved at the instance level. | ||
return null; | ||
} | ||
|
||
@Override | ||
protected void onTextChanged(CharSequence text, int start, int lengthBefore, int lengthAfter) { | ||
super.onTextChanged(text, start, lengthBefore, lengthAfter); | ||
|
||
final int textLength = text.length(); | ||
if (getSelectionStart() != textLength || getSelectionEnd() != textLength) { | ||
// Pin the selection to the end of the current text. | ||
setSelection(textLength); | ||
} | ||
setTextSize(TypedValue.COMPLEX_UNIT_PX, getVariableTextSize(text.toString())); | ||
} | ||
|
||
@Override | ||
public void setTextSize(int unit, float size) { | ||
final float oldTextSize = getTextSize(); | ||
super.setTextSize(unit, size); | ||
|
||
if (mOnTextSizeChangeListener != null && getTextSize() != oldTextSize) { | ||
mOnTextSizeChangeListener.onTextSizeChanged(this, oldTextSize); | ||
} | ||
} | ||
|
||
public void setOnTextSizeChangeListener(OnTextSizeChangeListener listener) { | ||
mOnTextSizeChangeListener = listener; | ||
} | ||
|
||
public float getVariableTextSize(String text) { | ||
if (mWidthConstraint < 0 || mMaximumTextSize <= mMinimumTextSize) { | ||
// Not measured, bail early. | ||
return getTextSize(); | ||
} | ||
|
||
// Capture current paint state. | ||
mTempPaint.set(getPaint()); | ||
|
||
// Step through increasing text sizes until the text would no longer fit. | ||
float lastFitTextSize = mMinimumTextSize; | ||
while (lastFitTextSize < mMaximumTextSize) { | ||
final float nextSize = Math.min(lastFitTextSize + mStepTextSize, mMaximumTextSize); | ||
mTempPaint.setTextSize(nextSize); | ||
if (mTempPaint.measureText(text) > mWidthConstraint) { | ||
break; | ||
} else { | ||
lastFitTextSize = nextSize; | ||
} | ||
} | ||
|
||
return lastFitTextSize; | ||
} | ||
|
||
@Override | ||
public int getCompoundPaddingTop() { | ||
// Measure the top padding from the capital letter height of the text instead of the top, | ||
// but don't remove more than the available top padding otherwise clipping may occur. | ||
getPaint().getTextBounds("H", 0, 1, mTempRect); | ||
|
||
final FontMetricsInt fontMetrics = getPaint().getFontMetricsInt(); | ||
final int paddingOffset = -(fontMetrics.ascent + mTempRect.height()); | ||
return super.getCompoundPaddingTop() - Math.min(getPaddingTop(), paddingOffset); | ||
} | ||
|
||
@Override | ||
public int getCompoundPaddingBottom() { | ||
// Measure the bottom padding from the baseline of the text instead of the bottom, but don't | ||
// remove more than the available bottom padding otherwise clipping may occur. | ||
final FontMetricsInt fontMetrics = getPaint().getFontMetricsInt(); | ||
return super.getCompoundPaddingBottom() - Math.min(getPaddingBottom(), fontMetrics.descent); | ||
} | ||
|
||
public interface OnTextSizeChangeListener { | ||
void onTextSizeChanged(TextView textView, float oldSize); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.