forked from organicmaps/organicmaps
[android] Migrated to LinearLayout for description block, added placeholder for empty bookmarks list frag, add prefs for header on catalog screen
This commit is contained in:
parent
8fef64b2e1
commit
2e88c4f490
14 changed files with 126 additions and 518 deletions
|
@ -107,7 +107,6 @@ dependencies {
|
|||
exclude group: 'com.google.android.gms'
|
||||
}
|
||||
implementation 'com.android.support.constraint:constraint-layout:1.1.0'
|
||||
implementation 'com.ms-square:expandableTextView:0.1.4'
|
||||
}
|
||||
|
||||
def getDate() {
|
||||
|
|
|
@ -685,7 +685,6 @@ Java_com_mapswithme_maps_bookmarks_data_BookmarkManager_nativeGetBookmarkCategor
|
|||
auto const bookmarksCount = manager.GetUserMarkIds(data.m_id).size();
|
||||
auto const isVisible = manager.IsVisible(data.m_id);
|
||||
|
||||
|
||||
return env->NewObject(g_bookmarkCategoryClass,
|
||||
g_bookmarkCategoryConstructor,
|
||||
static_cast<jlong>(data.m_id),
|
||||
|
|
|
@ -1,6 +1,28 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<android.support.v7.widget.RecyclerView
|
||||
<android.support.constraint.ConstraintLayout
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:id="@+id/recycler"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="match_parent"/>
|
||||
android:layout_height="match_parent"
|
||||
xmlns:mapsme="http://schemas.android.com/apk/res-auto">
|
||||
<android.support.v7.widget.RecyclerView
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:id="@+id/recycler"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="match_parent"/>
|
||||
<com.mapswithme.maps.widget.PlaceholderView
|
||||
android:id="@+id/placeholder"
|
||||
mapsme:layout_constraintBottom_toBottomOf="parent"
|
||||
mapsme:layout_constraintTop_toTopOf="parent"
|
||||
mapsme:layout_constraintLeft_toLeftOf="parent"
|
||||
mapsme:layout_constraintRight_toRightOf="parent"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="match_parent"
|
||||
android:paddingLeft="@dimen/margin_double_and_half"
|
||||
android:paddingRight="@dimen/margin_double_and_half"
|
||||
android:paddingTop="@dimen/placeholder_margin_top"
|
||||
mapsme:imgSrcDefault="@drawable/img_empty_bookmarks"
|
||||
mapsme:titleDefault="@string/empty_bookmarks_list_title"
|
||||
mapsme:subTitleDefault="@string/empty_bookmarks_list_subtitle"/>
|
||||
|
||||
</android.support.constraint.ConstraintLayout>
|
||||
|
||||
|
|
|
@ -17,30 +17,26 @@
|
|||
android:textAppearance="@style/MwmTextAppearance.Body3"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"/>
|
||||
|
||||
<com.mapswithme.maps.bookmarks.ExpandableTextView
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:expandableTextView="http://schemas.android.com/apk/res-auto"
|
||||
<LinearLayout
|
||||
android:orientation="vertical"
|
||||
app:layout_constraintTop_toBottomOf="@id/author"
|
||||
android:id="@+id/description"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
expandableTextView:maxCollapsedLines="4"
|
||||
expandableTextView:animDuration="200">
|
||||
android:layout_height="wrap_content">
|
||||
<TextView
|
||||
android:id="@+id/expandable_text"
|
||||
app:layout_constraintTop_toBottomOf="@id/author"
|
||||
android:id="@+id/text"
|
||||
android:textAppearance="@style/MwmTextAppearance.Body3"
|
||||
android:lines="3"
|
||||
android:ellipsize="end"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"/>
|
||||
<TextView
|
||||
android:id="@id/expand_collapse"
|
||||
android:id="@+id/more_btn"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:layout_gravity="left|bottom"
|
||||
android:gravity="left|top"
|
||||
android:textColor="?attr/colorAccent"
|
||||
android:text="...more"
|
||||
android:text="@string/more"
|
||||
android:background="@android:color/transparent"/>
|
||||
</com.mapswithme.maps.bookmarks.ExpandableTextView>
|
||||
</LinearLayout>
|
||||
</android.support.constraint.ConstraintLayout>
|
||||
|
|
|
@ -1,15 +1,14 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<TextView
|
||||
xmlns:tools="http://schemas.android.com/tools"
|
||||
android:id="@+id/text"
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:tools="http://schemas.android.com/tools"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="@dimen/height_item_oneline"
|
||||
android:gravity="left|center_vertical"
|
||||
android:paddingLeft="@dimen/margin_base"
|
||||
android:paddingTop="@dimen/margin_half"
|
||||
android:background="@color/bookmarks_list_item_bg"
|
||||
android:fontFamily="@string/robotoMedium"
|
||||
tools:targetApi="JELLY_BEAN"
|
||||
android:textAppearance="@style/MwmTextAppearance.Body3"
|
||||
android:text="@string/bookmarks"/>
|
||||
android:fontFamily="@string/robotoMedium"
|
||||
android:text="@string/bookmarks"
|
||||
tools:targetApi="jelly_bean"/>
|
||||
|
|
|
@ -1398,4 +1398,5 @@
|
|||
<string name="button_layer_got_it">Got it</string>
|
||||
<string name="bookmarks_empty_list_title">This list is empty</string>
|
||||
<string name="bookmarks_empty_list_message">To add a bookmark, tap a place on the map and then tap the star icon</string>
|
||||
<string name="more">…more</string>
|
||||
</resources>
|
||||
|
|
|
@ -1 +1 @@
|
|||
include '3rd_party:BottomSheet'
|
||||
include '3rd_party:BottomSheet'
|
||||
|
|
|
@ -153,7 +153,7 @@ public class BookmarkListAdapter extends RecyclerView.Adapter<Holders.BaseBookma
|
|||
public Object getItem(int position)
|
||||
{
|
||||
if (getItemViewType(position) == TYPE_DESC)
|
||||
throw new UnsupportedOperationException("Not supported here!");
|
||||
throw new UnsupportedOperationException("Not supported here! Position = " + position);
|
||||
|
||||
if (getItemViewType(position) == TYPE_TRACK)
|
||||
{
|
||||
|
|
|
@ -28,6 +28,7 @@ import com.mapswithme.maps.widget.placepage.Sponsored;
|
|||
import com.mapswithme.maps.widget.recycler.RecyclerClickListener;
|
||||
import com.mapswithme.maps.widget.recycler.RecyclerLongClickListener;
|
||||
import com.mapswithme.util.BottomSheetHelper;
|
||||
import com.mapswithme.util.Utils;
|
||||
import com.mapswithme.util.sharing.ShareOption;
|
||||
import com.mapswithme.util.sharing.SharingHelper;
|
||||
|
||||
|
@ -69,6 +70,9 @@ public class BookmarksListFragment extends BaseMwmRecyclerFragment
|
|||
super.onViewCreated(view, savedInstanceState);
|
||||
configureAdapter();
|
||||
setHasOptionsMenu(true);
|
||||
boolean isEmptyDataSet = getAdapter().getItemCount() == 0;
|
||||
getRecyclerView().setVisibility(isEmptyDataSet ? View.GONE : View.VISIBLE);
|
||||
showPlaceholder(isEmptyDataSet);
|
||||
ActionBar bar = ((AppCompatActivity) getActivity()).getSupportActionBar();
|
||||
if (bar != null)
|
||||
bar.setTitle(mCategory.getName());
|
||||
|
|
|
@ -11,6 +11,8 @@ import android.view.ViewGroup;
|
|||
import com.mapswithme.maps.R;
|
||||
import com.mapswithme.maps.bookmarks.data.BookmarkCategory;
|
||||
import com.mapswithme.maps.bookmarks.data.BookmarkManager;
|
||||
import com.mapswithme.util.SharedPropertiesUtils;
|
||||
import com.mapswithme.util.StorageUtils;
|
||||
import com.mapswithme.util.UiUtils;
|
||||
import com.mapswithme.util.sharing.TargetUtils;
|
||||
|
||||
|
@ -35,7 +37,8 @@ public class CachedBookmarkCategoriesFragment extends BaseBookmarkCategoriesFrag
|
|||
downloadBtn.setOnClickListener(new DownloadRoutesClickListener());
|
||||
View closeHeaderBtn = root.findViewById(R.id.header_close);
|
||||
closeHeaderBtn.setOnClickListener(new CloseHeaderClickListener());
|
||||
|
||||
boolean isClosed = SharedPropertiesUtils.isCatalogCategoriesHeaderClosed(getContext());
|
||||
closeHeaderBtn.setVisibility(isClosed ? View.GONE : View.VISIBLE);
|
||||
return root;
|
||||
}
|
||||
|
||||
|
@ -168,6 +171,7 @@ public class CachedBookmarkCategoriesFragment extends BaseBookmarkCategoriesFrag
|
|||
{
|
||||
View header = mPayloadContainer.findViewById(R.id.header);
|
||||
header.setVisibility(View.GONE);
|
||||
SharedPropertiesUtils.setCatalogCategoriesHeaderClosed(getContext(), true);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -1,381 +0,0 @@
|
|||
/*
|
||||
* Copyright (C) 2011 The Android Open Source Project
|
||||
* Copyright 2014 Manabu Shimobe
|
||||
*
|
||||
* 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.mapswithme.maps.bookmarks;
|
||||
|
||||
import android.annotation.TargetApi;
|
||||
import android.content.Context;
|
||||
import android.content.res.Resources;
|
||||
import android.graphics.drawable.Drawable;
|
||||
import android.os.Build;
|
||||
import android.support.annotation.DrawableRes;
|
||||
import android.support.annotation.NonNull;
|
||||
import android.support.annotation.Nullable;
|
||||
import android.text.TextUtils;
|
||||
import android.util.AttributeSet;
|
||||
import android.util.SparseBooleanArray;
|
||||
import android.view.MotionEvent;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.view.animation.AlphaAnimation;
|
||||
import android.view.animation.Animation;
|
||||
import android.view.animation.Transformation;
|
||||
import android.widget.LinearLayout;
|
||||
import android.widget.TextView;
|
||||
|
||||
import com.mapswithme.maps.R;
|
||||
|
||||
public class ExpandableTextView extends LinearLayout implements View.OnClickListener {
|
||||
|
||||
private static final String TAG = ExpandableTextView.class.getSimpleName();
|
||||
|
||||
/* The default number of lines */
|
||||
private static final int MAX_COLLAPSED_LINES = 4;
|
||||
|
||||
/* 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 TextView mTv;
|
||||
|
||||
protected View mButton; // 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 Drawable mExpandDrawable;
|
||||
|
||||
private Drawable mCollapseDrawable;
|
||||
|
||||
private int mAnimationDuration;
|
||||
|
||||
private float mAnimAlphaStart;
|
||||
|
||||
private boolean mAnimating;
|
||||
|
||||
/* Listener for callback */
|
||||
private OnExpandStateChangeListener mListener;
|
||||
|
||||
/* For saving collapsed status when used in ListView */
|
||||
private SparseBooleanArray mCollapsedStatus;
|
||||
private int mPosition;
|
||||
@Nullable
|
||||
private OnClickListener mBtnClickListener;
|
||||
private boolean mForceCollapsingRestricted;
|
||||
|
||||
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 (mButton.getVisibility() != View.VISIBLE) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (mBtnClickListener != null) {
|
||||
mBtnClickListener.onClick(view);
|
||||
mMarginBetweenTxtAndBottom = getHeight() - getRealTextViewHeight(mTv);
|
||||
mTextHeightWithMaxLines = getRealTextViewHeight(mTv);
|
||||
}
|
||||
|
||||
mForceCollapsingRestricted = true;
|
||||
mCollapsed = !mCollapsed;
|
||||
// mButton.setImageDrawable(mCollapsed ? mExpandDrawable : mCollapseDrawable);
|
||||
|
||||
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);
|
||||
}
|
||||
mButton.setVisibility(mCollapsed ? VISIBLE : GONE);
|
||||
}
|
||||
@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
|
||||
mButton.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 && mForceCollapsingRestricted) {
|
||||
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);
|
||||
}
|
||||
mButton.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 setOnButtonClickListener(@NonNull OnClickListener clickListener) {
|
||||
mBtnClickListener = clickListener;
|
||||
}
|
||||
|
||||
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;
|
||||
// mButton.setImageDrawable(mCollapsed ? mExpandDrawable : mCollapseDrawable);
|
||||
setText(text);
|
||||
getLayoutParams().height = ViewGroup.LayoutParams.WRAP_CONTENT;
|
||||
requestLayout();
|
||||
}
|
||||
|
||||
@Nullable
|
||||
public CharSequence getText() {
|
||||
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/*)*/;
|
||||
mForceCollapsingRestricted = false;
|
||||
/* mExpandDrawable = typedArray.getDrawable(R.styleable.ExpandableTextView_expandDrawable*//*)*//*;
|
||||
mCollapseDrawable = typedArray.getDrawable(R.styleable.ExpandableTextView_collapseDrawable);*/
|
||||
|
||||
if (mExpandDrawable == null) {
|
||||
mExpandDrawable = getDrawable(getContext(), R.drawable.ic_expand_small_holo_light);
|
||||
}
|
||||
if (mCollapseDrawable == null) {
|
||||
mCollapseDrawable = getDrawable(getContext(), R.drawable.ic_collapse_small_holo_light);
|
||||
}
|
||||
|
||||
// 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);
|
||||
mButton = findViewById(R.id.expand_collapse);
|
||||
// mButton.setImageDrawable(mCollapsed ? mExpandDrawable : mCollapseDrawable);
|
||||
mButton.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 Drawable getDrawable(@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 ExpandCollapseAnimation extends Animation {
|
||||
private final View mTargetView;
|
||||
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);
|
||||
}
|
||||
}
|
|
@ -8,6 +8,8 @@ import android.support.annotation.NonNull;
|
|||
import android.support.annotation.Nullable;
|
||||
import android.support.annotation.PluralsRes;
|
||||
import android.support.v7.widget.RecyclerView;
|
||||
import android.text.Layout;
|
||||
import android.text.StaticLayout;
|
||||
import android.text.TextUtils;
|
||||
import android.view.View;
|
||||
import android.widget.CheckBox;
|
||||
|
@ -268,9 +270,8 @@ public class Holders
|
|||
return -1;
|
||||
|
||||
int beforeCurrentSectionItemsCount = getTracksSectionPosition(category);
|
||||
return (beforeCurrentSectionItemsCount == -1
|
||||
? getDescItemCount(category)
|
||||
: beforeCurrentSectionItemsCount)
|
||||
return (beforeCurrentSectionItemsCount == -1 ? getDescItemCount(category)
|
||||
: beforeCurrentSectionItemsCount)
|
||||
+ getTrackItemCount(category);
|
||||
}
|
||||
|
||||
|
@ -370,9 +371,9 @@ public class Holders
|
|||
@NonNull
|
||||
private final TextView mDistance;
|
||||
|
||||
TrackViewHolder(@NonNull View itemView, BookmarkCategory categoryId)
|
||||
TrackViewHolder(@NonNull View itemView, @NonNull BookmarkCategory category)
|
||||
{
|
||||
super(itemView, categoryId);
|
||||
super(itemView, category);
|
||||
mIcon = itemView.findViewById(R.id.iv__bookmark_color);
|
||||
mName = itemView.findViewById(R.id.tv__bookmark_name);
|
||||
mDistance = itemView.findViewById(R.id.tv__bookmark_distance);
|
||||
|
@ -401,9 +402,9 @@ public class Holders
|
|||
@NonNull
|
||||
private final TextView mView;
|
||||
|
||||
SectionViewHolder(@NonNull TextView itemView, BookmarkCategory categoryId)
|
||||
SectionViewHolder(@NonNull TextView itemView, @NonNull BookmarkCategory category)
|
||||
{
|
||||
super(itemView, categoryId);
|
||||
super(itemView, category);
|
||||
mView = itemView;
|
||||
}
|
||||
|
||||
|
@ -412,7 +413,6 @@ public class Holders
|
|||
{
|
||||
final int sectionIndex = getSectionForPosition(mCategory, position);
|
||||
mView.setText(getSections().get(sectionIndex));
|
||||
mView.setText(getSections().get(sectionIndex));
|
||||
}
|
||||
|
||||
private List<String> getSections()
|
||||
|
@ -427,34 +427,56 @@ public class Holders
|
|||
|
||||
static class DescriptionViewHolder extends BaseBookmarkHolder
|
||||
{
|
||||
@NonNull
|
||||
private final ExpandableTextView mContentView;
|
||||
static final float SPACING_MULTIPLE = 1.0f;
|
||||
static final float SPACING_ADD = 0.0f;
|
||||
@NonNull
|
||||
private final TextView mTitle;
|
||||
@NonNull
|
||||
private final TextView mAuthor;
|
||||
@NonNull
|
||||
private final TextView mDescText;
|
||||
@NonNull
|
||||
private final View mMoreBtn;
|
||||
|
||||
DescriptionViewHolder(@NonNull View itemView, @NonNull BookmarkCategory category)
|
||||
{
|
||||
super(itemView, category);
|
||||
mContentView = itemView.findViewById(R.id.description);
|
||||
mContentView.setOnButtonClickListener(new View.OnClickListener()
|
||||
{
|
||||
@Override
|
||||
public void onClick(View v)
|
||||
{
|
||||
mContentView.setText(category.getDescription());
|
||||
}
|
||||
});
|
||||
mDescText = itemView.findViewById(R.id.text);
|
||||
mMoreBtn = itemView.findViewById(R.id.more_btn);
|
||||
mMoreBtn.setOnClickListener(this::onMoreBtnClicked);
|
||||
mTitle = itemView.findViewById(R.id.title);
|
||||
mAuthor = itemView.findViewById(R.id.author);
|
||||
}
|
||||
|
||||
private void onMoreBtnClicked(View v)
|
||||
{
|
||||
int lineCount = calcLineCount(mDescText, mCategory.getDescription());
|
||||
mDescText.setLines(lineCount);
|
||||
mDescText.setText(mCategory.getDescription());
|
||||
v.setVisibility(View.GONE);
|
||||
}
|
||||
|
||||
@Override
|
||||
void bind(int position)
|
||||
{
|
||||
mTitle.setText(mCategory.getName());
|
||||
mContentView.setText(mCategory.getAnnotation());
|
||||
bindAuthor();
|
||||
bindDescriptionIfEmpty();
|
||||
}
|
||||
|
||||
private void bindDescriptionIfEmpty()
|
||||
{
|
||||
if (TextUtils.isEmpty(mDescText.getText()))
|
||||
{
|
||||
String desc = TextUtils.isEmpty(mCategory.getAnnotation())
|
||||
? mCategory.getDescription()
|
||||
: mCategory.getAnnotation();
|
||||
mDescText.setText(desc);
|
||||
}
|
||||
}
|
||||
|
||||
private void bindAuthor()
|
||||
{
|
||||
BookmarkCategory.Author author = mCategory.getAuthor();
|
||||
Context c = itemView.getContext();
|
||||
CharSequence authorName = author == null
|
||||
|
@ -462,5 +484,18 @@ public class Holders
|
|||
: BookmarkCategory.Author.getRepresentation(c, author);
|
||||
mAuthor.setText(authorName);
|
||||
}
|
||||
|
||||
private static int calcLineCount(@NonNull TextView textView, @NonNull String src)
|
||||
{
|
||||
StaticLayout staticLayout = new StaticLayout(src,
|
||||
textView.getPaint(),
|
||||
textView.getWidth(),
|
||||
Layout.Alignment.ALIGN_NORMAL,
|
||||
SPACING_MULTIPLE,
|
||||
SPACING_ADD,
|
||||
true);
|
||||
|
||||
return staticLayout.getLineCount();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -15,92 +15,6 @@ import com.mapswithme.util.TypeConverter;
|
|||
|
||||
public class BookmarkCategory implements Parcelable
|
||||
{
|
||||
public static final String DESCRIPTION = "\n" +
|
||||
"Владимир Семёнович Высоцкий\n" +
|
||||
"Vladimir Vysotsky.jpg\n" +
|
||||
"Владимир Высоцкий на концерте 23 апреля 1979 года. Фото Игоря Пальмина\n" +
|
||||
"Дата рождения:\t25 января 1938\n" +
|
||||
"Место рождения:\tМосква, СССР\n" +
|
||||
"Дата смерти:\t25 июля 1980 (42 года)\n" +
|
||||
"Место смерти:\tМосква, СССР\n" +
|
||||
"Страна:\t\n" +
|
||||
"Flag of the Soviet Union (1955–1980).svg СССР\n" +
|
||||
"Род деятельности:\t\n" +
|
||||
"поэт, поэт-песенник, автор-исполнитель, прозаик, актёр театра и кино, певец, " +
|
||||
"композитор, гитарист\n" +
|
||||
"Место работы:\t\n" +
|
||||
"Московский драматический театр имени А. С. Пушкина\n" +
|
||||
"Театр на Таганке\n" +
|
||||
"Высшее образование:\t\n" +
|
||||
"Школа-студия МХАТ\n" +
|
||||
"Награды и премии:\tГосударственная премия СССР — 1987 Premia mvd.jpg[1][2]\n" +
|
||||
"Подпись:\tПодпись\n" +
|
||||
"Отец:\tСемён Владимирович Высоцкий\n" +
|
||||
"Мать:\tНина Максимовна Высоцкая\n" +
|
||||
"Супруг(а):\t\n" +
|
||||
"• Изольда Жукова (Мешкова)[⇨], \n" +
|
||||
"• Людмила Абрамова[⇨], \n" +
|
||||
"• Марина Влади[⇨]\n" +
|
||||
"Дети:\tАркадий Высоцкий, Никита Высоцкий[⇨]\n" +
|
||||
"Похоронен\t\n" +
|
||||
"Ваганьковское кладбище\n" +
|
||||
"IMDb:\tID 0904584\n" +
|
||||
"Логотип Викицитатника Цитаты в Викицитатнике\n" +
|
||||
"Commons-logo.svg Владимир Семёнович Высоцкий на Викискладе\n" +
|
||||
"Влади́мир Семёнович Высо́цкий (25 января 1938, Москва — 25 июля 1980, Москва)" +
|
||||
" — советский поэт, актёр театра и кино, автор-исполнитель песен (бард); автор" +
|
||||
" прозаических произведений и сценариев. Лауреат Государственной премии СССР " +
|
||||
"(«за создание образа Жеглова в телевизионном художественном фильме „Место " +
|
||||
"встречи изменить нельзя“ и авторское исполнение песен», 1987, посмертно).\n" +
|
||||
"\n" +
|
||||
"Как поэт Высоцкий реализовал себя прежде всего в жанре авторской песни. " +
|
||||
"Первые из написанных им произведений относятся к началу 1960-х годов. " +
|
||||
"Изначально они исполнялись в кругу друзей; позже получили широкую известность" +
|
||||
" благодаря распространявшимся по стране магнитофонным записям. Поэзия " +
|
||||
"Высоцкого отличалась многообразием тем (уличные, лагерные, военные, " +
|
||||
"сатирические, бытовые, сказочные, «спортивные» песни), остротой смыслового " +
|
||||
"подтекста и акцентированной социально-нравственной позицией автора. В его " +
|
||||
"произведениях, рассказывающих о внутреннем выборе людей, поставленных в " +
|
||||
"экстремальные обстоятельства, прослеживались экзистенциальные мотивы. " +
|
||||
"Творческая эволюция Высоцкого ознаменовалась несколькими этапами. В его " +
|
||||
"раннем творчестве преобладали уличные и дворовые песни. С середины 1960-х " +
|
||||
"годов тематика произведений начала расширяться, а песенные циклы складываться" +
|
||||
" в новую «энциклопедию русской жизни». В 1970-х годах значительную часть " +
|
||||
"творчества Высоцкого составляли песни и стихотворения " +
|
||||
"исповедально-философского характера, поэт часто обращался к вечным вопросам " +
|
||||
"бытия.\n" +
|
||||
"\n" +
|
||||
"Театральная биография Высоцкого, окончившего в 1960 году Школу-студию МХАТ, " +
|
||||
"связана главным образом с работой в Театре на Таганке. На его сцене актёр " +
|
||||
"играл Галилея (спектакль «Жизнь Галилея», 1966), Хлопушу («Пугачёв», 1967), " +
|
||||
"Гамлета («Гамлет», 1971), Лопахина («Вишнёвый сад», 1975), Свидригайлова " +
|
||||
"(«Преступление и наказание», 1979). Дебют Высоцкого в кино состоялся в 1959 " +
|
||||
"году, когда он сыграл эпизодическую роль в фильме «Сверстницы». За годы " +
|
||||
"работы в кинематографе актёр снялся более чем в двадцати пяти фильмах. " +
|
||||
"Кинобиография Высоцкого включает роли подпольщика Бродского («Интервенция», " +
|
||||
"1968), зоолога фон Корена («Плохой хороший человек», 1973), капитана Жеглова " +
|
||||
"(«Место встречи изменить нельзя», 1979), Дон Гуана («Маленькие трагедии», " +
|
||||
"1979) и другие. Исследователи отмечали, что в сценических и экранных работах " +
|
||||
"Высоцкого экспрессивность сочеталась с психологической достоверностью. В ряде" +
|
||||
" спектаклей, а также в художественных фильмах, теле- и радиопостановках он " +
|
||||
"выступал и как автор песен.\n" +
|
||||
"\n" +
|
||||
"При жизни Высоцкого его песни не получили в СССР официального признания. В " +
|
||||
"1968 году в рамках газетной кампании, дискредитирующей его " +
|
||||
"музыкально-поэтическое творчество, они были подвергнуты резкой критике. " +
|
||||
"Вплоть до 1981 года ни одно советское издательство не выпустило книгу с его " +
|
||||
"текстами. Цензурные ограничения частично были сняты только после смерти " +
|
||||
"Высоцкого, когда вышел в свет сборник его поэтических произведений «Нерв» " +
|
||||
"(составитель — Роберт Рождественский). Тем не менее цензорский контроль за " +
|
||||
"публикациями стихов и песен Высоцкого, а также посвящённых ему " +
|
||||
"газетно-журнальных статей продолжал действовать вплоть до перестройки. " +
|
||||
"Легализация его творчества началась в Советском Союзе в 1986 году, когда при " +
|
||||
"Союзе писателей СССР была создана комиссия по литературному наследию " +
|
||||
"Высоцкого. Со второй половины 1980-х годов начался выпуск книг и собраний " +
|
||||
"сочинений поэта, ведётся исследовательская работа, посвящённая его творчеству" +
|
||||
". По некоторым оценкам, Высоцкий, занимающий одно из центральных мест в " +
|
||||
"истории русской культуры XX века, «оказал сильное влияние на формирование " +
|
||||
"взглядов своих современников и последующих поколений».";
|
||||
private final long mId;
|
||||
@NonNull
|
||||
private final String mName;
|
||||
|
@ -122,8 +36,8 @@ public class BookmarkCategory implements Parcelable
|
|||
{
|
||||
mId = id;
|
||||
mName = name;
|
||||
mAnnotation = "asdasdasdasdasdasdasdasdasdasdasdasd";
|
||||
mDescription = DESCRIPTION;
|
||||
mAnnotation = annotation;
|
||||
mDescription = description;
|
||||
mTracksCount = tracksCount;
|
||||
mBookmarksCount = bookmarksCount;
|
||||
mTypeIndex = fromCatalog ? Type.CATALOG.ordinal() : Type.PRIVATE.ordinal();
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
package com.mapswithme.util;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.SharedPreferences;
|
||||
import android.preference.PreferenceManager;
|
||||
import android.support.annotation.NonNull;
|
||||
|
@ -15,9 +16,16 @@ public final class SharedPropertiesUtils
|
|||
private static final String PREFS_SHOW_EMULATE_BAD_STORAGE_SETTING = "ShowEmulateBadStorageSetting";
|
||||
private static final String PREFS_BACKUP_WIDGET_EXPANDED = "BackupWidgetExpanded";
|
||||
private static final String PREFS_WHATS_NEW_TITLE_CONCATENATION = "WhatsNewTitleConcatenation";
|
||||
private static final String PREFS_CATALOG_CATEGORIES_HEADER_CLOSED = "catalog_categories_header_closed";
|
||||
private static final SharedPreferences PREFS
|
||||
= PreferenceManager.getDefaultSharedPreferences(MwmApplication.get());
|
||||
|
||||
//Utils class
|
||||
private SharedPropertiesUtils()
|
||||
{
|
||||
throw new IllegalStateException("Try instantiate utility class SharedPropertiesUtils");
|
||||
}
|
||||
|
||||
public static boolean isShowcaseSwitchedOnLocal()
|
||||
{
|
||||
return PreferenceManager.getDefaultSharedPreferences(MwmApplication.get())
|
||||
|
@ -71,9 +79,17 @@ public final class SharedPropertiesUtils
|
|||
PREFS.edit().putString(PREFS_WHATS_NEW_TITLE_CONCATENATION, concatenation).apply();
|
||||
}
|
||||
|
||||
//Utils class
|
||||
private SharedPropertiesUtils()
|
||||
public static boolean isCatalogCategoriesHeaderClosed(Context context)
|
||||
{
|
||||
throw new IllegalStateException("Try instantiate utility class SharedPropertiesUtils");
|
||||
return MwmApplication.prefs(context)
|
||||
.getBoolean(PREFS_CATALOG_CATEGORIES_HEADER_CLOSED, false);
|
||||
}
|
||||
|
||||
public static void setCatalogCategoriesHeaderClosed(Context context, boolean value)
|
||||
{
|
||||
MwmApplication.prefs(context)
|
||||
.edit()
|
||||
.putBoolean(PREFS_CATALOG_CATEGORIES_HEADER_CLOSED, value)
|
||||
.apply();
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue