开源可翻折的TextViewExpandableTextView

Version:1.0StartHTML:000000221EndHTML:000087781StartFragment:000027003EndFragment:000087708StartSelection:000027003EndSelection:000087702SourceURL:https://blog.csdn.net/qq_22060403/article/details/80649484

实现的效果图如下 ;

点击查看详情把显示不全的数据显示出来 产接下里

接下来  我们来看下布局 

                    >" android:textsize="12sp" android:textcolor="@color/colorPrimary" android:paddingright="10dp" android:layout_margintop="9dp" android:background="@android:color/transparent" android:layout_gravity="right|bottom">   

布局文件相对比较简单   

java代码相对比较简单     下面上代码  

public class MainActivityextends AppCompatActivity {

ExpandableTextViewexpand_text_view;

    @Override

    protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main);

        expand_text_view = findViewById(R.id.expand_text_view);

        expand_text_view.setText("adasd asd        " +

"  asd asd asd asd asd asdasdasddasdasdasdasdasdddddsadsdasd s dasd as das dasd sad asd    asd asd asd " +

" ads asd asd as das ads asd d asd as dasd asd这里是内容信息正文," +

"这里是内容信息正文,这里是内容信息正文,这里是内容信息正文,这里是内容信息正文,这里是内容信息正文,这里是内容信息正文,这里是内容信息正文,这里是内容信息正文。 ");

    }

}

自定义view

class ExpandableTextViewextends LinearLayoutimplements View.OnClickListener {

private static final StringTAG = ExpandableTextView.class.getSimpleName();

    /* The default number of lines */

    private static final int MAX_COLLAPSED_LINES =8;

    /* The default animation duration */

    private static final int DEFAULT_ANIM_DURATION =300;

    /* The default alpha value when the animation starts */

    private static final float DEFAULT_ANIM_ALPHA_START =0.7f;

    protected TextViewmTv;

    protected TextViewmTextView; // Button to expand/collapse

    private boolean mRelayout;

    private boolean mCollapsed =true; // Show short version as default.

    private int mCollapsedHeight;

    private int mTextHeightWithMaxLines;

    private int mMaxCollapsedLines;

    private int mMarginBetweenTxtAndBottom;

    private int mAnimationDuration;

    private float mAnimAlphaStart;

    private boolean mAnimating;

    /* Listener for callback */

    private OnExpandStateChangeListenermListener;

    /* For saving collapsed status when used in ListView */

    private SparseBooleanArraymCollapsedStatus;

    private int mPosition;

    public ExpandableTextView(Context context) {

this(context, null);

    }

public ExpandableTextView(Context context, AttributeSet attrs) {

super(context, attrs);

        init(attrs);

    }

@TargetApi(Build.VERSION_CODES.HONEYCOMB)

public ExpandableTextView(Context context, AttributeSet attrs, int defStyle) {

super(context, attrs, defStyle);

        init(attrs);

    }

@Override

    public void setOrientation(int orientation) {

if (LinearLayout.HORIZONTAL == orientation) {

throw new IllegalArgumentException("ExpandableTextView only supports Vertical Orientation.");

        }

super.setOrientation(orientation);

    }

@Override

    public void onClick(View view) {

if (mTextView.getVisibility() != View.VISIBLE) {

return;

        }

mCollapsed = !mCollapsed;

        if (mCollapsedStatus !=null) {

mCollapsedStatus.put(mPosition, mCollapsed);

        }

// mark that the animation is in progress

        mAnimating =true;

        Animation animation;

        if (mCollapsed) {

animation =new ExpandCollapseAnimation(this, getHeight(), mCollapsedHeight);

        }else {

animation =new ExpandCollapseAnimation(this, getHeight(), getHeight() +

mTextHeightWithMaxLines -mTv.getHeight());

        }

animation.setFillAfter(true);

        animation.setAnimationListener(new Animation.AnimationListener() {

@Override

            public void onAnimationStart(Animation animation) {

applyAlphaAnimation(mTv, mAnimAlphaStart);

            }

@Override

            public void onAnimationEnd(Animation animation) {

// clear animation here to avoid repeated applyTransformation() calls

                clearAnimation();

                // clear the animation flag

                mAnimating =false;

                // notify the listener

                if (mListener !=null) {

mListener.onExpandStateChanged(mTv, !mCollapsed);

                }

}

@Override

            public void onAnimationRepeat(Animation animation) {

}

});

        clearAnimation();

        startAnimation(animation);

    }

@Override

    public boolean onInterceptTouchEvent(MotionEvent ev) {

// while an animation is in progress, intercept all the touch events to children to

// prevent extra clicks during the animation

        return mAnimating;

    }

@Override

    protected void onFinishInflate() {

super.onFinishInflate();

        findViews();

    }

@Override

    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

// If no change, measure and return

        if (!mRelayout || getVisibility() == View.GONE) {

super.onMeasure(widthMeasureSpec, heightMeasureSpec);

return;

        }

mRelayout =false;

        // Setup with optimistic case

// i.e. Everything fits. No button needed

        mTextView.setVisibility(View.GONE);

        mTv.setMaxLines(Integer.MAX_VALUE);

        // Measure

        super.onMeasure(widthMeasureSpec, heightMeasureSpec);

        // If the text fits in collapsed mode, we are done.

        if (mTv.getLineCount() <=mMaxCollapsedLines) {

return;

        }

// Saves the text height w/ max lines

        mTextHeightWithMaxLines =getRealTextViewHeight(mTv);

        // Doesn't fit in collapsed mode. Collapse text view as needed. Show

// button.

        if (mCollapsed) {

mTv.setMaxLines(mMaxCollapsedLines);

        }

mTextView.setVisibility(View.VISIBLE);

        // Re-measure with new setup

        super.onMeasure(widthMeasureSpec, heightMeasureSpec);

        if (mCollapsed) {

// Gets the margin between the TextView's bottom and the ViewGroup's bottom

            mTv.post(new Runnable() {

@Override

                public void run() {

mMarginBetweenTxtAndBottom = getHeight() -mTv.getHeight();

                }

});

            // Saves the collapsed height of this ViewGroup

            mCollapsedHeight = getMeasuredHeight();

        }

}

public void setOnExpandStateChangeListener(@Nullable OnExpandStateChangeListener listener) {

mListener = listener;

    }

public void setText(@Nullable CharSequence text) {

mRelayout =true;

        mTv.setText(text);

        setVisibility(TextUtils.isEmpty(text) ? View.GONE : View.VISIBLE);

    }

public void setText(@Nullable CharSequence text, @NonNull SparseBooleanArray collapsedStatus, int position) {

mCollapsedStatus = collapsedStatus;

        mPosition = position;

        boolean isCollapsed = collapsedStatus.get(position, true);

        clearAnimation();

        mCollapsed = isCollapsed;

        setText(text);

        getLayoutParams().height = ViewGroup.LayoutParams.WRAP_CONTENT;

        requestLayout();

    }

@Nullable

    public CharSequencegetText() {

if (mTv ==null) {

return "";

        }

return mTv.getText();

    }

private void init(AttributeSet attrs) {

TypedArray typedArray = getContext().obtainStyledAttributes(attrs, R.styleable.ExpandableTextView);

        mMaxCollapsedLines = typedArray.getInt(R.styleable.ExpandableTextView_maxCollapsedLines, MAX_COLLAPSED_LINES);

        mAnimationDuration = typedArray.getInt(R.styleable.ExpandableTextView_animDuration, DEFAULT_ANIM_DURATION);

        mAnimAlphaStart = typedArray.getFloat(R.styleable.ExpandableTextView_animAlphaStart, DEFAULT_ANIM_ALPHA_START);

        typedArray.recycle();

        // enforces vertical orientation

        setOrientation(LinearLayout.VERTICAL);

        // default visibility is gone

        setVisibility(GONE);

    }

private void findViews() {

mTv = (TextView) findViewById(R.id.expandable_text);

        mTv.setOnClickListener(this);

        mTextView = (TextView) findViewById(R.id.expand_collapse);

        mTextView.setOnClickListener(this);

    }

private static boolean isPostHoneycomb() {

return Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB;

    }

private static boolean isPostLolipop() {

return Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP;

    }

@TargetApi(Build.VERSION_CODES.HONEYCOMB)

private static void applyAlphaAnimation(View view, float alpha) {

if (isPostHoneycomb()) {

view.setAlpha(alpha);

        }else {

AlphaAnimation alphaAnimation =new AlphaAnimation(alpha, alpha);

            // make it instant

            alphaAnimation.setDuration(0);

            alphaAnimation.setFillAfter(true);

            view.startAnimation(alphaAnimation);

        }

}

@TargetApi(Build.VERSION_CODES.LOLLIPOP)

private static DrawablegetDrawable(@NonNull Context context, @DrawableRes int resId) {

Resources resources = context.getResources();

        if (isPostLolipop()) {

return resources.getDrawable(resId, context.getTheme());

        }else {

return resources.getDrawable(resId);

        }

}

private static int getRealTextViewHeight(@NonNull TextView textView) {

int textHeight = textView.getLayout().getLineTop(textView.getLineCount());

        int padding = textView.getCompoundPaddingTop() + textView.getCompoundPaddingBottom();

        return textHeight + padding;

    }

class ExpandCollapseAnimationextends Animation {

private final ViewmTargetView;

        private final int mStartHeight;

        private final int mEndHeight;

        public ExpandCollapseAnimation(View view, int startHeight, int endHeight) {

mTargetView = view;

            mStartHeight = startHeight;

            mEndHeight = endHeight;

            setDuration(mAnimationDuration);

        }

@Override

        protected void applyTransformation(float interpolatedTime, Transformation t) {

final int newHeight = (int) ((mEndHeight -mStartHeight) * interpolatedTime +mStartHeight);

            mTv.setMaxHeight(newHeight -mMarginBetweenTxtAndBottom);

            if (Float.compare(mAnimAlphaStart, 1.0f) !=0) {

applyAlphaAnimation(mTv, mAnimAlphaStart + interpolatedTime * (1.0f -mAnimAlphaStart));

            }

mTargetView.getLayoutParams().height = newHeight;

            mTargetView.requestLayout();

        }

@Override

        public void initialize(int width, int height, int parentWidth, int parentHeight) {

super.initialize(width, height, parentWidth, parentHeight);

        }

@Override

        public boolean willChangeBounds() {

return true;

        }

}

public interface OnExpandStateChangeListener {

/**

* Called when the expand/collapse animation has been finished

*

        * @param textView  - TextView being expanded/collapsed

        * @param isExpanded - true if the TextView has been expanded

*/

        void onExpandStateChanged(TextView textView, boolean isExpanded);

    }

}

这样就能简单的实现我们想要的效果  至于其他类似布局 仅仅在此基础上做稍微改动即可 

你可能感兴趣的:(开源可翻折的TextViewExpandableTextView)