国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 系統(tǒng) > Android > 正文

Android仿Boss直聘文本日期混合滾輪選擇器示例

2019-12-12 01:09:21
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

1、需求分析

GitHub上面有一款iOS風(fēng)格的滾輪選擇器Android-PickerView,它分為時(shí)間選擇器代碼TimePickerView和選項(xiàng)選擇器OptionsPickerView,不但可以選擇時(shí)間日期,可以選擇我們自定義的數(shù)據(jù),比如性別、年齡等。我一直都用它。直到最近遇到了一個(gè)需求,它的選項(xiàng)里面既有文字也有時(shí)間,大體效果如Boss直聘添加項(xiàng)目經(jīng)驗(yàn)中的時(shí)間選擇功能:

從圖中我們可以看出,除了常規(guī)的年份和月份的選擇,選項(xiàng)中還包含了文本。其中,最新的時(shí)間是“至今”,而最早可供選擇的時(shí)間則是“1900以前”。所以看起來(lái)似乎TimePickerViewOptionsPickerView都無(wú)法實(shí)現(xiàn)這個(gè)功能。我們都沮喪地認(rèn)為這下要么得自定義控件,要么得修改Android-PickerView這個(gè)庫(kù)了。但我轉(zhuǎn)念一想,為什么要把“時(shí)間選擇”和“選項(xiàng)選擇”分得那么開(kāi)呢?時(shí)間選擇其實(shí)也是選項(xiàng)選擇的一種嘛。比如我要選擇2017年12月,那就是從年份中選擇2017,從月份中選擇12。只要設(shè)置好一級(jí)選項(xiàng)和二級(jí)選項(xiàng)就可以了。

2、選項(xiàng)結(jié)構(gòu)分析

有了思路之后,我們來(lái)分析一下選項(xiàng)的數(shù)據(jù)結(jié)構(gòu)。年份可以分為3種情況:

  1. 最新年份,其實(shí)也是最新的時(shí)間:“至今”;
  2. 常規(guī)的年份:1990~當(dāng)前年份(2018);
  3. 最早的年份,也就是最早的時(shí)間:“1990以前”。

我在Boss直聘的基礎(chǔ)上加了一些限制:當(dāng)前年份下對(duì)應(yīng)的可供選擇的月份范圍只能是從月到當(dāng)前月份,比如現(xiàn)在是2018年2月,那么選好年份為2018后,月份就只能選擇1和2。這樣一來(lái),月份就有四種情況了:

  1. 最新月份:“至今”;
  2. 當(dāng)前年份下對(duì)應(yīng)的月份范圍:1~當(dāng)前月份;
  3. 完整的月份,即1~12;
  4. 最早月份:“1990以前”。

可以總結(jié)為如下的表格:

年份 月份
最新年份“至今” 最新年份“至今”
當(dāng)前年份 1~當(dāng)前月份
1990~當(dāng)前年份-1 月份1~12
最早年份“1990以前” 最早月份“1990以前”

3、書(shū)寫(xiě)代碼

在開(kāi)始寫(xiě)代碼之前,我建議你先去GitHub上看看Android-PickerView的使用用法,它使用了構(gòu)造者模式,用起來(lái)很簡(jiǎn)單。

現(xiàn)在,就開(kāi)始寫(xiě)我們的代碼了。

3.1 界面布局

布局就是一個(gè)按鈕,點(diǎn)擊后彈出滾輪選擇器,選好后點(diǎn)擊確認(rèn)即將數(shù)據(jù)在TextView上顯示出來(lái)。

<?xml version="1.0" encoding="utf-8"?><LinearLayout  xmlns:android="http://schemas.android.com/apk/res/android"  xmlns:app="http://schemas.android.com/apk/res-auto"  android:orientation="vertical"  android:layout_width="match_parent"  android:layout_height="match_parent">  <Button    android:textAllCaps="false"    android:text="顯示PickerView"    android:onClick="showPickerView"    android:layout_width="match_parent"    android:layout_height="wrap_content" />  <TextView    android:id="@+id/tv_time"    android:textSize="16sp"    android:gravity="center"    android:layout_width="match_parent"    android:layout_height="wrap_content" /></LinearLayout>

3.2 Activity代碼

借助強(qiáng)大的PickerView,我們實(shí)現(xiàn)起來(lái)很簡(jiǎn)單,請(qǐng)看如下的代碼:

public class MultipleOptionActivity extends AppCompatActivity {  private TextView tvTime;  /**   * 完整的月份數(shù)據(jù)1~12   */  private List<String> monthList = new ArrayList<>();  /**   * 滾輪選擇器中年份的選項(xiàng)數(shù)據(jù)   */  private List<String> optionYears = new ArrayList<>();  /**   * 滾輪選擇器中月份的選項(xiàng)數(shù)據(jù)   */  private List<List<String>> optionMonths = new ArrayList<>();  @Override  protected void onCreate(Bundle savedInstanceState) {    super.onCreate(savedInstanceState);    setContentView(R.layout.activity_multiple_option);    tvTime = (TextView) findViewById(R.id.tv_time);    initData();  }  /**   * 初始化數(shù)據(jù)   */  private void initData() {    //設(shè)置完整的月份數(shù)據(jù),即1~12    for (int i = 1; i <= 12; i++) {      monthList.add(String.valueOf(i));    }    Calendar calendar = Calendar.getInstance();    int curYear = calendar.get(Calendar.YEAR);    //月份獲取到的數(shù)據(jù)是0~11,所以要加1    int curMonth = calendar.get(Calendar.MONTH) + 1;    for (int i = curYear + 1; i >= 1989; i--) {      //對(duì)應(yīng)年份的月份數(shù)據(jù)集合      List<String> tempMonths = new ArrayList<>();      if (i == curYear + 1) {        //設(shè)置最新時(shí)間“至今”        optionYears.add("至今");        tempMonths.add("至今");        optionMonths.add(tempMonths);      } else if (i == curYear) {        //設(shè)置當(dāng)前年份及其對(duì)應(yīng)的月份        optionYears.add(String.valueOf(i));        for (int j = 1; j <= curMonth; j++) {          tempMonths.add(String.valueOf(j));        }        optionMonths.add(tempMonths);      } else if (i == 1989) {        //設(shè)置最早時(shí)間“1900以前”        optionYears.add("1990以前");        tempMonths.add("1990以前");        optionMonths.add(tempMonths);      } else {        //設(shè)置常規(guī)時(shí)間        optionYears.add(String.valueOf(i));        optionMonths.add(monthList);      }    }  }  /**   * 顯示滾輪   *   * @param view   */  public void showPickerView(View view) {    OptionsPickerView multipleOp = new OptionsPickerView.Builder(this, new OptionsPickerView.OnOptionsSelectListener() {      @Override      public void onOptionsSelect(int options1, int options2, int options3, View v) {        if (options1 == 0 || options1 == optionYears.size() - 1) {          //選中最新和最早時(shí)間時(shí)直接顯示文字,不需要拼接月份          tvTime.setText(optionYears.get(options1));        } else {          //常規(guī)的時(shí)間,需要拼接年份和月份          tvTime.setText(new StringBuffer(optionYears.get(options1)).append("―").append(monthList.get(options2)));        }      }    }).setTitleText("請(qǐng)選擇時(shí)間")        .build();    multipleOp.setPicker(optionYears, optionMonths);    multipleOp.show();  }}

代碼很少,注釋我也寫(xiě)得很清楚了,相信大家很容易理解。我們重點(diǎn)關(guān)注OptionsPickerViewsetPicker方法,它可以傳入三個(gè)參數(shù),每個(gè)參數(shù)都是集合,但每個(gè)參數(shù)的類(lèi)型都不同。第一個(gè)參數(shù)是List,第二個(gè)參數(shù)是List<List>,第三個(gè)參數(shù)是List<List<list>>。看到這里你就明白了,我們每個(gè)年份對(duì)應(yīng)的月份數(shù)據(jù)就是一個(gè)集合(當(dāng)然,集合大小不相同),比如年份2017,對(duì)應(yīng)的月份就是有著12個(gè)元素的集合。理清楚這一點(diǎn)之后,也就理解initData方法里面對(duì)數(shù)據(jù)的設(shè)置了。

最后在TextView中顯示數(shù)據(jù)時(shí)自然也要分類(lèi)了,對(duì)于“至今”和“1990以前”我們至今顯示文本,其他的再拼接一下,看起來(lái)像是時(shí)間就行了。

看看我們最后實(shí)現(xiàn)的效果圖:

4、總結(jié)

在項(xiàng)目中使用一些好的第三方庫(kù)是可以大大節(jié)省我們的開(kāi)發(fā)時(shí)間的,但是在使用過(guò)程中也要靈活一點(diǎn)。比如我們?cè)谝粋€(gè)頁(yè)面中需要多次用到滾輪選擇器(比如選擇開(kāi)始時(shí)間和結(jié)束時(shí)間),那么每次都要設(shè)置一遍滾輪的樣式和寫(xiě)一次點(diǎn)擊事件也太麻煩了。這時(shí),我們就可以將滾輪樣式的設(shè)置代碼抽取出來(lái):

  /**  * 設(shè)置滾輪樣式  * @return  */  private OptionsPickerView.Builder createBuilder(){    OptionsPickerView.Builder builder = new OptionsPickerView.Builder(MultipleOptionActivity.this,this)        .setBgColor(ContextCompat.getColor(this,R.color.colorAccent))        .setSubmitText("確定")        .setCancelText("取消");    //下面可以繼續(xù)設(shè)置樣式    return builder;  }

然后顯示滾輪的時(shí)候只要這樣寫(xiě):

OptionsPickerView op = createBuilder().build();op.setPicker(數(shù)據(jù)1,數(shù)據(jù)2);op.show();

點(diǎn)擊事件也可以封裝起來(lái),讓我們的Activity繼承OptionsPickerView.OnOptionsSelectListener,然后實(shí)現(xiàn)點(diǎn)擊事件:

  /**   * 滾輪的監(jiān)聽(tīng)事件   * @param options1   * @param options2   * @param options3   * @param v   */  @Override  public void onOptionsSelect(int options1, int options2, int options3, View v) {    switch (v.getId()){      //根據(jù)所點(diǎn)擊的控件Id來(lái)區(qū)分點(diǎn)擊事件      case R.id.btn_show:        break;      default:        break;    }  }

那么OptionsPickerView怎么獲取到點(diǎn)擊View的id的呢?我們?cè)谡{(diào)用show方法的時(shí)候傳入點(diǎn)擊View的對(duì)象就可以了。以上是我個(gè)人的一點(diǎn)心得,希望對(duì)大家有所幫助。

最后給一下源碼吧:源碼

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持武林網(wǎng)。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 天镇县| 沙河市| 凌海市| 台州市| 丹巴县| 台州市| 三江| 宜兰县| 刚察县| 景东| 北宁市| 蓬莱市| 油尖旺区| 十堰市| 银川市| 驻马店市| 兴城市| 潢川县| 永昌县| 宁乡县| 广元市| 会东县| 大冶市| 辽宁省| 孟州市| 慈溪市| 长治县| 武冈市| 周口市| 英吉沙县| 永川市| 长海县| 平阴县| 宣化县| 龙里县| 家居| 济源市| 遂昌县| 吉木乃县| 扎囊县| 阳信县|