您当前的位置: 首页 >  xiangzhihong8

仿qq最新侧滑菜单

xiangzhihong8 发布时间:2014-10-21 23:29:06 ,浏览量:6

为了后续对这个项目进行优化,比如透明度动画、背景图的位移动画,以及性能上的优化。

我把这个项目上传到github上面,请大家随时关注。

github地址 https://github.com/sunguowei

最近项目要做一个QQ5.0的侧滑菜单效果,和传统的侧滑菜单存在着一些差异。想必大家都已经见识过了。

为了不重复发明轮子,先去github上面搜索了一番。

发现了几个类似的,但是还是有一些不同。

下面是搜索到的类似的开源项目。

RESideMenu(ios项目)

https://github.com/romaonthego/RESideMenu

AndroidResideMenu

https://github.com/SpecialCyCi/AndroidResideMenu

ResideLayout

https://github.com/kyze8439690/ResideLayout

研究了一下这些开源项目的源代码。感觉并不是特别适用于我们自己的项目。所以,我自己又研究了一下。最后的效果如下。当然了,还有很多可以优化的地方,后续再慢慢优化。

备注:如果图片动画显示不出来,可以点击这个网址查看。

https://img-blog.csdn.net/20140902225149282?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbWFub2Vs/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast

我是基于SlidingMenu库进行的二次修改,增加了一些转场动画。

大家对这个库应该比较熟悉,下面是SlidingMenu的github地址。非常感谢Jeremy Feinstein提供的这个库,让广大Android Developers省去了非常多的麻烦。

https://github.com/jfeinstein10/SlidingMenu

备注:SlidingMenu使用了SherlockActionBar这个库,配置起来会比较麻烦,在文章的最后我会把demo上传,供大家下载,减去了大家自己配置项目的麻烦。

我主要修改了2个类,SlidingMenu.java和CustonViewAbove.java,只是增加了一些功能,并没有修改原本的功能。

做了修改的地方,我做了中文注释,其实实现很简单,几行代码而已。推荐大家下载Demo,然后自己调试一下。Demo的下载地址在文章的末尾。

废话不多说,直接上代码,略微有点长。

