-
-
Save asbadve/6d2ed4e7522685b44f81 to your computer and use it in GitHub Desktop.
Used for the sliding tab for lollipop
This file contains hidden or 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
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" | |
xmlns:tools="http://schemas.android.com/tools" | |
android:layout_width="match_parent" | |
android:layout_height="match_parent" | |
android:orientation="vertical" | |
tools:context=".MainActivity"> | |
<android.support.v7.widget.Toolbar | |
android:id="@+id/toolbar" | |
android:layout_height="wrap_content" | |
android:layout_width="match_parent" | |
android:minHeight="?attr/actionBarSize" | |
android:background="#00BCD4" /> | |
<LinearLayout | |
android:layout_width="match_parent" | |
android:layout_height="wrap_content"> | |
<com.arkony.flatuisample.SlidingTabLayout | |
android:id="@+id/sliding_tabs" | |
android:layout_width="match_parent" | |
android:layout_height="wrap_content" | |
android:background="#00BCD4" /> | |
</LinearLayout> | |
<android.support.v4.view.ViewPager | |
android:id="@+id/viewpager" | |
android:layout_width="match_parent" | |
android:layout_height="match_parent" | |
android:layout_weight="1" | |
android:background="@android:color/white"/> | |
</LinearLayout> |
This file contains hidden or 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
import android.graphics.Color; | |
import android.os.Bundle; | |
import android.support.v4.view.PagerAdapter; | |
import android.support.v4.view.ViewPager; | |
import android.support.v7.app.ActionBar; | |
import android.support.v7.app.ActionBarActivity; | |
import android.support.v7.widget.Toolbar; | |
import android.util.Log; | |
import android.view.Menu; | |
import android.view.MenuItem; | |
import android.view.View; | |
import android.view.ViewGroup; | |
import android.widget.TextView; | |
import YOUR.PACKAGE.NAME.SlidingTabLayout; | |
public class MainActivity extends ActionBarActivity implements ActionBar.TabListener { | |
/** | |
* The {@link ViewPager} that will host the section contents. | |
*/ | |
ViewPager mViewPager; | |
static final String LOG_TAG = "SlidingTabsBasicFragment"; | |
private SlidingTabLayout mSlidingTabLayout; | |
@Override | |
protected void onCreate(Bundle savedInstanceState) { | |
super.onCreate(savedInstanceState); | |
setContentView(R.layout.activity_main); | |
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar); | |
setSupportActionBar(toolbar); | |
mViewPager = (ViewPager) findViewById(R.id.viewpager); | |
mViewPager.setAdapter(new SamplePagerAdapter()); | |
mSlidingTabLayout = (SlidingTabLayout) findViewById(R.id.sliding_tabs); | |
mSlidingTabLayout.setTitleColor(Color.WHITE); | |
mSlidingTabLayout.setFittingChildren(true); | |
mSlidingTabLayout.setTabType(SlidingTabLayout.TabType.TEXT); | |
mSlidingTabLayout.setViewPager(mViewPager); | |
} | |
@Override | |
public boolean onCreateOptionsMenu(Menu menu) { | |
// Inflate the menu; this adds items to the action bar if it is present. | |
getMenuInflater().inflate(R.menu.menu_main, menu); | |
return true; | |
} | |
@Override | |
public boolean onOptionsItemSelected(MenuItem item) { | |
// Handle action bar item clicks here. The action bar will | |
// automatically handle clicks on the Home/Up button, so long | |
// as you specify a parent activity in AndroidManifest.xml. | |
int id = item.getItemId(); | |
//noinspection SimplifiableIfStatement | |
if (id == R.id.action_settings) { | |
return true; | |
} | |
return super.onOptionsItemSelected(item); | |
} | |
@Override | |
public void onTabSelected(ActionBar.Tab tab, android.support.v4.app.FragmentTransaction fragmentTransaction) { | |
// When the given tab is selected, switch to the corresponding page in | |
// the ViewPager. | |
mViewPager.setCurrentItem(tab.getPosition()); | |
} | |
@Override | |
public void onTabUnselected(ActionBar.Tab tab, android.support.v4.app.FragmentTransaction fragmentTransaction) { | |
} | |
@Override | |
public void onTabReselected(ActionBar.Tab tab, android.support.v4.app.FragmentTransaction fragmentTransaction) { | |
} | |
class SamplePagerAdapter extends PagerAdapter { | |
/** | |
* @return the number of pages to display | |
*/ | |
@Override | |
public int getCount() { | |
return 4; | |
} | |
/** | |
* @return true if the value returned from | |
* {@link #instantiateItem(ViewGroup, int)} is the same object | |
* as the {@link View} added to the {@link ViewPager}. | |
*/ | |
@Override | |
public boolean isViewFromObject(View view, Object o) { | |
return o == view; | |
} | |
/** | |
* Return the title of the item at {@code position}. This is important | |
* as what this method returns is what is displayed in the | |
* {@link SlidingTabLayout}. | |
* <p/> | |
* Here we construct one using the position value, but for real | |
* application the title should refer to the item's contents. | |
*/ | |
@Override | |
public CharSequence getPageTitle(int position) { | |
return "Item " + (position + 1); | |
} | |
/** | |
* Instantiate the {@link View} which should be displayed at | |
* {@code position}. Here we inflate a layout from the apps resources | |
* and then change the text view to signify the position. | |
*/ | |
@Override | |
public Object instantiateItem(ViewGroup container, int position) { | |
// Inflate a new layout from our resources | |
View view = getLayoutInflater().inflate(R.layout.pager_item, container, false); | |
// Add the newly created View to the ViewPager | |
container.addView(view); | |
// Retrieve a TextView from the inflated View, and update it's text | |
TextView title = (TextView) view.findViewById(R.id.item_title); | |
title.setText(String.valueOf(position + 1)); | |
Log.i(LOG_TAG, "instantiateItem() [position: " + position + "]"); | |
// Return the View | |
return view; | |
} | |
/** | |
* Destroy the item from the {@link ViewPager}. In our case this is | |
* simply removing the {@link View}. | |
*/ | |
@Override | |
public void destroyItem(ViewGroup container, int position, Object object) { | |
container.removeView((View) object); | |
Log.i(LOG_TAG, "destroyItem() [position: " + position + "]"); | |
} | |
} | |
} |
This file contains hidden or 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
import android.content.Context; | |
import android.graphics.Typeface; | |
import android.os.Build; | |
import android.support.v4.view.PagerAdapter; | |
import android.support.v4.view.ViewPager; | |
import android.util.AttributeSet; | |
import android.util.TypedValue; | |
import android.view.Gravity; | |
import android.view.LayoutInflater; | |
import android.view.View; | |
import android.widget.HorizontalScrollView; | |
import android.widget.ImageView; | |
import android.widget.TextView; | |
/** | |
* To be used with ViewPager to provide a tab indicator component which give constant feedback as to | |
* the user's scroll progress. | |
* <p/> | |
* To use the component, simply add it to your view hierarchy. Then in your | |
* {@link android.app.Activity} or {@link android.support.v4.app.Fragment} call | |
* {@link #setViewPager(ViewPager)} providing it the ViewPager this layout is being used for. | |
* <p/> | |
* The colors can be customized in two ways. The first and simplest is to provide an array of colors | |
* via {@link #setSelectedIndicatorColors(int...)} and {@link #setDividerColors(int...)}. The | |
* alternative is via the {@link TabColorizer} interface which provides you complete control over | |
* which color is used for any individual position. | |
* <p/> | |
* The views used as tabs can be customized by calling {@link #setCustomTabView(int, int, int)}, | |
* providing the layout ID of your custom layout. | |
*/ | |
public class SlidingTabLayout extends HorizontalScrollView { | |
/** | |
* Allows complete control over the colors drawn in the tab layout. Set with | |
* {@link #setCustomTabColorizer(TabColorizer)}. | |
*/ | |
public interface TabColorizer { | |
/** | |
* @return return the color of the indicator used when {@code position} is selected. | |
*/ | |
int getIndicatorColor(int position); | |
/** | |
* @return return the color of the divider drawn to the right of {@code position}. | |
*/ | |
int getDividerColor(int position); | |
} | |
private static final int TITLE_OFFSET_DIPS = 24; | |
private static final int TAB_VIEW_PADDING_DIPS = 16; | |
private static final int TAB_VIEW_TEXT_SIZE_SP = 12; | |
private TabType tabType = TabType.TEXT; | |
private int mTitleOffset; | |
private int mTabViewLayoutId; | |
private int mTabViewTextViewId; | |
private int mTabViewImageViewId; | |
private Integer titleColor = -1; | |
private ViewPager mViewPager; | |
private ViewPager.OnPageChangeListener mViewPagerPageChangeListener; | |
private final SlidingTabStrip mTabStrip; | |
public SlidingTabLayout(Context context) { | |
this(context, null); | |
} | |
public SlidingTabLayout(Context context, AttributeSet attrs) { | |
this(context, attrs, 0); | |
} | |
public SlidingTabLayout(Context context, AttributeSet attrs, int defStyle) { | |
super(context, attrs, defStyle); | |
// Disable the Scroll Bar | |
setHorizontalScrollBarEnabled(false); | |
// Make sure that the Tab Strips fills this View | |
setFillViewport(true); | |
mTitleOffset = (int) (TITLE_OFFSET_DIPS * getResources().getDisplayMetrics().density); | |
mTabStrip = new SlidingTabStrip(context); | |
addView(mTabStrip, LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT); | |
} | |
/** | |
* Sets the title color showed in tabs. | |
* | |
* @param titleColor | |
*/ | |
public void setTitleColor(int titleColor) { | |
this.titleColor = titleColor; | |
} | |
/** | |
* Sets isFittingChildren of the tab layout. If isFittingChildren is true, then the children will | |
* share the total width of the tab layout. Otherwise the children are placed as default. | |
* | |
* @param isFittingChildren | |
*/ | |
public void setFittingChildren(boolean isFittingChildren) { | |
mTabStrip.setFittingChildren(isFittingChildren); | |
} | |
/** | |
* Sets the tab type of the tab layout. Type TEXT shows title as text, ICON shows icon that | |
* is found with the title name as suffix(ic_TITLE_THAT_IS_PROVIDED). | |
* | |
* @param tabType type as enum. TEXT or ICON | |
*/ | |
public void setTabType(TabType tabType) { | |
this.tabType = tabType; | |
} | |
/** | |
* Set the custom {@link TabColorizer} to be used. | |
* <p/> | |
* If you only require simple custmisation then you can use | |
* {@link #setSelectedIndicatorColors(int...)} and {@link #setDividerColors(int...)} to achieve | |
* similar effects. | |
*/ | |
public void setCustomTabColorizer(TabColorizer tabColorizer) { | |
mTabStrip.setCustomTabColorizer(tabColorizer); | |
} | |
/** | |
* Sets the colors to be used for indicating the selected tab. These colors are treated as a | |
* circular array. Providing one color will mean that all tabs are indicated with the same color. | |
*/ | |
public void setSelectedIndicatorColors(int... colors) { | |
mTabStrip.setSelectedIndicatorColors(colors); | |
} | |
/** | |
* Sets the colors to be used for tab dividers. These colors are treated as a circular array. | |
* Providing one color will mean that all tabs are indicated with the same color. | |
*/ | |
public void setDividerColors(int... colors) { | |
mTabStrip.setDividerColors(colors); | |
} | |
/** | |
* Set the {@link ViewPager.OnPageChangeListener}. When using {@link SlidingTabLayout} you are | |
* required to set any {@link ViewPager.OnPageChangeListener} through this method. This is so | |
* that the layout can update it's scroll position correctly. | |
* | |
* @see ViewPager#setOnPageChangeListener(ViewPager.OnPageChangeListener) | |
*/ | |
public void setOnPageChangeListener(ViewPager.OnPageChangeListener listener) { | |
mViewPagerPageChangeListener = listener; | |
} | |
/** | |
* Set the custom layout to be inflated for the tab views. | |
* | |
* @param layoutResId Layout id to be inflated | |
* @param textViewId id of the {@link TextView} in the inflated view | |
*/ | |
public void setCustomTabView(int layoutResId, int textViewId, int imageViewId) { | |
mTabViewLayoutId = layoutResId; | |
mTabViewTextViewId = textViewId; | |
mTabViewImageViewId = imageViewId; | |
} | |
/** | |
* Sets the associated view pager. Note that the assumption here is that the pager content | |
* (number of tabs and tab titles) does not change after this call has been made. | |
*/ | |
public void setViewPager(ViewPager viewPager) { | |
mTabStrip.removeAllViews(); | |
mViewPager = viewPager; | |
if (viewPager != null) { | |
viewPager.setOnPageChangeListener(new InternalViewPagerListener()); | |
populateTabStrip(); | |
} | |
} | |
/** | |
* Create a default view to be used for tabs. This is called if a custom tab view is not set via | |
* {@link #setCustomTabView(int, int, int)}. | |
*/ | |
protected ImageView createDefaultIconTabView(Context context) { | |
ImageView tabIconImageView = new ImageView(context); | |
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) { | |
// If we're running on Honeycomb or newer, then we can use the Theme's | |
// selectableItemBackground to ensure that the View has a pressed state | |
TypedValue outValue = new TypedValue(); | |
getContext().getTheme().resolveAttribute(android.R.attr.selectableItemBackground, | |
outValue, true); | |
tabIconImageView.setBackgroundResource(outValue.resourceId); | |
} | |
int padding = (int) (TAB_VIEW_PADDING_DIPS * getResources().getDisplayMetrics().density); | |
tabIconImageView.setPadding(padding, padding, padding, padding); | |
return tabIconImageView; | |
} | |
/** | |
* Create a default view to be used for tabs. This is called if a custom tab view is not set via | |
* {@link #setCustomTabView(int, int, int)}. | |
*/ | |
protected TextView createDefaultTextTabView(Context context) { | |
TextView textView = new TextView(context); | |
textView.setGravity(Gravity.CENTER); | |
textView.setTextSize(TypedValue.COMPLEX_UNIT_SP, TAB_VIEW_TEXT_SIZE_SP); | |
textView.setTypeface(Typeface.DEFAULT_BOLD); | |
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) { | |
// If we're running on Honeycomb or newer, then we can use the Theme's | |
// selectableItemBackground to ensure that the View has a pressed state | |
TypedValue outValue = new TypedValue(); | |
getContext().getTheme().resolveAttribute(android.R.attr.selectableItemBackground, | |
outValue, true); | |
textView.setBackgroundResource(outValue.resourceId); | |
} | |
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) { | |
// If we're running on ICS or newer, enable all-caps to match the Action Bar tab style | |
textView.setAllCaps(true); | |
} | |
int padding = (int) (TAB_VIEW_PADDING_DIPS * getResources().getDisplayMetrics().density); | |
textView.setPadding(padding, padding, padding, padding); | |
System.out.println("c " + titleColor); | |
if (titleColor != null) { | |
textView.setTextColor(titleColor); | |
} | |
return textView; | |
} | |
private void populateTabStrip() { | |
final PagerAdapter adapter = mViewPager.getAdapter(); | |
final View.OnClickListener tabClickListener = new TabClickListener(); | |
for (int i = 0; i < adapter.getCount(); i++) { | |
View tabView = null; | |
TextView tabTitleView = null; | |
ImageView tabIconView = null; | |
if (mTabViewLayoutId != 0) { | |
// If there is a custom tab view layout id set, try and inflate it | |
tabView = LayoutInflater.from(getContext()).inflate(mTabViewLayoutId, mTabStrip, | |
false); | |
tabTitleView = (TextView) tabView.findViewById(mTabViewTextViewId); | |
tabIconView = (ImageView) tabView.findViewById(mTabViewImageViewId); | |
} | |
if (tabView == null) { | |
if (tabType.equals(TabType.TEXT)) | |
tabView = createDefaultTextTabView(getContext()); | |
else if (tabType.equals(TabType.ICON)) | |
tabView = createDefaultIconTabView(getContext()); | |
} | |
if (tabTitleView == null && TextView.class.isInstance(tabView)) { | |
tabTitleView = (TextView) tabView; | |
} | |
// setting title to tabView or its' child | |
if (tabTitleView != null) { | |
// if tabTitleView could be found in tabView, then set title to it | |
tabTitleView.setText(adapter.getPageTitle(i)); | |
} else if (TextView.class.isInstance(tabView)) { | |
// if generated tabView is a text view, get page title and set it to the text view | |
((TextView) tabView).setText(adapter.getPageTitle(i)); | |
} | |
// setting icon to tabView or its' child | |
if (tabIconView != null) { | |
// if tabIconView could be found in tabView, then set icon to it | |
tabIconView.setImageResource( | |
getResourceIdByName("ic_" + adapter.getPageTitle(i).toString())); | |
} else if (ImageView.class.isInstance(tabView)) { | |
// if generated tabView is an image view, find the resource with the page | |
// title as resource name suffix. ex: ic_PAGE_TITLE | |
((ImageView) tabView).setImageResource( | |
getResourceIdByName("ic_" + adapter.getPageTitle(i).toString())); | |
} | |
tabView.setOnClickListener(tabClickListener); | |
mTabStrip.addView(tabView); | |
} | |
} | |
@Override | |
protected void onAttachedToWindow() { | |
super.onAttachedToWindow(); | |
if (mViewPager != null) { | |
scrollToTab(mViewPager.getCurrentItem(), 0); | |
} | |
} | |
private int getResourceIdByName(String resourceName) { | |
return getResources().getIdentifier(resourceName, "drawable", getContext().getPackageName()); | |
} | |
private void scrollToTab(int tabIndex, int positionOffset) { | |
final int tabStripChildCount = mTabStrip.getChildCount(); | |
if (tabStripChildCount == 0 || tabIndex < 0 || tabIndex >= tabStripChildCount) { | |
return; | |
} | |
View selectedChild = mTabStrip.getChildAt(tabIndex); | |
if (selectedChild != null) { | |
int targetScrollX = selectedChild.getLeft() + positionOffset; | |
if (tabIndex > 0 || positionOffset > 0) { | |
// If we're not at the first child and are mid-scroll, make sure we obey the offset | |
targetScrollX -= mTitleOffset; | |
} | |
scrollTo(targetScrollX, 0); | |
} | |
} | |
private class InternalViewPagerListener implements ViewPager.OnPageChangeListener { | |
private int mScrollState; | |
@Override | |
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) { | |
int tabStripChildCount = mTabStrip.getChildCount(); | |
if ((tabStripChildCount == 0) || (position < 0) || (position >= tabStripChildCount)) { | |
return; | |
} | |
mTabStrip.onViewPagerPageChanged(position, positionOffset); | |
View selectedTitle = mTabStrip.getChildAt(position); | |
int extraOffset = (selectedTitle != null) | |
? (int) (positionOffset * selectedTitle.getWidth()) | |
: 0; | |
scrollToTab(position, extraOffset); | |
if (mViewPagerPageChangeListener != null) { | |
mViewPagerPageChangeListener.onPageScrolled(position, positionOffset, | |
positionOffsetPixels); | |
} | |
} | |
@Override | |
public void onPageScrollStateChanged(int state) { | |
mScrollState = state; | |
if (mViewPagerPageChangeListener != null) { | |
mViewPagerPageChangeListener.onPageScrollStateChanged(state); | |
} | |
} | |
@Override | |
public void onPageSelected(int position) { | |
if (mScrollState == ViewPager.SCROLL_STATE_IDLE) { | |
mTabStrip.onViewPagerPageChanged(position, 0f); | |
scrollToTab(position, 0); | |
} | |
if (mViewPagerPageChangeListener != null) { | |
mViewPagerPageChangeListener.onPageSelected(position); | |
} | |
} | |
} | |
private class TabClickListener implements View.OnClickListener { | |
@Override | |
public void onClick(View v) { | |
for (int i = 0; i < mTabStrip.getChildCount(); i++) { | |
if (v == mTabStrip.getChildAt(i)) { | |
mViewPager.setCurrentItem(i); | |
return; | |
} | |
} | |
} | |
} | |
public enum TabType { | |
TEXT, ICON; | |
} | |
} |
This file contains hidden or 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
import android.R; | |
import android.content.Context; | |
import android.graphics.Canvas; | |
import android.graphics.Color; | |
import android.graphics.Paint; | |
import android.util.AttributeSet; | |
import android.util.TypedValue; | |
import android.view.View; | |
import android.widget.LinearLayout; | |
class SlidingTabStrip extends LinearLayout { | |
private static final int DEFAULT_BOTTOM_BORDER_THICKNESS_DIPS = 0; | |
private static final byte DEFAULT_BOTTOM_BORDER_COLOR_ALPHA = 0x26; | |
private static final int SELECTED_INDICATOR_THICKNESS_DIPS = 3; | |
private static final int DEFAULT_SELECTED_INDICATOR_COLOR = 0xFFF2FC90; | |
private static final int DEFAULT_DIVIDER_THICKNESS_DIPS = 1; | |
private static final byte DEFAULT_DIVIDER_COLOR_ALPHA = 0x0; | |
private static final float DEFAULT_DIVIDER_HEIGHT = 0.5f; | |
private final int mBottomBorderThickness; | |
private final Paint mBottomBorderPaint; | |
private final int mSelectedIndicatorThickness; | |
private final Paint mSelectedIndicatorPaint; | |
private final int mDefaultBottomBorderColor; | |
private final Paint mDividerPaint; | |
private final float mDividerHeight; | |
private int mSelectedPosition; | |
private float mSelectionOffset; | |
private SlidingTabLayout.TabColorizer mCustomTabColorizer; | |
private final SimpleTabColorizer mDefaultTabColorizer; | |
private boolean isFittingChildren = false; | |
SlidingTabStrip(Context context) { | |
this(context, null); | |
} | |
SlidingTabStrip(Context context, AttributeSet attrs) { | |
super(context, attrs); | |
setWillNotDraw(false); | |
final float density = getResources().getDisplayMetrics().density; | |
TypedValue outValue = new TypedValue(); | |
context.getTheme().resolveAttribute(R.attr.colorForeground, outValue, true); | |
final int themeForegroundColor = outValue.data; | |
mDefaultBottomBorderColor = setColorAlpha(themeForegroundColor, | |
DEFAULT_BOTTOM_BORDER_COLOR_ALPHA); | |
mDefaultTabColorizer = new SimpleTabColorizer(); | |
mDefaultTabColorizer.setIndicatorColors(DEFAULT_SELECTED_INDICATOR_COLOR); | |
mDefaultTabColorizer.setDividerColors(setColorAlpha(themeForegroundColor, | |
DEFAULT_DIVIDER_COLOR_ALPHA)); | |
mBottomBorderThickness = (int) (DEFAULT_BOTTOM_BORDER_THICKNESS_DIPS * density); | |
mBottomBorderPaint = new Paint(); | |
mBottomBorderPaint.setColor(mDefaultBottomBorderColor); | |
mSelectedIndicatorThickness = (int) (SELECTED_INDICATOR_THICKNESS_DIPS * density); | |
mSelectedIndicatorPaint = new Paint(); | |
mDividerHeight = DEFAULT_DIVIDER_HEIGHT; | |
mDividerPaint = new Paint(); | |
mDividerPaint.setStrokeWidth((int) (DEFAULT_DIVIDER_THICKNESS_DIPS * density)); | |
} | |
void setCustomTabColorizer(SlidingTabLayout.TabColorizer customTabColorizer) { | |
mCustomTabColorizer = customTabColorizer; | |
invalidate(); | |
} | |
void setSelectedIndicatorColors(int... colors) { | |
// Make sure that the custom colorizer is removed | |
mCustomTabColorizer = null; | |
mDefaultTabColorizer.setIndicatorColors(colors); | |
invalidate(); | |
} | |
void setDividerColors(int... colors) { | |
// Make sure that the custom colorizer is removed | |
mCustomTabColorizer = null; | |
mDefaultTabColorizer.setDividerColors(colors); | |
invalidate(); | |
} | |
void onViewPagerPageChanged(int position, float positionOffset) { | |
mSelectedPosition = position; | |
mSelectionOffset = positionOffset; | |
invalidate(); | |
} | |
public void setFittingChildren(boolean isFittingChildren) { | |
this.isFittingChildren = isFittingChildren; | |
} | |
@Override | |
public void addView(View child) { | |
super.addView(child); | |
if (isFittingChildren) { | |
// fitting children into layout if isFittingChildren is set true | |
int childCount = getChildCount(); | |
float weightPerChild = (float) 1 / (float) childCount; | |
for (int i = 0; i < childCount; i++) { | |
System.out.println(weightPerChild); | |
View v = getChildAt(i); | |
v.setLayoutParams(new LinearLayout.LayoutParams(LayoutParams.WRAP_CONTENT, | |
LayoutParams.WRAP_CONTENT, weightPerChild)); | |
} | |
} | |
} | |
@Override | |
protected void onDraw(Canvas canvas) { | |
final int height = getHeight(); | |
final int childCount = getChildCount(); | |
final int dividerHeightPx = (int) (Math.min(Math.max(0f, mDividerHeight), 1f) * height); | |
final SlidingTabLayout.TabColorizer tabColorizer = mCustomTabColorizer != null | |
? mCustomTabColorizer | |
: mDefaultTabColorizer; | |
// Thick colored underline below the current selection | |
if (childCount > 0) { | |
View selectedTitle = getChildAt(mSelectedPosition); | |
int left = selectedTitle.getLeft(); | |
int right = selectedTitle.getRight(); | |
int color = tabColorizer.getIndicatorColor(mSelectedPosition); | |
if (mSelectionOffset > 0f && mSelectedPosition < (getChildCount() - 1)) { | |
int nextColor = tabColorizer.getIndicatorColor(mSelectedPosition + 1); | |
if (color != nextColor) { | |
color = blendColors(nextColor, color, mSelectionOffset); | |
} | |
// Draw the selection partway between the tabs | |
View nextTitle = getChildAt(mSelectedPosition + 1); | |
left = (int) (mSelectionOffset * nextTitle.getLeft() + | |
(1.0f - mSelectionOffset) * left); | |
right = (int) (mSelectionOffset * nextTitle.getRight() + | |
(1.0f - mSelectionOffset) * right); | |
} | |
mSelectedIndicatorPaint.setColor(color); | |
canvas.drawRect(left, height - mSelectedIndicatorThickness, right, | |
height, mSelectedIndicatorPaint); | |
} | |
// Thin underline along the entire bottom edge | |
canvas.drawRect(0, height - mBottomBorderThickness, getWidth(), height, mBottomBorderPaint); | |
// Vertical separators between the titles | |
int separatorTop = (height - dividerHeightPx) / 2; | |
for (int i = 0; i < childCount - 1; i++) { | |
View child = getChildAt(i); | |
mDividerPaint.setColor(tabColorizer.getDividerColor(i)); | |
canvas.drawLine(child.getRight(), separatorTop, child.getRight(), | |
separatorTop + dividerHeightPx, mDividerPaint); | |
} | |
} | |
/** | |
* Set the alpha value of the {@code color} to be the given {@code alpha} value. | |
*/ | |
private static int setColorAlpha(int color, byte alpha) { | |
return Color.argb(alpha, Color.red(color), Color.green(color), Color.blue(color)); | |
} | |
/** | |
* Blend {@code color1} and {@code color2} using the given ratio. | |
* | |
* @param ratio of which to blend. 1.0 will return {@code color1}, 0.5 will give an even blend, | |
* 0.0 will return {@code color2}. | |
*/ | |
private static int blendColors(int color1, int color2, float ratio) { | |
final float inverseRation = 1f - ratio; | |
float r = (Color.red(color1) * ratio) + (Color.red(color2) * inverseRation); | |
float g = (Color.green(color1) * ratio) + (Color.green(color2) * inverseRation); | |
float b = (Color.blue(color1) * ratio) + (Color.blue(color2) * inverseRation); | |
return Color.rgb((int) r, (int) g, (int) b); | |
} | |
private static class SimpleTabColorizer implements SlidingTabLayout.TabColorizer { | |
private int[] mIndicatorColors; | |
private int[] mDividerColors; | |
@Override | |
public final int getIndicatorColor(int position) { | |
return mIndicatorColors[position % mIndicatorColors.length]; | |
} | |
@Override | |
public final int getDividerColor(int position) { | |
return mDividerColors[position % mDividerColors.length]; | |
} | |
void setIndicatorColors(int... colors) { | |
mIndicatorColors = colors; | |
} | |
void setDividerColors(int... colors) { | |
mDividerColors = colors; | |
} | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment