Android 日歷控件

a.gif

想寫個日歷選擇的控件,如上圖的效果,我也曾想站在巨人的肩膀上,奈何巨人也是坑哎..

起初,在codeKK上找了個第三方的庫,用起來很爽,然后我Demo了之后就網(wǎng)項目里面依賴了,結(jié)果和原項目的一個庫中的字段有沖突,想著那就該字段吧,轉(zhuǎn)念一想,字段改了豈不是項目中所有用到的這個字段都得改呢.我嫌麻煩,真的嫌麻煩

再說項目依賴的庫真的也有點多,所以我準備自己擼出來一個空間,已知原項目中,已有如下的一個庫.是WheelView

compile 'com.bigkoo:pickerview:2.0.8'

那么好,既然有了個WheelView的庫,那就在這個基礎(chǔ)上開始寫就好了.于是呢,我就真的開始寫了..

我計劃用一個dialogFragment 來彈出來選則框.dialogFragment和dialog的區(qū)別呢,就不多說了.

然后我就開始了

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:orientation="vertical">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal"
        >

        <TextView
            android:id="@+id/cancel"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:paddingBottom="10dp"
            android:paddingLeft="10dp"
            android:paddingRight="10dp"
            android:paddingTop="5dp"
            android:text="取消"
            android:textColor="#99999999"
            android:textSize="16sp"
            />

        <View
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            />

        <TextView
            android:id="@+id/confrim"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:paddingBottom="10dp"
            android:paddingLeft="10dp"
            android:paddingRight="10dp"
            android:paddingTop="5dp"
            android:text="確定"
            android:textColor="#342564"
            android:textSize="16sp"
            />
    </LinearLayout>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal"
        >

        <com.bigkoo.pickerview.lib.WheelView
            android:id="@+id/year"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            >
        </com.bigkoo.pickerview.lib.WheelView>

        <com.bigkoo.pickerview.lib.WheelView
            android:id="@+id/month"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1">
        </com.bigkoo.pickerview.lib.WheelView>

        <com.bigkoo.pickerview.lib.WheelView
            android:id="@+id/day"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            >
        </com.bigkoo.pickerview.lib.WheelView>
    </LinearLayout>
</LinearLayout>

以上是布局文件,其中com.bigkoo.pickerview.lib.WheelView為原項目所依賴的庫,就是滾輪類似的控件.

布局寫完,我開始著手寫dialogFragment了