[java] view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public class SlidingMenu extends RelativeLayout {  
  2.   
  3.     private static final String TAG = SlidingMenu.class.getSimpleName();  
  4.   
  5.     public static final int SLIDING_WINDOW = 0;  
  6.     public static final int SLIDING_CONTENT = 1;  
  7.     private boolean mActionbarOverlay = false;  
  8.   
  9.     /** 
  10.      * Constant value for use with setTouchModeAbove(). Allows the SlidingMenu 
  11.      * to be opened with a swipe gesture on the screen's margin 
  12.      */  
  13.     public static final int TOUCHMODE_MARGIN = 0;  
  14.   
  15.     /** 
  16.      * Constant value for use with setTouchModeAbove(). Allows the SlidingMenu 
  17.      * to be opened with a swipe gesture anywhere on the screen 
  18.      */  
  19.     public static final int TOUCHMODE_FULLSCREEN = 1;  
  20.   
  21.     /** 
  22.      * Constant value for use with setTouchModeAbove(). Denies the SlidingMenu 
  23.      * to be opened with a swipe gesture 
  24.      */  
  25.     public static final int TOUCHMODE_NONE = 2;  
  26.   
  27.     /** 
  28.      * Constant value for use with setMode(). Puts the menu to the left of the 
  29.      * content. 
  30.      */  
  31.     public static final int LEFT = 0;  
  32.   
  33.     /** 
  34.      * Constant value for use with setMode(). Puts the menu to the right of the 
  35.      * content. 
  36.      */  
  37.     public static final int RIGHT = 1;  
  38.   
  39.     /** 
  40.      * Constant value for use with setMode(). Puts menus to the left and right 
  41.      * of the content. 
  42.      */  
  43.     public static final int LEFT_RIGHT = 2;  
  44.   
  45.     private CustomViewAbove mViewAbove;  
  46.   
  47.     private CustomViewBehind mViewBehind;  
  48.   
  49.         /** 整体的背景,用一个ImageView代替 */  
  50.     private ImageView mViewBackground;  
  51.   
  52.     private OnOpenListener mOpenListener;  
  53.   
  54.     private OnOpenListener mSecondaryOpenListner;  
  55.   
  56.     private OnCloseListener mCloseListener;  
  57.   
  58.     /** 
  59.      * The listener interface for receiving onOpen events. The class that is 
  60.      * interested in processing a onOpen event implements this interface, and 
  61.      * the object created with that class is registered with a component using 
  62.      * the component's addOnOpenListener method. When 
  63.      * the onOpen event occurs, that object's appropriate 
  64.      * method is invoked 
  65.      */  
  66.     public interface OnOpenListener {  
  67.   
  68.         /** 
  69.          * On open. 
  70.          */  
  71.         public void onOpen();  
  72.     }  
  73.   
  74.     /** 
  75.      * The listener interface for receiving onOpened events. The class that is 
  76.      * interested in processing a onOpened event implements this interface, and 
  77.      * the object created with that class is registered with a component using 
  78.      * the component's addOnOpenedListener method. When 
  79.      * the onOpened event occurs, that object's appropriate 
  80.      * method is invoked. 
  81.      *  
  82.      * @see OnOpenedEvent 
  83.      */  
  84.     public interface OnOpenedListener {  
  85.   
  86.         /** 
  87.          * On opened. 
  88.          */  
  89.         public void onOpened();  
  90.     }  
  91.   
  92.     /** 
  93.      * The listener interface for receiving onClose events. The class that is 
  94.      * interested in processing a onClose event implements this interface, and 
  95.      * the object created with that class is registered with a component using 
  96.      * the component's addOnCloseListener method. When 
  97.      * the onClose event occurs, that object's appropriate 
  98.      * method is invoked. 
  99.      *  
  100.      * @see OnCloseEvent 
  101.      */  
  102.     public interface OnCloseListener {  
  103.   
  104.         /** 
  105.          * On close. 
  106.          */  
  107.         public void onClose();  
  108.     }  
  109.   
  110.     /** 
  111.      * The listener interface for receiving onClosed events. The class that is 
  112.      * interested in processing a onClosed event implements this interface, and 
  113.      * the object created with that class is registered with a component using 
  114.      * the component's addOnClosedListener method. When 
  115.      * the onClosed event occurs, that object's appropriate 
  116.      * method is invoked. 
  117.      *  
  118.      * @see OnClosedEvent 
  119.      */  
  120.     public interface OnClosedListener {  
  121.   
  122.         /** 
  123.          * On closed. 
  124.          */  
  125.         public void onClosed();  
  126.     }  
  127.   
  128.     /** 
  129.      * The Interface CanvasTransformer. 
  130.      */  
  131.     public interface CanvasTransformer {  
  132.   
  133.         /** 
  134.          * Transform canvas. 
  135.          *  
  136.          * @param canvas 
  137.          *            the canvas 
  138.          * @param percentOpen 
  139.          *            the percent open 
  140.          */  
  141.         public void transformCanvas(Canvas canvas, float percentOpen);  
  142.     }  
  143.   
  144.     /** 
  145.      * Instantiates a new SlidingMenu. 
  146.      *  
  147.      * @param context 
  148.      *            the associated Context 
  149.      */  
  150.     public SlidingMenu(Context context) {  
  151.         this(context, null);  
  152.     }  
  153.   
  154.     /** 
  155.      * Instantiates a new SlidingMenu and attach to Activity. 
  156.      *  
  157.      * @param activity 
  158.      *            the activity to attach slidingmenu 
  159.      * @param slideStyle 
  160.      *            the slidingmenu style 
  161.      */  
  162.     public SlidingMenu(Activity activity, int slideStyle) {  
  163.         this(activity, null);  
  164.         this.attachToActivity(activity, slideStyle);  
  165.     }  
  166.   
  167.     /** 
  168.      * Instantiates a new SlidingMenu. 
  169.      *  
  170.      * @param context 
  171.      *            the associated Context 
  172.      * @param attrs 
  173.      *            the attrs 
  174.      */  
  175.     public SlidingMenu(Context context, AttributeSet attrs) {  
  176.         this(context, attrs, 0);  
  177.     }  
  178.   
  179.     /** 
  180.      * Instantiates a new SlidingMenu. 
  181.      *  
  182.      * @param context 
  183.      *            the associated Context 
  184.      * @param attrs 
  185.      *            the attrs 
  186.      * @param defStyle 
  187.      *            the def style 
  188.      */  
  189.     public SlidingMenu(Context context, AttributeSet attrs, int defStyle) {  
  190.         super(context, attrs, defStyle);  
  191.                 /** SlidingMenu是一个RelativeLayout,这里把背景图ImageView添加到RelativeLayout的最底层。*/  
  192.         LayoutParams backgroundParams = new LayoutParams(  
  193.                 LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT);  
  194.         mViewBackground = new ImageView(context);  
  195.         mViewBackground.setScaleType(ImageView.ScaleType.CENTER_CROP);  
  196.         addView(mViewBackground, backgroundParams);  
  197.           
  198.                 LayoutParams behindParams = new LayoutParams(LayoutParams.MATCH_PARENT,  
  199.                 LayoutParams.MATCH_PARENT);  
  200.         mViewBehind = new CustomViewBehind(context);  
  201.         addView(mViewBehind, behindParams);  
  202.         LayoutParams aboveParams = new LayoutParams(LayoutParams.MATCH_PARENT,  
  203.                 LayoutParams.MATCH_PARENT);  
  204.         mViewAbove = new CustomViewAbove(context);  
  205.         addView(mViewAbove, aboveParams);  
  206.         // register the CustomViewBehind with the CustomViewAbove  
  207.         mViewAbove.setCustomViewBehind(mViewBehind);  
  208.         mViewBehind.setCustomViewAbove(mViewAbove);  
  209.         mViewAbove.setOnPageChangeListener(new OnPageChangeListener() {  
  210.             public static final int POSITION_OPEN = 0;  
  211.             public static final int POSITION_CLOSE = 1;  
  212.             public static final int POSITION_SECONDARY_OPEN = 2;  
  213.   
  214.             public void onPageScrolled(int position, float positionOffset,  
  215.                     int positionOffsetPixels) {  
  216.             }  
  217.   
  218.             public void onPageSelected(int position) {  
  219.                 if (position == POSITION_OPEN && mOpenListener != null) {  
  220.                     mOpenListener.onOpen();  
  221.                 } else if (position == POSITION_CLOSE && mCloseListener != null) {  
  222.                     mCloseListener.onClose();  
  223.                 } else if (position == POSITION_SECONDARY_OPEN  
  224.                         && mSecondaryOpenListner != null) {  
  225.                     mSecondaryOpenListner.onOpen();  
  226.                 }  
  227.             }  
  228.         });  
  229.   
  230.         // now style everything!  
  231.         TypedArray ta = context.obtainStyledAttributes(attrs,  
  232.                 R.styleable.SlidingMenu);  
  233.         // set the above and behind views if defined in xml  
  234.         int mode = ta.getInt(R.styleable.SlidingMenu_mode, LEFT);  
  235.         setMode(mode);  
  236.         int viewAbove = ta.getResourceId(R.styleable.SlidingMenu_viewAbove, -1);  
  237.         if (viewAbove != -1) {  
  238.             setContent(viewAbove);  
  239.         } else {  
  240.             setContent(new FrameLayout(context));  
  241.         }  
  242.         int viewBehind = ta.getResourceId(R.styleable.SlidingMenu_viewBehind,  
  243.                 -1);  
  244.         if (viewBehind != -1) {  
  245.             setMenu(viewBehind);  
  246.         } else {  
  247.             setMenu(new FrameLayout(context));  
  248.         }  
  249.         int touchModeAbove = ta.getInt(R.styleable.SlidingMenu_touchModeAbove,  
  250.                 TOUCHMODE_MARGIN);  
  251.         setTouchModeAbove(touchModeAbove);  
  252.         int touchModeBehind = ta.getInt(  
  253.                 R.styleable.SlidingMenu_touchModeBehind, TOUCHMODE_MARGIN);  
  254.         setTouchModeBehind(touchModeBehind);  
  255.   
  256.         int offsetBehind = (int) ta.getDimension(  
  257.                 R.styleable.SlidingMenu_behindOffset, -1);  
  258.         int widthBehind = (int) ta.getDimension(  
  259.                 R.styleable.SlidingMenu_behindWidth, -1);  
  260.         if (offsetBehind != -1 && widthBehind != -1)  
  261.             throw new IllegalStateException(  
  262.                     "Cannot set both behindOffset and behindWidth for a SlidingMenu");  
  263.         else if (offsetBehind != -1)  
  264.             setBehindOffset(offsetBehind);  
  265.         else if (widthBehind != -1)  
  266.             setBehindWidth(widthBehind);  
  267.         else  
  268.             setBehindOffset(0);  
  269.         float scrollOffsetBehind = ta.getFloat(  
  270.                 R.styleable.SlidingMenu_behindScrollScale, 0.33f);  
  271.         setBehindScrollScale(scrollOffsetBehind);  
  272.         int shadowRes = ta.getResourceId(  
  273.                 R.styleable.SlidingMenu_shadowDrawable, -1);  
  274.         if (shadowRes != -1) {  
  275.             setShadowDrawable(shadowRes);  
  276.         }  
  277.         int shadowWidth = (int) ta.getDimension(  
  278.                 R.styleable.SlidingMenu_shadowWidth, 0);  
  279.         setShadowWidth(shadowWidth);  
  280.         boolean fadeEnabled = ta.getBoolean(  
  281.                 R.styleable.SlidingMenu_fadeEnabled, true);  
  282.         setFadeEnabled(fadeEnabled);  
  283.         float fadeDeg = ta.getFloat(R.styleable.SlidingMenu_fadeDegree, 0.33f);  
  284.         setFadeDegree(fadeDeg);  
  285.         boolean selectorEnabled = ta.getBoolean(  
  286.                 R.styleable.SlidingMenu_selectorEnabled, false);  
  287.         setSelectorEnabled(selectorEnabled);  
  288.         int selectorRes = ta.getResourceId(  
  289.                 R.styleable.SlidingMenu_selectorDrawable, -1);  
  290.         if (selectorRes != -1)  
  291.             setSelectorDrawable(selectorRes);  
  292.         ta.recycle();  
  293.     }  
  294.   
  295.     /** 
  296.      * Attaches the SlidingMenu to an entire Activity 
  297.      *  
  298.      * @param activity 
  299.      *            the Activity 
  300.      * @param slideStyle 
  301.      *            either SLIDING_CONTENT or SLIDING_WINDOW 
  302.      */  
  303.     public void attachToActivity(Activity activity, int slideStyle) {  
  304.         attachToActivity(activity, slideStyle, false);  
  305.     }  
  306.   
  307.     /** 
  308.      * Attaches the SlidingMenu to an entire Activity 
  309.      *  
  310.      * @param activity 
  311.      *            the Activity 
  312.      * @param slideStyle 
  313.      *            either SLIDING_CONTENT or SLIDING_WINDOW 
  314.      * @param actionbarOverlay 
  315.      *            whether or not the ActionBar is overlaid 
  316.      */  
  317.     public void attachToActivity(Activity activity, int slideStyle,  
  318.             boolean actionbarOverlay) {  
  319.         if (slideStyle != SLIDING_WINDOW && slideStyle != SLIDING_CONTENT)  
  320.             throw new IllegalArgumentException(  
  321.                     "slideStyle must be either SLIDING_WINDOW or SLIDING_CONTENT");  
  322.   
  323.         if (getParent() != null)  
  324.             throw new IllegalStateException(  
  325.                     "This SlidingMenu appears to already be attached");  
  326.   
  327.         // get the window background  
  328.         TypedArray a = activity.getTheme().obtainStyledAttributes(  
  329.                 new int[] { android.R.attr.windowBackground });  
  330.         int background = a.getResourceId(0, 0);  
  331.         a.recycle();  
  332.   
  333.         switch (slideStyle) {  
  334.         case SLIDING_WINDOW:  
  335.             mActionbarOverlay = false;  
  336.             ViewGroup decor = (ViewGroup) activity.getWindow().getDecorView();  
  337.             ViewGroup decorChild = (ViewGroup) decor.getChildAt(0);  
  338.             // save ActionBar themes that have transparent assets  
  339.             decorChild.setBackgroundResource(background);  
  340.             decor.removeView(decorChild);  
  341.             decor.addView(this);  
  342.             setContent(decorChild);  
  343.             break;  
  344.         case SLIDING_CONTENT:  
  345.             mActionbarOverlay = actionbarOverlay;  
  346.             // take the above view out of  
  347.             ViewGroup contentParent = (ViewGroup) activity  
  348.                     .findViewById(android.R.id.content);  
  349.             View content = contentParent.getChildAt(0);  
  350.             contentParent.removeView(content);  
  351.             contentParent.addView(this);  
  352.             setContent(content);  
  353.             // save people from having transparent backgrounds  
  354.             if (content.getBackground() == null)  
  355.                 content.setBackgroundResource(background);  
  356.             break;  
  357.         }  
  358.     }  
  359.   
  360.     /** 
  361.      * Set the above view content from a layout resource. The resource will be 
  362.      * inflated, adding all top-level views to the above view. 
  363.      *  
  364.      * @param res 
  365.      *            the new content 
  366.      */  
  367.     public void setContent(int res) {  
  368.         setContent(LayoutInflater.from(getContext()).inflate(res, null));  
  369.     }  
  370.   
  371.     /** 
  372.      * Set the above view content to the given View. 
  373.      *  
  374.      * @param view 
  375.      *            The desired content to display. 
  376.      */  
  377.     public void setContent(View view) {  
  378.         mViewAbove.setContent(view);  
  379.         showContent();  
  380.     }  
  381.   
  382.     /** 
  383.      * 设置背景图片 
  384.      *  
  385.      * @param resid 
  386.      */  
  387.     public void setBackgroundImage(int resid) {  
  388.         mViewBackground.setBackgroundResource(resid);  
  389.     }  
  390.   
  391.     /** 
  392.      * Retrieves the current content. 
  393.      *  
  394.      * @return the current content 
  395.      */  
  396.     public View getContent() {  
  397.         return mViewAbove.getContent();  
  398.     }  
  399.   
  400.     /** 
  401.      * Set the behind view (menu) content from a layout resource. The resource 
  402.      * will be inflated, adding all top-level views to the behind view. 
  403.      *  
  404.      * @param res 
  405.      *            the new content 
  406.      */  
  407.     public void setMenu(int res) {  
  408.         setMenu(LayoutInflater.from(getContext()).inflate(res, null));  
  409.     }  
  410.   
  411.     /** 
  412.      * Set the behind view (menu) content to the given View. 
  413.      *  
  414.      * @param view 
  415.      *            The desired content to display. 
  416.      */  
  417.     public void setMenu(View v) {  
  418.         mViewBehind.setContent(v);  
  419.     }  
  420.   
  421.     /** 
  422.      * Retrieves the main menu. 
  423.      *  
  424.      * @return the main menu 
  425.      */  
  426.     public View getMenu() {  
  427.         return mViewBehind.getContent();  
  428.     }  
  429.   
  430.     /** 
  431.      * Set the secondary behind view (right menu) content from a layout 
  432.      * resource. The resource will be inflated, adding all top-level views to 
  433.      * the behind view. 
  434.      *  
  435.      * @param res 
  436.      *            the new content 
  437.      */  
  438.     public void setSecondaryMenu(int res) {  
  439.         setSecondaryMenu(LayoutInflater.from(getContext()).inflate(res, null));  
  440.     }  
  441.   
  442.     /** 
  443.      * Set the secondary behind view (right menu) content to the given View. 
  444.      *  
  445.      * @param view 
  446.      *            The desired content to display. 
  447.      */  
  448.     public void setSecondaryMenu(View v) {  
  449.         mViewBehind.setSecondaryContent(v);  
  450.         // mViewBehind.invalidate();  
  451.     }  
  452.   
  453.     /** 
  454.      * Retrieves the current secondary menu (right). 
  455.      *  
  456.      * @return the current menu 
  457.      */  
  458.     public View getSecondaryMenu() {  
  459.         return mViewBehind.getSecondaryContent();  
  460.     }  
  461.   
  462.     /** 
  463.      * Sets the sliding enabled. 
  464.      *  
  465.      * @param b 
  466.      *            true to enable sliding, false to disable it. 
  467.      */  
  468.     public void setSlidingEnabled(boolean b) {  
  469.         mViewAbove.setSlidingEnabled(b);  
  470.     }  
  471.   
  472.     /** 
  473.      * Checks if is sliding enabled. 
  474.      *  
  475.      * @return true, if is sliding enabled 
  476.      */  
  477.     public boolean isSlidingEnabled() {  
  478.         return mViewAbove.isSlidingEnabled();  
  479.     }  
  480.   
  481.     /** 
  482.      * Sets which side the SlidingMenu should appear on. 
  483.      *  
  484.      * @param mode 
  485.      *            must be either SlidingMenu.LEFT or SlidingMenu.RIGHT 
  486.      */  
  487.     public void setMode(int mode) {  
  488.         if (mode != LEFT && mode != RIGHT && mode != LEFT_RIGHT) {  
  489.             throw new IllegalStateException(  
  490.                     "SlidingMenu mode must be LEFT, RIGHT, or LEFT_RIGHT");  
  491.         }  
  492.         mViewBehind.setMode(mode);  
  493.     }  
  494.   
  495.     /** 
  496.      * Returns the current side that the SlidingMenu is on. 
  497.      *  
  498.      * @return the current mode, either SlidingMenu.LEFT or SlidingMenu.RIGHT 
  499.      */  
  500.     public int getMode() {  
  501.         return mViewBehind.getMode();  
  502.     }  
  503.   
  504.     /** 
  505.      * Sets whether or not the SlidingMenu is in static mode (i.e. nothing is 
  506.      * moving and everything is showing) 
  507.      *  
  508.      * @param b 
  509.      *            true to set static mode, false to disable static mode. 
  510.      */  
  511.     public void setStatic(boolean b) {  
  512.         if (b) {  
  513.             setSlidingEnabled(false);  
  514.             mViewAbove.setCustomViewBehind(null);  
  515.             mViewAbove.setCurrentItem(1);  
  516.             // mViewBehind.setCurrentItem(0);  
  517.         } else {  
  518.             mViewAbove.setCurrentItem(1);  
  519.             // mViewBehind.setCurrentItem(1);  
  520.             mViewAbove.setCustomViewBehind(mViewBehind);  
  521.             setSlidingEnabled(true);  
  522.         }  
  523.     }  
  524.   
  525.     /** 
  526.      * Opens the menu and shows the menu view. 
  527.      */  
  528.     public void showMenu() {  
  529.         showMenu(true);  
  530.     }  
  531.   
  532.     /** 
  533.      * Opens the menu and shows the menu view. 
  534.      *  
  535.      * @param animate 
  536.      *            true to animate the transition, false to ignore animation 
  537.      */  
  538.     public void showMenu(boolean animate) {  
  539.         mViewAbove.setCurrentItem(0, animate);  
  540.     }  
  541.   
  542.     /** 
  543.      * Opens the menu and shows the secondary menu view. Will default to the 
  544.      * regular menu if there is only one. 
  545.      */  
  546.     public void showSecondaryMenu() {  
  547.         showSecondaryMenu(true);  
  548.     }  
  549.   
  550.     /** 
  551.      * Opens the menu and shows the secondary (right) menu view. Will default to 
  552.      * the regular menu if there is only one. 
  553.      *  
  554.      * @param animate 
  555.      *            true to animate the transition, false to ignore animation 
  556.      */  
  557.     public void showSecondaryMenu(boolean animate) {  
  558.         mViewAbove.setCurrentItem(2, animate);  
  559.     }  
  560.   
  561.     /** 
  562.      * Closes the menu and shows the above view. 
  563.      */  
  564.     public void showContent() {  
  565.         showContent(true);  
  566.     }  
  567.   
  568.     /** 
  569.      * Closes the menu and shows the above view. 
  570.      *  
  571.      * @param animate 
  572.      *            true to animate the transition, false to ignore animation 
  573.      */  
  574.     public void showContent(boolean animate) {  
  575.         mViewAbove.setCurrentItem(1, animate);  
  576.     }  
  577.   
  578.     /** 
  579.      * Toggle the SlidingMenu. If it is open, it will be closed, and vice versa. 
  580.      */  
  581.     public void toggle() {  
  582.         toggle(true);  
  583.     }  
  584.   
  585.     /** 
  586.      * Toggle the SlidingMenu. If it is open, it will be closed, and vice versa. 
  587.      *  
  588.      * @param animate 
  589.      *            true to animate the transition, false to ignore animation 
  590.      */  
  591.     public void toggle(boolean animate) {  
  592.         if (isMenuShowing()) {  
  593.             showContent(animate);  
  594.         } else {  
  595.             showMenu(animate);  
  596.         }  
  597.     }  
  598.   
  599.     /** 
  600.      * Checks if is the behind view showing. 
  601.      *  
  602.      * @return Whether or not the behind view is showing 
  603.      */  
  604.     public boolean isMenuShowing() {  
  605.         return mViewAbove.getCurrentItem() == 0  
  606.                 || mViewAbove.getCurrentItem() == 2;  
  607.     }  
  608.   
  609.     /** 
  610.      * Checks if is the behind view showing. 
  611.      *  
  612.      * @return Whether or not the behind view is showing 
  613.      */  
  614.     public boolean isSecondaryMenuShowing() {  
  615.         return mViewAbove.getCurrentItem() == 2;  
  616.     }  
  617.   
  618.     /** 
  619.      * Gets the behind offset. 
  620.      *  
  621.      * @return The margin on the right of the screen that the behind view 
  622.      *         scrolls to 
  623.      */  
  624.     public int getBehindOffset() {  
  625.         return ((RelativeLayout.LayoutParams) mViewBehind.getLayoutParams()).rightMargin;  
  626.     }  
  627.   
  628.     /** 
  629.      * Sets the behind offset. 
  630.      *  
  631.      * @param i 
  632.      *            The margin, in pixels, on the right of the screen that the 
  633.      *            behind view scrolls to. 
  634.      */  
  635.     public void setBehindOffset(int i) {  
  636.         // RelativeLayout.LayoutParams params =  
  637.         // ((RelativeLayout.LayoutParams)mViewBehind.getLayoutParams());  
  638.         // int bottom = params.bottomMargin;  
  639.         // int top = params.topMargin;  
  640.         // int left = params.leftMargin;  
  641.         // params.setMargins(left, top, i, bottom);  
  642.         mViewBehind.setWidthOffset(i);  
  643.     }  
  644.   
  645.     /** 
  646.      * Sets the behind offset. 
  647.      *  
  648.      * @param resID 
  649.      *            The dimension resource id to be set as the behind offset. The 
  650.      *            menu, when open, will leave this width margin on the right of 
  651.      *            the screen. 
  652.      */  
  653.     public void setBehindOffsetRes(int resID) {  
  654.         int i = (int) getContext().getResources().getDimension(resID);  
  655.         setBehindOffset(i);  
  656.     }  
  657.   
  658.     /** 
  659.      * Sets the above offset. 
  660.      *  
  661.      * @param i 
  662.      *            the new above offset, in pixels 
  663.      */  
  664.     public void setAboveOffset(int i) {  
  665.         mViewAbove.setAboveOffset(i);  
  666.     }  
  667.   
  668.     /** 
  669.      * Sets the above offset. 
  670.      *  
  671.      * @param resID 
  672.      *            The dimension resource id to be set as the above offset. 
  673.      */  
  674.     public void setAboveOffsetRes(int resID) {  
  675.         int i = (int) getContext().getResources().getDimension(resID);  
  676.         setAboveOffset(i);  
  677.     }  
  678.   
  679.     /** 
  680.      * Sets the behind width. 
  681.      *  
  682.      * @param i 
  683.      *            The width the Sliding Menu will open to, in pixels 
  684.      */  
  685.     @SuppressWarnings("deprecation")  
  686.     public void setBehindWidth(int i) {  
  687.         int width;  
  688.         Display display = ((WindowManager) getContext().getSystemService(  
  689.                 Context.WINDOW_SERVICE)).getDefaultDisplay();  
  690.         try {  
  691.             Class cls = Display.class;  
  692.             Class[] parameterTypes = { Point.class };  
  693.             Point parameter = new Point();  
  694.             Method method = cls.getMethod("getSize", parameterTypes);  
  695.             method.invoke(display, parameter);  
  696.             width = parameter.x;  
  697.         } catch (Exception e) {  
  698.             width = display.getWidth();  
  699.         }  
  700.         setBehindOffset(width - i);  
  701.     }  
  702.   
  703.     /** 
  704.      * Sets the behind width. 
  705.      *  
  706.      * @param res 
  707.      *            The dimension resource id to be set as the behind width 
  708.      *            offset. The menu, when open, will open this wide. 
  709.      */  
  710.     public void setBehindWidthRes(int res) {  
  711.         int i = (int) getContext().getResources().getDimension(res);  
  712.         setBehindWidth(i);  
  713.     }  
  714.   
  715.     /** 
  716.      * Gets the behind scroll scale. 
  717.      *  
  718.      * @return The scale of the parallax scroll 
  719.      */  
  720.     public float getBehindScrollScale() {  
  721.         return mViewBehind.getScrollScale();  
  722.     }  
  723.   
  724.     /** 
  725.      * Gets the touch mode margin threshold 
  726.      *  
  727.      * @return the touch mode margin threshold 
  728.      */  
  729.     public int getTouchmodeMarginThreshold() {  
  730.         return mViewBehind.getMarginThreshold();  
  731.     }  
  732.   
  733.     /** 
  734.      * Set the touch mode margin threshold 
  735.      *  
  736.      * @param touchmodeMarginThreshold 
  737.      */  
  738.     public void setTouchmodeMarginThreshold(int touchmodeMarginThreshold) {  
  739.         mViewBehind.setMarginThreshold(touchmodeMarginThreshold);  
  740.     }  
  741.   
  742.     /** 
  743.      * Sets the behind scroll scale. 
  744.      *  
  745.      * @param f 
  746.      *            The scale of the parallax scroll (i.e. 1.0f scrolls 1 pixel 
  747.      *            for every 1 pixel that the above view scrolls and 0.0f scrolls 
  748.      *            0 pixels) 
  749.      */  
  750.     public void setBehindScrollScale(float f) {  
  751.         if (f  1)  
  752.             throw new IllegalStateException(  
  753.                     "ScrollScale must be between 0 and 1");  
  754.         mViewBehind.setScrollScale(f);  
  755.     }  
  756.   
  757.     /** 
  758.      * Sets the behind canvas transformer. 
  759.      *  
  760.      * @param t 
  761.      *            the new behind canvas transformer 
  762.      */  
  763.     public void setBehindCanvasTransformer(CanvasTransformer t) {  
  764.         mViewBehind.setCanvasTransformer(t);  
  765.     }  
  766.   
  767.     /** 
  768.      * 设置右侧视图的转场动画 
  769.      *  
  770.      * @param t 
  771.      *            the new above canvas transformer 
  772.      */  
  773.     public void setAboveCanvasTransformer(CanvasTransformer t) {  
  774.         mViewAbove.setCanvasTransformer(t);  
  775.     }  
  776.   
  777.     /** 
  778.      * Gets the touch mode above. 
  779.      *  
  780.      * @return the touch mode above 
  781.      */  
  782.     public int getTouchModeAbove() {  
  783.         return mViewAbove.getTouchMode();  
  784.     }  
  785.   
  786.     /** 
  787.      * Controls whether the SlidingMenu can be opened with a swipe gesture. 
  788.      * Options are {@link #TOUCHMODE_MARGIN TOUCHMODE_MARGIN}, 
  789.      * {@link #TOUCHMODE_FULLSCREEN TOUCHMODE_FULLSCREEN}, or 
  790.      * {@link #TOUCHMODE_NONE TOUCHMODE_NONE} 
  791.      *  
  792.      * @param i 
  793.      *            the new touch mode 
  794.      */  
  795.     public void setTouchModeAbove(int i) {  
  796.         if (i != TOUCHMODE_FULLSCREEN && i != TOUCHMODE_MARGIN  
  797.                 && i != TOUCHMODE_NONE) {  
  798.             throw new IllegalStateException(  
  799.                     "TouchMode must be set to either"  
  800.                             + "TOUCHMODE_FULLSCREEN or TOUCHMODE_MARGIN or TOUCHMODE_NONE.");  
  801.         }  
  802.         mViewAbove.setTouchMode(i);  
  803.     }  
  804.   
  805.     /** 
  806.      * Controls whether the SlidingMenu can be opened with a swipe gesture. 
  807.      * Options are {@link #TOUCHMODE_MARGIN TOUCHMODE_MARGIN}, 
  808.      * {@link #TOUCHMODE_FULLSCREEN TOUCHMODE_FULLSCREEN}, or 
  809.      * {@link #TOUCHMODE_NONE TOUCHMODE_NONE} 
  810.      *  
  811.      * @param i 
  812.      *            the new touch mode 
  813.      */  
  814.     public void setTouchModeBehind(int i) {  
  815.         if (i != TOUCHMODE_FULLSCREEN && i != TOUCHMODE_MARGIN  
  816.                 && i != TOUCHMODE_NONE) {  
  817.             throw new IllegalStateException(  
  818.                     "TouchMode must be set to either"  
  819.                             + "TOUCHMODE_FULLSCREEN or TOUCHMODE_MARGIN or TOUCHMODE_NONE.");  
  820.         }  
  821.         mViewBehind.setTouchMode(i);  
  822.     }  
  823.   
  824.     /** 
  825.      * Sets the shadow drawable. 
  826.      *  
  827.      * @param resId 
  828.      *            the resource ID of the new shadow drawable 
  829.      */  
  830.     public void setShadowDrawable(int resId) {  
  831.         setShadowDrawable(getContext().getResources().getDrawable(resId));  
  832.     }  
  833.   
  834.     /** 
  835.      * Sets the shadow drawable. 
  836.      *  
  837.      * @param d 
  838.      *            the new shadow drawable 
  839.      */  
  840.     public void setShadowDrawable(Drawable d) {  
  841.         mViewBehind.setShadowDrawable(d);  
  842.     }  
  843.   
  844.     /** 
  845.      * Sets the secondary (right) shadow drawable. 
  846.      *  
  847.      * @param resId 
  848.      *            the resource ID of the new shadow drawable 
  849.      */  
  850.     public void setSecondaryShadowDrawable(int resId) {  
  851.         setSecondaryShadowDrawable(getContext().getResources().getDrawable(  
  852.                 resId));  
  853.     }  
  854.   
  855.     /** 
  856.      * Sets the secondary (right) shadow drawable. 
  857.      *  
  858.      * @param d 
  859.      *            the new shadow drawable 
  860.      */  
  861.     public void setSecondaryShadowDrawable(Drawable d) {  
  862.         mViewBehind.setSecondaryShadowDrawable(d);  
  863.     }  
  864.   
  865.     /** 
  866.      * Sets the shadow width. 
  867.      *  
  868.      * @param resId 
  869.      *            The dimension resource id to be set as the shadow width. 
  870.      */  
  871.     public void setShadowWidthRes(int resId) {  
  872.         setShadowWidth((int) getResources().getDimension(resId));  
  873.     }  
  874.   
  875.     /** 
  876.      * Sets the shadow width. 
  877.      *  
  878.      * @param pixels 
  879.      *            the new shadow width, in pixels 
  880.      */  
  881.     public void setShadowWidth(int pixels) {  
  882.         mViewBehind.setShadowWidth(pixels);  
  883.     }  
  884.   
  885.     /** 
  886.      * Enables or disables the SlidingMenu's fade in and out 
  887.      *  
  888.      * @param b 
  889.      *            true to enable fade, false to disable it 
  890.      */  
  891.     public void setFadeEnabled(boolean b) {  
  892.         mViewBehind.setFadeEnabled(b);  
  893.     }  
  894.   
  895.     /** 
  896.      * Sets how much the SlidingMenu fades in and out. Fade must be enabled, see 
  897.      * {@link #setFadeEnabled(boolean) setFadeEnabled(boolean)} 
  898.      *  
  899.      * @param f 
  900.      *            the new fade degree, between 0.0f and 1.0f 
  901.      */  
  902.     public void setFadeDegree(float f) {  
  903.         mViewBehind.setFadeDegree(f);  
  904.     }  
  905.   
  906.     /** 
  907.      * Enables or disables whether the selector is drawn 
  908.      *  
  909.      * @param b 
  910.      *            true to draw the selector, false to not draw the selector 
  911.      */  
  912.     public void setSelectorEnabled(boolean b) {  
  913.         mViewBehind.setSelectorEnabled(true);  
  914.     }  
  915.   
  916.     /** 
  917.      * Sets the selected view. The selector will be drawn here 
  918.      *  
  919.      * @param v 
  920.      *            the new selected view 
  921.      */  
  922.     public void setSelectedView(View v) {  
  923.         mViewBehind.setSelectedView(v);  
  924.     }  
  925.   
  926.     /** 
  927.      * Sets the selector drawable. 
  928.      *  
  929.      * @param res 
  930.      *            a resource ID for the selector drawable 
  931.      */  
  932.     public void setSelectorDrawable(int res) {  
  933.         mViewBehind.setSelectorBitmap(BitmapFactory.decodeResource(  
  934.                 getResources(), res));  
  935.     }  
  936.   
  937.     /** 
  938.      * Sets the selector drawable. 
  939.      *  
  940.      * @param b 
  941.      *            the new selector bitmap 
  942.      */  
  943.     public void setSelectorBitmap(Bitmap b) {  
  944.         mViewBehind.setSelectorBitmap(b);  
  945.     }  
  946.   
  947.     /** 
  948.      * Add a View ignored by the Touch Down event when mode is Fullscreen 
  949.      *  
  950.      * @param v 
  951.      *            a view to be ignored 
  952.      */  
  953.     public void addIgnoredView(View v) {  
  954.         mViewAbove.addIgnoredView(v);  
  955.     }  
  956.   
  957.     /** 
  958.      * Remove a View ignored by the Touch Down event when mode is Fullscreen 
  959.      *  
  960.      * @param v 
  961.      *            a view not wanted to be ignored anymore 
  962.      */  
  963.     public void removeIgnoredView(View v) {  
  964.         mViewAbove.removeIgnoredView(v);  
  965.     }  
  966.   
  967.     /** 
  968.      * Clear the list of Views ignored by the Touch Down event when mode is 
  969.      * Fullscreen 
  970.      */  
  971.     public void clearIgnoredViews() {  
  972.         mViewAbove.clearIgnoredViews();  
  973.     }  
  974.   
  975.     /** 
  976.      * Sets the OnOpenListener. {@link OnOpenListener#onOpen() 
  977.      * OnOpenListener.onOpen()} will be called when the SlidingMenu is opened 
  978.      *  
  979.      * @param listener 
  980.      *            the new OnOpenListener 
  981.      */  
  982.     public void setOnOpenListener(OnOpenListener listener) {  
  983.         // mViewAbove.setOnOpenListener(listener);  
  984.         mOpenListener = listener;  
  985.     }  
  986.   
  987.     /** 
  988.      * Sets the OnOpenListner for secondary menu {@link OnOpenListener#onOpen() 
  989.      * OnOpenListener.onOpen()} will be called when the secondary SlidingMenu is 
  990.      * opened 
  991.      *  
  992.      * @param listener 
  993.      *            the new OnOpenListener 
  994.      */  
  995.   
  996.     public void setSecondaryOnOpenListner(OnOpenListener listener) {  
  997.         mSecondaryOpenListner = listener;  
  998.     }  
  999.   
  1000.     /** 
  1001.      * Sets the OnCloseListener. {@link OnCloseListener#onClose() 
  1002.      * OnCloseListener.onClose()} will be called when any one of the SlidingMenu 
  1003.      * is closed 
  1004.      *  
  1005.      * @param listener 
  1006.      *            the new setOnCloseListener 
  1007.      */  
  1008.     public void setOnCloseListener(OnCloseListener listener) {  
  1009.         // mViewAbove.setOnCloseListener(listener);  
  1010.         mCloseListener = listener;  
  1011.     }  
  1012.   
  1013.     /** 
  1014.      * Sets the OnOpenedListener. {@link OnOpenedListener#onOpened() 
  1015.      * OnOpenedListener.onOpened()} will be called after the SlidingMenu is 
  1016.      * opened 
  1017.      *  
  1018.      * @param listener 
  1019.      *            the new OnOpenedListener 
  1020.      */  
  1021.     public void setOnOpenedListener(OnOpenedListener listener) {  
  1022.         mViewAbove.setOnOpenedListener(listener);  
  1023.     }  
  1024.   
  1025.     /** 
  1026.      * Sets the OnClosedListener. {@link OnClosedListener#onClosed() 
  1027.      * OnClosedListener.onClosed()} will be called after the SlidingMenu is 
  1028.      * closed 
  1029.      *  
  1030.      * @param listener 
  1031.      *            the new OnClosedListener 
  1032.      */  
  1033.     public void setOnClosedListener(OnClosedListener listener) {  
  1034.         mViewAbove.setOnClosedListener(listener);  
  1035.     }  
  1036.   
  1037.     public static class SavedState extends BaseSavedState {  
  1038.   
  1039.         private final int mItem;  
  1040.   
  1041.         public SavedState(Parcelable superState, int item) {  
  1042.             super(superState);  
  1043.             mItem = item;  
  1044.         }  
  1045.   
  1046.         private SavedState(Parcel in) {  
  1047.             super(in);  
  1048.             mItem = in.readInt();  
  1049.         }  
  1050.   
  1051.         public int getItem() {  
  1052.             return mItem;  
  1053.         }  
  1054.   
  1055.         /* 
  1056.          * (non-Javadoc) 
  1057.          *  
  1058.          * @see android.view.AbsSavedState#writeToParcel(android.os.Parcel, int) 
  1059.          */  
  1060.         public void writeToParcel(Parcel out, int flags) {  
  1061.             super.writeToParcel(out, flags);  
  1062.             out.writeInt(mItem);  
  1063.         }  
  1064.   
  1065.         public static final Parcelable.Creator CREATOR = new Parcelable.Creator() {  
  1066.             public SavedState createFromParcel(Parcel in) {  
  1067.                 return new SavedState(in);  
  1068.             }  
  1069.   
  1070.             public SavedState[] newArray(int size) {  
  1071.                 return new SavedState[size];  
  1072.             }  
  1073.         };  
  1074.   
  1075.     }  
  1076.   
  1077.     /* 
  1078.      * (non-Javadoc) 
  1079.      *  
  1080.      * @see android.view.View#onSaveInstanceState() 
  1081.      */  
  1082.     @Override  
  1083.     protected Parcelable onSaveInstanceState() {  
  1084.         Parcelable superState = super.onSaveInstanceState();  
  1085.         SavedState ss = new SavedState(superState, mViewAbove.getCurrentItem());  
  1086.         return ss;  
  1087.     }  
  1088.   
  1089.     /* 
  1090.      * (non-Javadoc) 
  1091.      *  
  1092.      * @see android.view.View#onRestoreInstanceState(android.os.Parcelable) 
  1093.      */  
  1094.     @Override  
  1095.     protected void onRestoreInstanceState(Parcelable state) {  
  1096.         SavedState ss = (SavedState) state;  
  1097.         super.onRestoreInstanceState(ss.getSuperState());  
  1098.         mViewAbove.setCurrentItem(ss.getItem());  
  1099.     }  
  1100.   
  1101.     /* 
  1102.      * (non-Javadoc) 
  1103.      *  
  1104.      * @see android.view.ViewGroup#fitSystemWindows(android.graphics.Rect) 
  1105.      */  
  1106.     @SuppressLint("NewApi")  
  1107.     @Override  
  1108.     protected boolean fitSystemWindows(Rect insets) {  
  1109.         int leftPadding = insets.left;  
  1110.         int rightPadding = insets.right;  
  1111.         int topPadding = insets.top;  
  1112.         int bottomPadding = insets.bottom;  
  1113.         if (!mActionbarOverlay) {  
  1114.             Log.v(TAG, "setting padding!");  
  1115.             setPadding(leftPadding, topPadding, rightPadding, bottomPadding);  
  1116.         }  
  1117.         return true;  
  1118.     }  
  1119.   
  1120.     @TargetApi(Build.VERSION_CODES.HONEYCOMB)  
  1121.     public void manageLayers(float percentOpen) {  
  1122.         if (Build.VERSION.SDK_INT  0.0f && percentOpen  0) {  
  1123.             duration = 4 * Math.round(1000 * Math.abs(distance / velocity));  
  1124.         } else {  
  1125.             final float pageDelta = (float) Math.abs(dx) / width;  
  1126.             duration = (int) ((pageDelta + 1) * 100);  
  1127.             duration = MAX_SETTLE_DURATION;  
  1128.         }  
  1129.         duration = Math.min(duration, MAX_SETTLE_DURATION);  
  1130.   
  1131.         mScroller.startScroll(sx, sy, dx, dy, duration);  
  1132.         invalidate();  
  1133.     }  
  1134.   
  1135.     public void setContent(View v) {  
  1136.         if (mContent != null)   
  1137.             this.removeView(mContent);  
  1138.         mContent = v;  
  1139.         addView(mContent);  
  1140.     }  
  1141.   
  1142.     public View getContent() {  
  1143.         return mContent;  
  1144.     }  
  1145.   
  1146.     public void setCustomViewBehind(CustomViewBehind cvb) {  
  1147.         mViewBehind = cvb;  
  1148.     }  
  1149.   
  1150.     @Override  
  1151.     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {  
  1152.   
  1153.         int width = getDefaultSize(0, widthMeasureSpec);  
  1154.         int height = getDefaultSize(0, heightMeasureSpec);  
  1155.         setMeasuredDimension(width, height);  
  1156.   
  1157.         final int contentWidth = getChildMeasureSpec(widthMeasureSpec, 0, width);  
  1158.         final int contentHeight = getChildMeasureSpec(heightMeasureSpec, 0, height);  
  1159.         mContent.measure(contentWidth, contentHeight);  
  1160.     }  
  1161.   
  1162.     @Override  
  1163.     protected void onSizeChanged(int w, int h, int oldw, int oldh) {  
  1164.         super.onSizeChanged(w, h, oldw, oldh);  
  1165.         // Make sure scroll position is set correctly.  
  1166.         if (w != oldw) {  
  1167.             // [ChrisJ] - This fixes the onConfiguration change for orientation issue..  
  1168.             // maybe worth having a look why the recomputeScroll pos is screwing  
  1169.             // up?  
  1170.             completeScroll();  
  1171.             scrollTo(getDestScrollX(mCurItem), getScrollY());  
  1172.         }  
  1173.     }  
  1174.   
  1175.     @Override  
  1176.     protected void onLayout(boolean changed, int l, int t, int r, int b) {  
  1177.         final int width = r - l;  
  1178.         final int height = b - t;  
  1179.         mContent.layout(0, 0, width, height);  
  1180.     }  
  1181.   
  1182.     public void setAboveOffset(int i) {  
  1183.         //      RelativeLayout.LayoutParams params = ((RelativeLayout.LayoutParams)mContent.getLayoutParams());  
  1184.         //      params.setMargins(i, params.topMargin, params.rightMargin, params.bottomMargin);  
  1185.         mContent.setPadding(i, mContent.getPaddingTop(),   
  1186.                 mContent.getPaddingRight(), mContent.getPaddingBottom());  
  1187.     }  
  1188.   
  1189.   
  1190.     @Override  
  1191.     public void computeScroll() {  
  1192.         if (!mScroller.isFinished()) {  
  1193.             if (mScroller.computeScrollOffset()) {  
  1194.                 int oldX = getScrollX();  
  1195.                 int oldY = getScrollY();  
  1196.                 int x = mScroller.getCurrX();  
  1197.                 int y = mScroller.getCurrY();  
  1198.   
  1199.                 if (oldX != x || oldY != y) {  
  1200.                     scrollTo(x, y);  
  1201.                     pageScrolled(x);  
  1202.                 }  
  1203.   
  1204.                 // Keep on drawing until the animation has finished.  
  1205.                 invalidate();  
  1206.                 return;  
  1207.             }  
  1208.         }  
  1209.   
  1210.         // Done with scroll, clean up state.  
  1211.         completeScroll();  
  1212.     }  
  1213.   
  1214.     private void pageScrolled(int xpos) {  
  1215.         final int widthWithMargin = getWidth();  
  1216.         final int position = xpos / widthWithMargin;  
  1217.         final int offsetPixels = xpos % widthWithMargin;  
  1218.         final float offset = (float) offsetPixels / widthWithMargin;  
  1219.   
  1220.         onPageScrolled(position, offset, offsetPixels);  
  1221.     }  
  1222.   
  1223.     /** 
  1224.      * This method will be invoked when the current page is scrolled, either as part 
  1225.      * of a programmatically initiated smooth scroll or a user initiated touch scroll. 
  1226.      * If you override this method you must call through to the superclass implementation 
  1227.      * (e.g. super.onPageScrolled(position, offset, offsetPixels)) before onPageScrolled 
  1228.      * returns. 
  1229.      * 
  1230.      * @param position Position index of the first page currently being displayed. 
  1231.      *                 Page position+1 will be visible if positionOffset is nonzero. 
  1232.      * @param offset Value from [0, 1) indicating the offset from the page at position. 
  1233.      * @param offsetPixels Value in pixels indicating the offset from position. 
  1234.      */  
  1235.     protected void onPageScrolled(int position, float offset, int offsetPixels) {  
  1236.         if (mOnPageChangeListener != null) {  
  1237.             mOnPageChangeListener.onPageScrolled(position, offset, offsetPixels);  
  1238.         }  
  1239.         if (mInternalPageChangeListener != null) {  
  1240.             mInternalPageChangeListener.onPageScrolled(position, offset, offsetPixels);  
  1241.         }  
  1242.     }  
  1243.   
  1244.     private void completeScroll() {  
  1245.         boolean needPopulate = mScrolling;  
  1246.         if (needPopulate) {  
  1247.             // Done with scroll, no longer want to cache view drawing.  
  1248.             setScrollingCacheEnabled(false);  
  1249.             mScroller.abortAnimation();  
  1250.             int oldX = getScrollX();  
  1251.             int oldY = getScrollY();  
  1252.             int x = mScroller.getCurrX();  
  1253.             int y = mScroller.getCurrY();  
  1254.             if (oldX != x || oldY != y) {  
  1255.                 scrollTo(x, y);  
  1256.             }  
  1257.             if (isMenuOpen()) {  
  1258.                 if (mOpenedListener != null)  
  1259.                     mOpenedListener.onOpened();  
  1260.             } else {  
  1261.                 if (mClosedListener != null)  
  1262.                     mClosedListener.onClosed();  
  1263.             }  
  1264.         }  
  1265.         mScrolling = false;  
  1266.     }  
  1267.   
  1268.     protected int mTouchMode = SlidingMenu.TOUCHMODE_MARGIN;  
  1269.   
  1270.     public void setTouchMode(int i) {  
  1271.         mTouchMode = i;  
  1272.     }  
  1273.   
  1274.     public int getTouchMode() {  
  1275.         return mTouchMode;  
  1276.     }  
  1277.   
  1278.     private boolean thisTouchAllowed(MotionEvent ev) {  
  1279.         int x = (int) (ev.getX() + mScrollX);  
  1280.         if (isMenuOpen()) {  
  1281.             return mViewBehind.menuOpenTouchAllowed(mContent, mCurItem, x);  
  1282.         } else {  
  1283.             switch (mTouchMode) {  
  1284.             case SlidingMenu.TOUCHMODE_FULLSCREEN:  
  1285.                 return !isInIgnoredView(ev);  
  1286.             case SlidingMenu.TOUCHMODE_NONE:  
  1287.                 return false;  
  1288.             case SlidingMenu.TOUCHMODE_MARGIN:  
  1289.                 return mViewBehind.marginTouchAllowed(mContent, x);  
  1290.             }  
  1291.         }  
  1292.         return false;  
  1293.     }  
  1294.   
  1295.     private boolean thisSlideAllowed(float dx) {  
  1296.         boolean allowed = false;  
  1297.         if (isMenuOpen()) {  
  1298.             allowed = mViewBehind.menuOpenSlideAllowed(dx);  
  1299.         } else {  
  1300.             allowed = mViewBehind.menuClosedSlideAllowed(dx);  
  1301.         }  
  1302.         if (DEBUG)  
  1303.             Log.v(TAG, "this slide allowed " + allowed + " dx: " + dx);  
  1304.         return allowed;  
  1305.     }  
  1306.   
  1307.     private int getPointerIndex(MotionEvent ev, int id) {  
  1308.         int activePointerIndex = MotionEventCompat.findPointerIndex(ev, id);  
  1309.         if (activePointerIndex == -1)  
  1310.             mActivePointerId = INVALID_POINTER;  
  1311.         return activePointerIndex;  
  1312.     }  
  1313.   
  1314.     private boolean mQuickReturn = false;  
  1315.   
  1316.     @Override  
  1317.     public boolean onInterceptTouchEvent(MotionEvent ev) {  
  1318.   
  1319.         if (!mEnabled)  
  1320.             return false;  
  1321.   
  1322.         final int action = ev.getAction() & MotionEventCompat.ACTION_MASK;  
  1323.   
  1324.         if (DEBUG)  
  1325.             if (action == MotionEvent.ACTION_DOWN)  
  1326.                 Log.v(TAG, "Received ACTION_DOWN");  
  1327.   
  1328.         if (action == MotionEvent.ACTION_CANCEL || action == MotionEvent.ACTION_UP  
  1329.                 || (action != MotionEvent.ACTION_DOWN && mIsUnableToDrag)) {  
  1330.             endDrag();  
  1331.             return false;  
  1332.         }  
  1333.   
  1334.         switch (action) {  
  1335.         case MotionEvent.ACTION_MOVE:  
  1336.             determineDrag(ev);  
  1337.             break;  
  1338.         case MotionEvent.ACTION_DOWN:  
  1339.             int index = MotionEventCompat.getActionIndex(ev);  
  1340.             mActivePointerId = MotionEventCompat.getPointerId(ev, index);  
  1341.             if (mActivePointerId == INVALID_POINTER)  
  1342.                 break;  
  1343.             mLastMotionX = mInitialMotionX = MotionEventCompat.getX(ev, index);  
  1344.             mLastMotionY = MotionEventCompat.getY(ev, index);  
  1345.             if (thisTouchAllowed(ev)) {  
  1346.                 mIsBeingDragged = false;  
  1347.                 mIsUnableToDrag = false;  
  1348.                 if (isMenuOpen() && mViewBehind.menuTouchInQuickReturn(mContent, mCurItem, ev.getX() + mScrollX)) {  
  1349.                     mQuickReturn = true;  
  1350.                 }  
  1351.             } else {  
  1352.                 mIsUnableToDrag = true;  
  1353.             }  
  1354.             break;  
  1355.         case MotionEventCompat.ACTION_POINTER_UP:  
  1356.             onSecondaryPointerUp(ev);  
  1357.             break;  
  1358.         }  
  1359.   
  1360.         if (!mIsBeingDragged) {  
  1361.             if (mVelocityTracker == null) {  
  1362.                 mVelocityTracker = VelocityTracker.obtain();  
  1363.             }  
  1364.             mVelocityTracker.addMovement(ev);  
  1365.         }  
  1366.         return mIsBeingDragged || mQuickReturn;  
  1367.     }  
  1368.   
  1369.   
  1370.     @Override  
  1371.     public boolean onTouchEvent(MotionEvent ev) {  
  1372.   
  1373.         if (!mEnabled)  
  1374.             return false;  
  1375.   
  1376.         if (!mIsBeingDragged && !thisTouchAllowed(ev))  
  1377.             return false;  
  1378.   
  1379.         //      if (!mIsBeingDragged && !mQuickReturn)  
  1380.         //          return false;  
  1381.   
  1382.         final int action = ev.getAction();  
  1383.   
  1384.         if (mVelocityTracker == null) {  
  1385.             mVelocityTracker = VelocityTracker.obtain();  
  1386.         }  
  1387.         mVelocityTracker.addMovement(ev);  
  1388.   
  1389.         switch (action & MotionEventCompat.ACTION_MASK) {  
  1390.         case MotionEvent.ACTION_DOWN:  
  1391.             /* 
  1392.              * If being flinged and user touches, stop the fling. isFinished 
  1393.              * will be false if being flinged. 
  1394.              */  
  1395.             completeScroll();  
  1396.   
  1397.             // Remember where the motion event started  
  1398.             int index = MotionEventCompat.getActionIndex(ev);  
  1399.             mActivePointerId = MotionEventCompat.getPointerId(ev, index);  
  1400.             mLastMotionX = mInitialMotionX = ev.getX();  
  1401.             break;  
  1402.         case MotionEvent.ACTION_MOVE:  
  1403.             if (!mIsBeingDragged) {   
  1404.                 determineDrag(ev);  
  1405.                 if (mIsUnableToDrag)  
  1406.                     return false;  
  1407.             }  
  1408.             if (mIsBeingDragged) {  
  1409.                 // Scroll to follow the motion event  
  1410.                 final int activePointerIndex = getPointerIndex(ev, mActivePointerId);  
  1411.                 if (mActivePointerId == INVALID_POINTER)  
  1412.                     break;  
  1413.                 final float x = MotionEventCompat.getX(ev, activePointerIndex);  
  1414.                 final float deltaX = mLastMotionX - x;  
  1415.                 mLastMotionX = x;  
  1416.                 float oldScrollX = getScrollX();  
  1417.                 float scrollX = oldScrollX + deltaX;  
  1418.                 final float leftBound = getLeftBound();  
  1419.                 final float rightBound = getRightBound();  
  1420.                 if (scrollX  rightBound) {  
  1421.                     scrollX = rightBound;  
  1422.                 }  
  1423.                 // Don't lose the rounded component  
  1424.                 mLastMotionX += scrollX - (int) scrollX;  
  1425.                 scrollTo((int) scrollX, getScrollY());  
  1426.                 pageScrolled((int) scrollX);  
  1427.             }  
  1428.             break;  
  1429.         case MotionEvent.ACTION_UP:  
  1430.             if (mIsBeingDragged) {  
  1431.                 final VelocityTracker velocityTracker = mVelocityTracker;  
  1432.                 velocityTracker.computeCurrentVelocity(1000, mMaximumVelocity);  
  1433.                 int initialVelocity = (int) VelocityTrackerCompat.getXVelocity(  
  1434.                         velocityTracker, mActivePointerId);  
  1435.                 final int scrollX = getScrollX();  
  1436.                 final float pageOffset = (float) (scrollX - getDestScrollX(mCurItem)) / getBehindWidth();  
  1437.                 final int activePointerIndex = getPointerIndex(ev, mActivePointerId);  
  1438.                 if (mActivePointerId != INVALID_POINTER) {  
  1439.                     final float x = MotionEventCompat.getX(ev, activePointerIndex);  
  1440.                     final int totalDelta = (int) (x - mInitialMotionX);  
  1441.                     int nextPage = determineTargetPage(pageOffset, initialVelocity, totalDelta);  
  1442.                     setCurrentItemInternal(nextPage, true, true, initialVelocity);  
  1443.                 } else {      
  1444.                     setCurrentItemInternal(mCurItem, true, true, initialVelocity);  
  1445.                 }  
  1446.                 mActivePointerId = INVALID_POINTER;  
  1447.                 endDrag();  
  1448.             } else if (mQuickReturn && mViewBehind.menuTouchInQuickReturn(mContent, mCurItem, ev.getX() + mScrollX)) {  
  1449.                 // close the menu  
  1450.                 setCurrentItem(1);  
  1451.                 endDrag();  
  1452.             }  
  1453.             break;  
  1454.         case MotionEvent.ACTION_CANCEL:  
  1455.             if (mIsBeingDragged) {  
  1456.                 setCurrentItemInternal(mCurItem, true, true);  
  1457.                 mActivePointerId = INVALID_POINTER;  
  1458.                 endDrag();  
  1459.             }  
  1460.             break;  
  1461.         case MotionEventCompat.ACTION_POINTER_DOWN: {  
  1462.             final int indexx = MotionEventCompat.getActionIndex(ev);  
  1463.             mLastMotionX = MotionEventCompat.getX(ev, indexx);  
  1464.             mActivePointerId = MotionEventCompat.getPointerId(ev, indexx);  
  1465.             break;  
  1466.         }  
  1467.         case MotionEventCompat.ACTION_POINTER_UP:  
  1468.             onSecondaryPointerUp(ev);  
  1469.             int pointerIndex = getPointerIndex(ev, mActivePointerId);  
  1470.             if (mActivePointerId == INVALID_POINTER)  
  1471.                 break;  
  1472.             mLastMotionX = MotionEventCompat.getX(ev, pointerIndex);  
  1473.             break;  
  1474.         }  
  1475.         return true;  
  1476.     }  
  1477.       
  1478.     private void determineDrag(MotionEvent ev) {  
  1479.         final int activePointerId = mActivePointerId;  
  1480.         final int pointerIndex = getPointerIndex(ev, activePointerId);  
  1481.         if (activePointerId == INVALID_POINTER || pointerIndex == INVALID_POINTER)  
  1482.             return;  
  1483.         final float x = MotionEventCompat.getX(ev, pointerIndex);  
  1484.         final float dx = x - mLastMotionX;  
  1485.         final float xDiff = Math.abs(dx);  
  1486.         final float y = MotionEventCompat.getY(ev, pointerIndex);  
  1487.         final float dy = y - mLastMotionY;  
  1488.         final float yDiff = Math.abs(dy);  
  1489.         if (xDiff > (isMenuOpen()?mTouchSlop/2:mTouchSlop) && xDiff > yDiff && thisSlideAllowed(dx)) {          
  1490.             startDrag();  
  1491.             mLastMotionX = x;  
  1492.             mLastMotionY = y;  
  1493.             setScrollingCacheEnabled(true);  
  1494.             // TODO add back in touch slop check  
  1495.         } else if (xDiff > mTouchSlop) {  
  1496.             mIsUnableToDrag = true;  
  1497.         }  
  1498.     }  
  1499.   
  1500.     @Override  
  1501.     public void scrollTo(int x, int y) {  
  1502.         super.scrollTo(x, y);  
  1503.         mScrollX = x;  
  1504.         mViewBehind.scrollBehindTo(mContent, x, y);   
  1505.        ((SlidingMenu)getParent()).manageLayers(getPercentOpen());  
  1506.           
  1507.         if (mTransformer != null) {  
  1508.             invalidate();  
  1509.         }  
  1510.     }  
  1511.   
  1512.     private int determineTargetPage(float pageOffset, int velocity, int deltaX) {  
  1513.         int targetPage = mCurItem;  
  1514.         if (Math.abs(deltaX) > mFlingDistance && Math.abs(velocity) > mMinimumVelocity) {  
  1515.             if (velocity > 0 && deltaX > 0) {  
  1516.                 targetPage -= 1;  
  1517.             } else if (velocity 
关注
打赏
1688896170
查看更多评论
0.0567s