ViewPager + HorizontalScrollView 实现可滚动的标签栏
来源:互联网 发布:商品价格比较软件 编辑:程序博客网 时间:2024/06/07 08:41
工程目录结构如下图所示
运行效果如下:
主界面布局文件activity_main.xml
- <RelativeLayout 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"
- tools:context=".MainActivity" >
- <RelativeLayout
- android:id="@+id/rl_tab"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:background="#F2F2F2" >
- <com.example.playtabtest.view.SyncHorizontalScrollView
- android:id="@+id/mHsv"
- android:layout_width="fill_parent"
- android:layout_height="40dip"
- android:fadingEdge="none"
- android:scrollbars="none" >
- <RelativeLayout
- android:id="@+id/rl_nav"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:layout_gravity="top"
- android:background="#5AB0EB" >
- <RadioGroup
- android:id="@+id/rg_nav_content"
- android:layout_width="fill_parent"
- android:layout_height="38dip"
- android:layout_alignParentTop="true"
- android:background="#F2F2F2"
- android:orientation="horizontal" >
- </RadioGroup>
- <ImageView
- android:id="@+id/iv_nav_indicator"
- android:layout_width="1dip"
- android:layout_height="5dip"
- android:layout_alignParentBottom="true"
- android:background="#5AB0EB"
- android:contentDescription="@string/nav_desc"
- android:scaleType="matrix" />
- </RelativeLayout>
- </com.example.playtabtest.view.SyncHorizontalScrollView>
- <ImageView
- android:id="@+id/iv_nav_left"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_alignParentLeft="true"
- android:layout_centerVertical="true"
- android:contentDescription="@string/nav_desc"
- android:paddingBottom="1dip"
- android:src="@drawable/iv_navagation_scroll_left"
- android:visibility="gone" >
- </ImageView>
- <ImageView
- android:id="@+id/iv_nav_right"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_alignParentRight="true"
- android:layout_centerVertical="true"
- android:contentDescription="@string/nav_desc"
- android:paddingBottom="1dip"
- android:src="@drawable/iv_navagation_scroll_right"
- android:visibility="visible" >
- </ImageView>
- </RelativeLayout>
- <android.support.v4.view.ViewPager
- android:id="@+id/mViewPager"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_alignParentBottom="true"
- android:layout_below="@id/rl_tab"
- android:layout_gravity="center"
- android:background="#ffffff"
- android:flipInterval="30"
- android:persistentDrawingCache="animation" />
- </RelativeLayout>
MainActivity.java
- package com.example.playtabtest;
- import android.os.Bundle;
- import android.support.v4.app.Fragment;
- import android.support.v4.app.FragmentActivity;
- import android.support.v4.app.FragmentManager;
- import android.support.v4.app.FragmentPagerAdapter;
- import android.support.v4.view.ViewPager;
- import android.support.v4.view.ViewPager.OnPageChangeListener;
- import android.util.DisplayMetrics;
- import android.view.LayoutInflater;
- import android.view.Menu;
- import android.view.ViewGroup.LayoutParams;
- import android.view.animation.LinearInterpolator;
- import android.view.animation.TranslateAnimation;
- import android.widget.ImageView;
- import android.widget.RadioButton;
- import android.widget.RadioGroup;
- import android.widget.RadioGroup.OnCheckedChangeListener;
- import android.widget.RelativeLayout;
- import com.example.playtabtest.fragment.CommonUIFragment;
- import com.example.playtabtest.fragment.LaunchUIFragment;
- import com.example.playtabtest.view.SyncHorizontalScrollView;
- public class MainActivity extends FragmentActivity {
- public static final String ARGUMENTS_NAME = "arg";
- private RelativeLayout rl_nav;
- private SyncHorizontalScrollView mHsv;
- private RadioGroup rg_nav_content;
- private ImageView iv_nav_indicator;
- private ImageView iv_nav_left;
- private ImageView iv_nav_right;
- private ViewPager mViewPager;
- private int indicatorWidth;
- public static String[] tabTitle = { "选项1", "选项2", "选项3", "选项4", "选项5" }; //标题
- private LayoutInflater mInflater;
- private TabFragmentPagerAdapter mAdapter;
- private int currentIndicatorLeft = 0;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- findViewById();
- initView();
- setListener();
- }
- private void setListener() {
- mViewPager.setOnPageChangeListener(new OnPageChangeListener() {
- @Override
- public void onPageSelected(int position) {
- if(rg_nav_content!=null && rg_nav_content.getChildCount()>position){
- ((RadioButton)rg_nav_content.getChildAt(position)).performClick();
- }
- }
- @Override
- public void onPageScrolled(int arg0, float arg1, int arg2) {
- }
- @Override
- public void onPageScrollStateChanged(int arg0) {
- }
- });
- rg_nav_content.setOnCheckedChangeListener(new OnCheckedChangeListener() {
- @Override
- public void onCheckedChanged(RadioGroup group, int checkedId) {
- if(rg_nav_content.getChildAt(checkedId)!=null){
- TranslateAnimation animation = new TranslateAnimation(
- currentIndicatorLeft ,
- ((RadioButton) rg_nav_content.getChildAt(checkedId)).getLeft(), 0f, 0f);
- animation.setInterpolator(new LinearInterpolator());
- animation.setDuration(100);
- animation.setFillAfter(true);
- //执行位移动画
- iv_nav_indicator.startAnimation(animation);
- mViewPager.setCurrentItem(checkedId); //ViewPager 跟随一起 切换
- //记录当前 下标的距最左侧的 距离
- currentIndicatorLeft = ((RadioButton) rg_nav_content.getChildAt(checkedId)).getLeft();
- mHsv.smoothScrollTo(
- (checkedId > 1 ? ((RadioButton) rg_nav_content.getChildAt(checkedId)).getLeft() : 0) - ((RadioButton) rg_nav_content.getChildAt(2)).getLeft(), 0);
- }
- }
- });
- }
- private void initView() {
- DisplayMetrics dm = new DisplayMetrics();
- getWindowManager().getDefaultDisplay().getMetrics(dm);
- indicatorWidth = dm.widthPixels / 4;
- LayoutParams cursor_Params = iv_nav_indicator.getLayoutParams();
- cursor_Params.width = indicatorWidth;// 初始化滑动下标的宽
- iv_nav_indicator.setLayoutParams(cursor_Params);
- mHsv.setSomeParam(rl_nav, iv_nav_left, iv_nav_right, this);
- //获取布局填充器
- mInflater = (LayoutInflater)this.getSystemService(LAYOUT_INFLATER_SERVICE);
- //另一种方式获取
- // LayoutInflater mInflater = LayoutInflater.from(this);
- initNavigationHSV();
- mAdapter = new TabFragmentPagerAdapter(getSupportFragmentManager());
- mViewPager.setAdapter(mAdapter);
- }
- private void initNavigationHSV() {
- rg_nav_content.removeAllViews();
- for(int i=0;i<tabTitle.length;i++){
- RadioButton rb = (RadioButton) mInflater.inflate(R.layout.nav_radiogroup_item, null);
- rb.setId(i);
- rb.setText(tabTitle[i]);
- rb.setLayoutParams(new LayoutParams(indicatorWidth,
- LayoutParams.MATCH_PARENT));
- rg_nav_content.addView(rb);
- }
- }
- private void findViewById() {
- rl_nav = (RelativeLayout) findViewById(R.id.rl_nav);
- mHsv = (SyncHorizontalScrollView) findViewById(R.id.mHsv);
- rg_nav_content = (RadioGroup) findViewById(R.id.rg_nav_content);
- iv_nav_indicator = (ImageView) findViewById(R.id.iv_nav_indicator);
- iv_nav_left = (ImageView) findViewById(R.id.iv_nav_left);
- iv_nav_right = (ImageView) findViewById(R.id.iv_nav_right);
- mViewPager = (ViewPager) findViewById(R.id.mViewPager);
- }
- @Override
- public boolean onCreateOptionsMenu(Menu menu) {
- getMenuInflater().inflate(R.menu.main, menu);
- return true;
- }
- public static class TabFragmentPagerAdapter extends FragmentPagerAdapter{
- public TabFragmentPagerAdapter(FragmentManager fm) {
- super(fm);
- }
- @Override
- public Fragment getItem(int arg0) {
- Fragment ft = null;
- switch (arg0) {
- case 0:
- ft = new LaunchUIFragment();
- break;
- default:
- ft = new CommonUIFragment();
- Bundle args = new Bundle();
- args.putString(ARGUMENTS_NAME, tabTitle[arg0]);
- ft.setArguments(args);
- break;
- }
- return ft;
- }
- @Override
- public int getCount() {
- return tabTitle.length;
- }
- }
- }
SyncHorizontalScrollView.java
- package com.example.playtabtest.view;
- import android.app.Activity;
- import android.content.Context;
- import android.util.AttributeSet;
- import android.util.DisplayMetrics;
- import android.view.View;
- import android.widget.HorizontalScrollView;
- import android.widget.ImageView;
- public class SyncHorizontalScrollView extends HorizontalScrollView {
- private View view;
- private ImageView leftImage;
- private ImageView rightImage;
- private int windowWitdh = 0;
- private Activity mContext;
- public void setSomeParam(View view, ImageView leftImage,
- ImageView rightImage, Activity context) {
- this.mContext = context;
- this.view = view;
- this.leftImage = leftImage;
- this.rightImage = rightImage;
- DisplayMetrics dm = new DisplayMetrics();
- this.mContext.getWindowManager().getDefaultDisplay().getMetrics(dm);
- windowWitdh = dm.widthPixels;
- }
- public SyncHorizontalScrollView(Context context) {
- super(context);
- // TODO Auto-generated constructor stub
- }
- public SyncHorizontalScrollView(Context context, AttributeSet attrs) {
- super(context, attrs);
- // TODO Auto-generated constructor stub
- }
- // 显示和隐藏左右两边的箭头
- public void showAndHideArrow() {
- if (!mContext.isFinishing() && view != null) {
- this.measure(0, 0);
- if (windowWitdh >= this.getMeasuredWidth()) {
- leftImage.setVisibility(View.GONE);
- rightImage.setVisibility(View.GONE);
- } else {
- if (this.getLeft() == 0) {
- leftImage.setVisibility(View.GONE);
- rightImage.setVisibility(View.VISIBLE);
- } else if (this.getRight() == this.getMeasuredWidth()
- - windowWitdh) {
- leftImage.setVisibility(View.VISIBLE);
- rightImage.setVisibility(View.GONE);
- } else {
- leftImage.setVisibility(View.VISIBLE);
- rightImage.setVisibility(View.VISIBLE);
- }
- }
- }
- }
- protected void onScrollChanged(int l, int t, int oldl, int oldt) {
- super.onScrollChanged(l, t, oldl, oldt);
- if (!mContext.isFinishing() && view != null && rightImage != null
- && leftImage != null) {
- if (view.getWidth() <= windowWitdh) {
- leftImage.setVisibility(View.GONE);
- rightImage.setVisibility(View.GONE);
- } else {
- if (l == 0) {
- leftImage.setVisibility(View.GONE);
- rightImage.setVisibility(View.VISIBLE);
- } else if (view.getWidth() - l == windowWitdh) {
- leftImage.setVisibility(View.VISIBLE);
- rightImage.setVisibility(View.GONE);
- } else {
- leftImage.setVisibility(View.VISIBLE);
- rightImage.setVisibility(View.VISIBLE);
- }
- }
- }
- }
- }
以下是两个Fragment的源代码
LaunchUIFragment.java
- package com.example.playtabtest.fragment;
- import android.os.Bundle;
- import android.support.v4.app.Fragment;
- import android.view.LayoutInflater;
- import android.view.View;
- import android.view.ViewGroup;
- import com.example.playtabtest.R;
- public class LaunchUIFragment extends Fragment {
- @Override
- public View onCreateView(LayoutInflater inflater, ViewGroup container,
- Bundle savedInstanceState) {
- View rootView = inflater.inflate(R.layout.fragment_selection_launch, container, false);
- return rootView;
- }
- @Override
- public void onActivityCreated(Bundle savedInstanceState) {
- // TODO Auto-generated method stub
- super.onActivityCreated(savedInstanceState);
- }
- }
CommonUIFragment.java
- package com.example.playtabtest.fragment;
- import android.os.Bundle;
- import android.support.v4.app.Fragment;
- import android.view.LayoutInflater;
- import android.view.View;
- import android.view.ViewGroup;
- import android.widget.TextView;
- import com.example.playtabtest.MainActivity;
- import com.example.playtabtest.R;
- public class CommonUIFragment extends Fragment {
- @Override
- public View onCreateView(LayoutInflater inflater, ViewGroup container,
- Bundle savedInstanceState) {
- View rootView = inflater.inflate(R.layout.fragment_selection_common, container, false);
- TextView tv_tabName = (TextView) rootView.findViewById(R.id.tv_tabName);
- Bundle bundle = getArguments();
- tv_tabName.setText(bundle.getString(MainActivity.ARGUMENTS_NAME, ""));
- return rootView;
- }
- @Override
- public void onActivityCreated(Bundle savedInstanceState) {
- // TODO Auto-generated method stub
- super.onActivityCreated(savedInstanceState);
- }
- }
两个Fragment的布局文件
fragment_selection_launch.xml
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:orientation="vertical" >
- <TextView android:id="@+id/tv_intro"
- android:layout_width="300dp"
- android:layout_height="wrap_content"
- android:layout_marginTop="20dp"
- android:layout_marginBottom="16dp"
- android:text="@string/launch_intro"/>
- <Button android:id="@+id/bt_click"
- android:layout_width="300dp"
- android:layout_height="wrap_content"
- android:layout_marginBottom="16dp"
- android:text="@string/launch_click_me"/>
- </LinearLayout>
fragment_selection_common.xml
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:orientation="vertical" >
- <TextView
- android:layout_width="300dp"
- android:layout_height="wrap_content"
- android:textSize="18sp"
- android:text="@string/common_intro"
- />
- <TextView
- android:id="@+id/tv_tabName"
- android:layout_marginTop="30dp"
- android:layout_width="wrap_content"
- android:layout_height="30dp"
- android:layout_gravity="center"
- android:textSize="20sp"
- />
- </LinearLayout>
源码下载地址:http://download.csdn.net/detail/fx_sky/5475985
根据上面内容做的整合
效果图:
主要布局文件:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <RelativeLayout android:id="@+id/rl_nav" android:layout_width="fill_parent" android:layout_height="wrap_content" android:layout_gravity="top" > <RadioGroup android:id="@+id/rg_nav_content" android:layout_width="fill_parent" android:layout_height="38dip" android:layout_alignParentTop="true" android:background="#F2F2F2" android:orientation="horizontal" > </RadioGroup> <ImageView android:id="@+id/iv_nav_indicator" android:layout_width="1dip" android:layout_height="5dip" android:layout_alignParentBottom="true" android:background="#FF0000" android:contentDescription="@string/mygo_share" android:scaleType="matrix" /> </RelativeLayout> </LinearLayout>
标签的布局:
sync_nav_radiogroup_item.xml
<?xml version="1.0" encoding="utf-8"?><RadioButton xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="0dip" android:layout_height="fill_parent" android:background="#F2F2F2" android:button="@null" android:checked="true" android:gravity="center" android:textSize="15sp" />
以下是工具类代码:
import android.app.Activity;import android.content.Context;import android.support.v4.view.ViewPager;import android.util.AttributeSet;import android.util.DisplayMetrics;import android.view.LayoutInflater;import android.view.View;import android.view.ViewGroup;import android.view.animation.LinearInterpolator;import android.view.animation.TranslateAnimation;import android.widget.HorizontalScrollView;import android.widget.ImageView;import android.widget.RadioButton;import android.widget.RadioGroup;import android.widget.RadioGroup.OnCheckedChangeListener;public class SyncHorizontalScrollView extends HorizontalScrollView { private View view; private ImageView leftImage; private ImageView rightImage; private int windowWitdh = 0; private Activity mContext; private RadioGroup rg_nav_content; private ImageView iv_nav_indicator; private LayoutInflater mInflater; private int count;// 屏幕显示的标签个数 private int indicatorWidth;// 每个标签所占的宽度 private int currentIndicatorLeft = 0;// 当前所在标签页面的位移 private ViewPager mViewPager; private int scrollX; public SyncHorizontalScrollView(Context context) { super(context); // TODO Auto-generated constructor stub } public SyncHorizontalScrollView(Context context, AttributeSet attrs) { super(context, attrs); // TODO Auto-generated constructor stub } /** * * 方法描述: * @param mViewPager * @param leftImage 左箭頭 * @param rightImage 右箭頭 * @param tabTitle 標籤欄的名稱 * @param count 一頁顯示的標籤個數 * @param context * <pre> * 修改日期 修改人 修改说明 * 2014-2-17 chen 新建 * </pre> */ public void setSomeParam(ViewPager mViewPager, ImageView leftImage, ImageView rightImage, String[] tabTitle, int count, Activity context) { this.mContext = context; this.mViewPager = mViewPager; // this.view = view; mInflater = LayoutInflater.from(context); this.view = mInflater.inflate(R.layout.sync_hsv_item, null); this.addView(view); this.leftImage = leftImage; this.rightImage = rightImage; DisplayMetrics dm = new DisplayMetrics(); context.getWindowManager().getDefaultDisplay().getMetrics(dm); windowWitdh = dm.widthPixels; this.count = count; indicatorWidth = windowWitdh / count; init(tabTitle); } private void init(String[] tabTitle) { rg_nav_content = (RadioGroup) view.findViewById(R.id.rg_nav_content); iv_nav_indicator = (ImageView) view.findViewById(R.id.iv_nav_indicator); initIndicatorWidth(); initNavigationHSV(tabTitle); setListener(); } // 初始化滑动下标的宽 private void initIndicatorWidth() { ViewGroup.LayoutParams cursor_Params = iv_nav_indicator .getLayoutParams(); cursor_Params.width = indicatorWidth; iv_nav_indicator.setLayoutParams(cursor_Params); } // 添加顶部标签 private void initNavigationHSV(String[] tabTitle) { rg_nav_content.removeAllViews(); for (int i = 0; i < tabTitle.length; i++) { RadioButton rb = (RadioButton) mInflater.inflate( R.layout.sync_nav_radiogroup_item, null); rb.setId(i); rb.setText(tabTitle[i]); rb.setLayoutParams(new LayoutParams(indicatorWidth, LayoutParams.MATCH_PARENT)); rg_nav_content.addView(rb); } RadioButton rb = (RadioButton) mInflater.inflate( R.layout.sync_nav_radiogroup_item, null); rg_nav_content.addView(rb); } private void setListener() { rg_nav_content .setOnCheckedChangeListener(new OnCheckedChangeListener() { @Override public void onCheckedChanged(RadioGroup group, int checkedId) { if (rg_nav_content.getChildAt(checkedId) != null) { moveAnimation(checkedId); mViewPager.setCurrentItem(checkedId); // ViewPager // 跟随一起 切换 } } }); } //动画移动效果 private void moveAnimation(int checkedId){ TranslateAnimation animation = new TranslateAnimation(currentIndicatorLeft, indicatorWidth * checkedId,0f, 0f); animation.setInterpolator(new LinearInterpolator()); animation.setDuration(100); animation.setFillAfter(true); // 执行位移动画 iv_nav_indicator.startAnimation(animation); // 记录当前 下标的距最左侧的 距离 currentIndicatorLeft = indicatorWidth * checkedId; scrollX = (checkedId > 1 ? currentIndicatorLeft: 0)- indicatorWidth * 2; this.post(runnable); } // 模拟点击事件 public void performLabelClick(int position) { if (rg_nav_content != null && rg_nav_content.getChildCount() > position) { ((RadioButton) rg_nav_content.getChildAt(position)).performClick(); } } private Runnable runnable = new Runnable() { @Override public void run() { // TODO Auto-generated method stub smoothScrollTo(scrollX, 0); } }; // 显示和隐藏左右两边的箭头 protected void onScrollChanged(int l, int t, int oldl, int oldt) { super.onScrollChanged(l, t, oldl, oldt); if (!mContext.isFinishing() && view != null && rightImage != null && leftImage != null) { if (view.getWidth() <= windowWitdh) { leftImage.setVisibility(View.GONE); rightImage.setVisibility(View.GONE); } else { if (l == 0) { leftImage.setVisibility(View.GONE); rightImage.setVisibility(View.VISIBLE); } else if (view.getWidth() - l == windowWitdh) { leftImage.setVisibility(View.VISIBLE); rightImage.setVisibility(View.GONE); } else { leftImage.setVisibility(View.VISIBLE); rightImage.setVisibility(View.VISIBLE); } } } } public int getIndicatorWidth(){ return indicatorWidth; }}
在调用时,先调用setSomeParam方法将需要用的的控件与数据传入,然后控件内部开始初始化。
由于项目有需求,要在进入此控件使用到的页面时,自动定位到某一个标签,在此需使用View.post方法执行HorizontalScrollView 控件的smoothScrollTo方法,才能确保进入页面后标签已定位。原因是scrollTo方法要等到界面显示完毕才能有效,而view.post方法也是在界面刷新完毕之后才执行。
定义一个performLabelClick方法,让外部调用此类来实现相应的页面跳转即可。
不过此处使用时有一个问题,就是从上一级页面跳转至此页面时,无法固定在最后一个标签位,调试时发现虽然执行了performLabelClick方法模拟点击事件,但是监听事件并没有被响应。暂时没找出问题所在。不过用了一个方法来解决此问题,即在调用initNavigationHSV方法添加标签时,为其多加一个空标签,宽度为0即可,这样即可解决标签栏上无法定位到最后一位的问题,因为真正的最后一位实际上是宽度为0的空标签。
如果大家有遇到此问题或是有解决方法,欢迎告知,谢谢。
附上代码下载地址:
http://www.oschina.net/code/snippet_1409622_33243
- ViewPager + HorizontalScrollView 实现可滚动的标签栏
- 实现HorizontalScrollView随ViewPager滚动的简易方法
- 自定义ViewPager、HorizontalScrollView、Indicator实现可滑动的头部
- HorizontalScrollView 实现 ViewPager 可滑动翻页的tab
- 实现ViewPager+Fragment+头部标签(HorizontalScrollView)
- ViewPager的实现:通过HorizontalScrollView
- ViewPager的实现:通过HorizontalScrollView
- ViewPager + Fragment 实现标签的滚动效果
- HorizontalScrollView 初始化滚动的实现
- android中HorizontalScrollView实现viewpager的效果
- 利用TabLayout+ViewPager+Fragment实现可滚动的Tab页面
- android FragmentTabhost+ViewPager实现可滑动的标签页
- HorizontalScrollView 实现水平滚动
- TabLayout实现viewpager +fragment 标签滚动
- 利用HorizontalScrollView加Viewpager实现多个的导航标题栏
- HorizontalScrollView 使用 实现textview和viewpager的联动
- Android 实现横向标题栏滚动效果(HorizontalScrollView + GridView + Viewpager + 自定义适配器)
- 利用HorizontalScrollView实现水平滚动
- [Err] 1093 - You can't specify target table 'matorder' for update in FROM clause 原句:delete from mat
- 初赛阅读程序写结果部分
- android抓包工具fiddler抓包使用方法 跟踪监控android数据包
- print函数和scanf函数重点
- PHP调用另一文件中的类
- ViewPager + HorizontalScrollView 实现可滚动的标签栏
- Oracle--sqlplus如何设置SQLPlus结果显示的宽度
- 使用信号量实现同步
- linux ssh 使用深度解析(key登录详解)
- Innosetup制作安装包的一些技巧
- cookie:记住用户最后一次的访问时间,并清空指定cookie
- linux waitpid/wait函数用法
- 关于微信公众平台的JS-SDK接口的使用
- 解决MySQL中varchar字段排序,多条数据相同字段提取