public class SccDateDialog extends DialogFragment {
    private static final String TAG = "SccDateDialog";
    private TextView confrim, cancel;
    private WheelView year, month, day;
    private List<String> years;
    private List<String> months;
    private List<String> days;
    private String mStringYear;
    private String mStringMonth;
    private String mStringDay;

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.sccdatedialog, null);
        return view;
    }

    @Override
    public void onActivityCreated(Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        initViews();
        initDatas();
        initWheelViews();
        getSelected();
    }

    /**
     * 彈出選擇的時間
     */
    private void getSelected() {
        confrim.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Toast.makeText(getActivity(), mStringYear + "-" + mStringMonth + "-" + mStringDay, Toast.LENGTH_SHORT).show();
                dismiss();
            }
        });
        cancel.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                dismiss();
            }
        });
    }

    /**
     * 給WheelView添加數(shù)據(jù)
     */
    private void initWheelViews() {
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
        String date = sdf.format(new Date(System.currentTimeMillis()));
        String[] strings = date.split("-");

        year.setCurrentItem(years.indexOf(strings[0]));
        Log.e(TAG, "initWheelViews: setCurrentItem(3)--->"+3 );
        mStringYear = years.get(years.indexOf(strings[0]));
        Log.e(TAG, "initWheelViews: getCurrentItem()--->"+year.getCurrentItem() );
        year.setCyclic(false);
        year.setAdapter(new WheelAdapter() {
            @Override
            public int getItemsCount() {
                return years.size();
            }

            @Override
            public Object getItem(int index) {
                return years.get(index);
            }

            @Override
            public int indexOf(Object o) {
                return years.indexOf(o);
            }
        });
        year.setOnItemSelectedListener(new OnItemSelectedListener() {
            @Override
            public void onItemSelected(int index) {
                mStringYear = years.get(year.getCurrentItem());
            }
        });
        month.setCurrentItem(months.indexOf(strings[1]));
        mStringMonth = months.get(months.indexOf(strings[1]));
        month.setCyclic(false);
        month.setAdapter(new WheelAdapter() {
            @Override
            public int getItemsCount() {
                return months.size();
            }

            @Override
            public Object getItem(int index) {
                return months.get(index);
            }

            @Override
            public int indexOf(Object o) {
                return months.indexOf(o);
            }
        });
        month.setOnItemSelectedListener(new OnItemSelectedListener() {
            @Override
            public void onItemSelected(int index) {
                mStringMonth = months.get(month.getCurrentItem());
            }
        });
        day.setCurrentItem(days.indexOf(strings[2]));
        mStringDay = days.get(days.indexOf(strings[2]));
        day.setCyclic(false);
        day.setAdapter(new WheelAdapter() {
            @Override
            public int getItemsCount() {
                return days.size();
            }

            @Override
            public Object getItem(int index) {
                return days.get(index);
            }

            @Override
            public int indexOf(Object o) {
                return days.indexOf(o);
            }
        });
        day.setOnItemSelectedListener(new OnItemSelectedListener() {
            @Override
            public void onItemSelected(int index) {
                mStringDay = days.get(day.getCurrentItem());
            }
        });
    }

    /**
     * 初始化數(shù)據(jù)
     */
    private void initDatas() {
        years = new ArrayList<>();
        months = new ArrayList<>();
        days = new ArrayList<>();
        for (int i = 2014; i < 2021; i++) {
            years.add(i + "");
        }
        for (int i = 1; i <= 12; i++) {
            if (i < 10) {
                months.add("0" + i);
            } else {
                months.add(i + "");
            }

        }
        for (int i = 1; i <= 31; i++) {
            if (i < 10) {
                days.add("0" + i);
            } else {
                days.add(i + "");
            }
        }
    }

    /**
     * 初始化控件
     */
    private void initViews() {
        confrim = (TextView) getView().findViewById(R.id.confrim);
        cancel = (TextView) getView().findViewById(R.id.cancel);
        year = (WheelView) getView().findViewById(R.id.year);
        month = (WheelView) getView().findViewById(R.id.month);
        day = (WheelView) getView().findViewById(R.id.day);
    }
}

此時,這個庫的坑給出來了.

        year.setCurrentItem(years.indexOf(strings[0]));
        Log.e(TAG, "initWheelViews: setCurrentItem(3)--->"+3 );
        mStringYear = years.get(years.indexOf(strings[0]));
        Log.e(TAG, "initWheelViews: getCurrentItem()--->"+year.getCurrentItem() );
        year.setCyclic(false);

year是顯示年的WheelView,setCurrentItem我起初設(shè)置為3,然后打印一下,因為set和get的設(shè)置值和返回值不一致.所以打印主要是想和getCurrentItem做一個對比. 結(jié)果真的對比做出來了..

1212.png

log打印如下, 我當時就起了怪了,set和get得到的值為啥不一樣啊,,.這讓我怎么搞呢, 出去抽根煙吧.

調(diào)整一下情緒和思路,開始看源碼.看作者到底是咋搞的..于是,打開了源碼

這是源碼中的get方法:

public final int getCurrentItem() {
        return selectedItem;
    }

返回了一個selectedItem這個字段.那搜一下這個字段吧.共兩處,第一處是聲明,哈哈

private int selectedItem;

第二處是:

if (translateY >= firstLineY && maxTextHeight + translateY <= secondLineY) {
                    // 中間條目
                    canvas.clipRect(0, 0, measuredWidth, (int) (itemHeight));
                    canvas.drawText(contentText, drawCenterContentStart, maxTextHeight - CENTERCONTENTOFFSET, paintCenterText);
                    int preSelectedItem = adapter.indexOf(visibles[counter]);
                    if(preSelectedItem != -1){
                        selectedItem = preSelectedItem;
                    }

那再看看preSelectedItem字段唄

int preSelectedItem = adapter.indexOf(visibles[counter]);

那就在看看這個visibles[counter],數(shù)組就不看了,看一下索引值吧

counter = 0;
        while (counter < itemsVisible) {
            canvas.save();
            // L(弧長)=α(弧度)* r(半徑) (弧度制)
            // 求弧度--> (L * π ) / (π * r)   (弧長X派/半圓周長)
            float itemHeight = maxTextHeight * lineSpacingMultiplier;
            double radian = ((itemHeight * counter - itemHeightOffset) * Math.PI) / halfCircumference;
            // 弧度轉(zhuǎn)換成角度(把半圓以Y軸為軸心向右轉(zhuǎn)90度拘领,使其處于第一象限及第四象限
            float angle = (float) (90D - (radian / Math.PI) * 180D);
            // 九十度以上的不繪制
            if (angle >= 90F || angle <= -90F) {
                canvas.restore();
            } else {
                String contentText = getContentText(visibles[counter]);

                //計算開始繪制的位置
                measuredCenterContentStart(contentText);
                measuredOutContentStart(contentText);
                float translateY = (float) (radius - Math.cos(radian) * radius - (Math.sin(radian) * maxTextHeight) / 2D);
                //根據(jù)Math.sin(radian)來更改canvas坐標系原點挽霉,然后縮放畫布,使得文字高度進行縮放判没,形成弧形3d視覺差
                canvas.translate(0.0F, translateY);
                canvas.scale(1.0F, (float) Math.sin(radian));
                if (translateY <= firstLineY && maxTextHeight + translateY >= firstLineY) {
                    // 條目經(jīng)過第一條線
                    canvas.save();
                    canvas.clipRect(0, 0, measuredWidth, firstLineY - translateY);
                    canvas.scale(1.0F, (float) Math.sin(radian) * SCALECONTENT);
                    canvas.drawText(contentText, drawOutContentStart, maxTextHeight, paintOuterText);
                    canvas.restore();
                    canvas.save();
                    canvas.clipRect(0, firstLineY - translateY, measuredWidth, (int) (itemHeight));
                    canvas.scale(1.0F, (float) Math.sin(radian) * 1F);
                    canvas.drawText(contentText, drawCenterContentStart, maxTextHeight - CENTERCONTENTOFFSET, paintCenterText);
                    canvas.restore();
                } else if (translateY <= secondLineY && maxTextHeight + translateY >= secondLineY) {
                    // 條目經(jīng)過第二條線
                    canvas.save();
                    canvas.clipRect(0, 0, measuredWidth, secondLineY - translateY);
                    canvas.scale(1.0F, (float) Math.sin(radian) * 1.0F);
                    canvas.drawText(contentText, drawCenterContentStart, maxTextHeight - CENTERCONTENTOFFSET, paintCenterText);
                    canvas.restore();
                    canvas.save();
                    canvas.clipRect(0, secondLineY - translateY, measuredWidth, (int) (itemHeight));
                    canvas.scale(1.0F, (float) Math.sin(radian) * SCALECONTENT);
                    canvas.drawText(contentText, drawOutContentStart, maxTextHeight, paintOuterText);
                    canvas.restore();
                } else if (translateY >= firstLineY && maxTextHeight + translateY <= secondLineY) {
                    // 中間條目
                    canvas.clipRect(0, 0, measuredWidth, (int) (itemHeight));
                    canvas.drawText(contentText, drawCenterContentStart, maxTextHeight - CENTERCONTENTOFFSET, paintCenterText);
                    int preSelectedItem = adapter.indexOf(visibles[counter]);
                    if(preSelectedItem != -1){
                        selectedItem = preSelectedItem;
                    }
                } else {
                    // 其他條目
                    canvas.save();
                    canvas.clipRect(0, 0, measuredWidth, (int) (itemHeight));
                    canvas.scale(1.0F, (float) Math.sin(radian) * SCALECONTENT);
                    canvas.drawText(contentText, drawOutContentStart, maxTextHeight, paintOuterText);
                    canvas.restore();
                }
                canvas.restore();
            }
            counter++;
        }

初始化為0,然后在監(jiān)聽中有++
所以日歷控件顯示的初始值為今天的話,點擊之后獲得的值是為0的item下標.哎,不說了,自己去改吧..改完后的代碼就是SccDateDialog的代碼,其實就是把初始化的值自己獲取一下..

其實原作者很厲害,只是差那么一點點就完美了..

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末仪吧,一起剝皮案震驚了整個濱河市庄新,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖择诈,帶你破解...
    沈念sama閱讀 216,692評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件械蹋,死亡現(xiàn)場離奇詭異,居然都是意外死亡羞芍,警方通過查閱死者的電腦和手機哗戈,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,482評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來荷科,“玉大人唯咬,你說我怎么就攤上這事∥方” “怎么了胆胰?”我有些...
    開封第一講書人閱讀 162,995評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長刻获。 經(jīng)常有香客問我蜀涨,道長,這世上最難降的妖魔是什么蝎毡? 我笑而不...
    開封第一講書人閱讀 58,223評論 1 292
  • 正文 為了忘掉前任厚柳,我火速辦了婚禮,結(jié)果婚禮上顶掉,老公的妹妹穿的比我還像新娘草娜。我一直安慰自己,他們只是感情好痒筒,可當我...
    茶點故事閱讀 67,245評論 6 388
  • 文/花漫 我一把揭開白布宰闰。 她就那樣靜靜地躺著,像睡著了一般簿透。 火紅的嫁衣襯著肌膚如雪移袍。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,208評論 1 299
  • 那天老充,我揣著相機與錄音葡盗,去河邊找鬼。 笑死啡浊,一個胖子當著我的面吹牛觅够,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播巷嚣,決...
    沈念sama閱讀 40,091評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼喘先,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了廷粒?” 一聲冷哼從身側(cè)響起窘拯,我...
    開封第一講書人閱讀 38,929評論 0 274
  • 序言:老撾萬榮一對情侶失蹤红且,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后涤姊,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體暇番,經(jīng)...
    沈念sama閱讀 45,346評論 1 311
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,570評論 2 333
  • 正文 我和宋清朗相戀三年思喊,在試婚紗的時候發(fā)現(xiàn)自己被綠了蛾狗。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片航邢。...
    茶點故事閱讀 39,739評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出领猾,到底是詐尸還是另有隱情应又,我是刑警寧澤鸯乃,帶...
    沈念sama閱讀 35,437評論 5 344
  • 正文 年R本政府宣布剖张,位于F島的核電站,受9級特大地震影響诬留,放射性物質(zhì)發(fā)生泄漏斜纪。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,037評論 3 326
  • 文/蒙蒙 一文兑、第九天 我趴在偏房一處隱蔽的房頂上張望盒刚。 院中可真熱鬧,春花似錦绿贞、人聲如沸因块。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,677評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽涡上。三九已至,卻和暖如春拒名,著一層夾襖步出監(jiān)牢的瞬間吩愧,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,833評論 1 269
  • 我被黑心中介騙來泰國打工增显, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留雁佳,地道東北人。 一個月前我還...
    沈念sama閱讀 47,760評論 2 369
  • 正文 我出身青樓同云,卻偏偏與公主長得像糖权,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子炸站,可洞房花燭夜當晚...
    茶點故事閱讀 44,647評論 2 354

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,082評論 25 707
  • 前言:最近一直做項目温兼,遇到許多問題,其中令我很不爽的是:項目需求要求的日歷控件一定要仿蘋果N淦酢!!安卓本身提供的Ti...
    鄭鴻翊閱讀 2,768評論 5 32
  • MaterialCalendarView項目開源地址:https://github.com/prolificint...
    徐敏敏閱讀 9,178評論 21 20
  • 1,本人應(yīng)產(chǎn)品需求, 要一個可上下滑動的一個日歷, 仿Clue實現(xiàn), 本人基于Square寫的一個Calendar...
    我有一口小白牙閱讀 1,248評論 0 1
  • 據(jù)第39次《中國互聯(lián)網(wǎng)絡(luò)發(fā)展狀況統(tǒng)計報告》顯示咒唆,截至2016年12月届垫,中國網(wǎng)民規(guī)模達7.31億,相當于歐洲人口總量...
    9c42a2489e1d閱讀 454評論 0 8