兩種常用的帶有底部導航的AndroidUI實現(xiàn)

最近決定聪舒,從頭整理Android基礎知識,和常用基本架構 和框架 虐急,方便以后使用箱残,更主要是知道自己到底會些什么,突破技術瓶頸V褂酢1患!

這一篇先從基本上每個應用都會搭建的軟件UI 總結做起敬惦,這里的軟件架構是指對現(xiàn)在流行軟件中都會搭建的UI界面底部導航盼理,并切換不同的片段或者跳轉頁面,如下圖:

底部均帶有導航欄

主要實現(xiàn)一共是以下兩種選擇:

1俄删、RadioButton 加幀布局并替換幀布局中的片段 實現(xiàn)點擊按鈕可切換片段的效果(這里面采用尚硅谷楊老師的封裝方法)

2宏怔、結合RadioButton 加ViewPager 填充片段實現(xiàn)可點擊并支持滑動切換片段的底部導航

一、RadioButton 加幀布局并替換幀布局中的片段

相信這種實現(xiàn)方式畴椰,很多人都會臊诊,這里面涉及到的主要是片段的生命周期,片段事務的使用迅矛,替換,添加潜叛,覆蓋秽褒,還有就是Radiobutton的使用已經很普遍了壶硅,寫在這里就算是總結,之后如果需要直接過來粘貼了销斟。
效果圖如下庐椒,很簡單的效果,但是這里面加入了一下代碼結構設計的思想蚂踊,方便提升约谈,還有就是防止片段重復,反復的初始化:


片段效果圖

1犁钟、首先我們需要一個主Activity棱诱,然后就是四個片段

片段的生命周期這里就不做贅述了,直接來看這幾個片段的創(chuàng)建過程涝动,我是很喜歡這種方式迈勋,也為之后寫代碼提供了一種思路:

首先一個BaseFragment作為基類:這樣就定義了一個規(guī)范,有時感覺定義接口并沒有這樣優(yōu)雅醋粟。

package com.wgd.fragment;

import android.content.Context;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

/**
 * Created by wangg on 2018/7/26.
 * * 作用:基類靡菇,公共類
 * CommonFragment,ThirdPartyFragment,CustomFragment,OtherFragment等都要繼承該類
 */
public abstract class BaseFragment extends Fragment {
    //上下文
    protected Context context;

    protected String TAG="FragmentTAG";
    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        context=getActivity();
    }

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        return initView();
    }
    @Override
    public void onActivityCreated(@Nullable Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        initData();
    }
    protected abstract View initView();
    /**
     * 當孩子需要初始化數(shù)據(jù)米愿,或者聯(lián)網請求綁定數(shù)據(jù)厦凤,展示數(shù)據(jù)的 等等可以重寫該方法
     */
    protected void initData(){}
}

然后四個片段主要是依照這個抽象類的標準去完成:
舉例:

package com.wgd.fragment;

import android.graphics.Color;
import android.util.Log;
import android.view.Gravity;
import android.view.View;
import android.widget.TextView;

/**
 * Created by 王國棟 on 2018/7/26.
 * qq 1350802989
 * 微信 WGDLOVELC
 * 你的煎熬,都是因為你的上進心和你的能力不成正比
 */
public class CommonFragment extends BaseFragment{
    TextView textView;
    @Override
    protected View initView() {
        Log.i(TAG, "initView: CommonFragment初始化了");
        textView=new TextView(context);
        textView.setTextSize(20);
        textView.setTextColor(Color.RED);
        textView.setGravity(Gravity.CENTER);
        return textView;
    }
    @Override
    protected void initData() {
        super.initData();
        textView.setText("common");
    }
}

以此類推創(chuàng)建另外三個育苟,最后添加到主Activity中较鼓,在Activity中完成片段的添加替換和覆蓋操作,什么是覆蓋宙搬?就是為了防止片段反復初始化笨腥!
主Activity代碼:

package com.wgd.example1;

import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentTransaction;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.Window;
import android.view.WindowManager;
import android.widget.RadioGroup;

import com.wgd.fragment.BaseFragment;
import com.wgd.fragment.CommonFragment;
import com.wgd.fragment.CustomFragment;
import com.wgd.fragment.OtherFragment;
import com.wgd.fragment.ThirdPartyFragment;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity {
    private RadioGroup mRg_main;
    private List<BaseFragment> mBaseFragment;
    /**
     * 選中的Fragment的對應的位置
     */
    private int position;
    /**
     * 上次切換的Fragment
     */
    private Fragment mContent;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //初始化View
        initView();
        //初始化Fragment
        initFragment();
        //設置RadioGroup的監(jiān)聽
        setListener();
    }
    private void setListener() {
        mRg_main.setOnCheckedChangeListener(new MyOnCheckedChangeListener());
        //設置默認選中常用框架
        mRg_main.check(R.id.rb_common_frame);
    }

    class MyOnCheckedChangeListener implements RadioGroup.OnCheckedChangeListener {
        @Override
        public void onCheckedChanged(RadioGroup group, int checkedId) {
            switch (checkedId){
                case R.id.rb_common_frame://常用框架
                    position = 0;
                    break;
                case R.id.rb_thirdparty://第三方
                    position = 1;
                    break;
                case R.id.rb_custom://自定義
                    position = 2;
                    break;
                case R.id.rb_other://其他
                    position = 3;
                    break;
                default:
                    position = 0;
                    break;
            }
            //根據(jù)位置得到對應的Fragment
            BaseFragment to = getFragment();
            //替換
            switchFrament(mContent,to);
        }
    }
    /**
     *
     * @param from 剛顯示的Fragment,馬上就要被隱藏了
     * @param to 馬上要切換到的Fragment,一會要顯示
     */
    private void switchFrament(Fragment from,Fragment to) {
        if(from != to){
            mContent = to;
            FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
            //才切換
            //判斷有沒有被添加
            if(!to.isAdded()){
                //to沒有被添加
                //from隱藏
                if(from != null){
                    ft.hide(from);
                }
                //添加to
                if(to != null){
                    ft.add(R.id.fl_content,to).commit();
                }
            }else{
                //to已經被添加
                // from隱藏
                if(from != null){
                    ft.hide(from);
                }
                //顯示to
                if(to != null){
                    ft.show(to).commit();
                }
            }
        }

    }
//    private void switchFrament(BaseFragment fragment) {
//        //1.得到FragmentManger
//        FragmentManager fm = getSupportFragmentManager();
//        //2.開啟事務
//        FragmentTransaction transaction = fm.beginTransaction();
//        //3.替換
//        transaction.replace(R.id.fl_content, fragment);
//        //4.提交事務
//        transaction.commit();
//    }

    /**
     * 根據(jù)位置得到對應的Fragment
     * @return
     */
    private BaseFragment getFragment() {
        BaseFragment fragment = mBaseFragment.get(position);
        return fragment;
    }

    private void initFragment() {
        mBaseFragment = new ArrayList<>();
        mBaseFragment.add(new CommonFragment());//常用框架Fragment
        mBaseFragment.add(new ThirdPartyFragment());//第三方Fragment
        mBaseFragment.add(new CustomFragment());//自定義控件Fragment
        mBaseFragment.add(new OtherFragment());//其他Fragment
    }

    private void initView() {
        setContentView(R.layout.activity_main);
        mRg_main = (RadioGroup) findViewById(R.id.rg_main);
    }
}

activty_main.xml文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context="com.wgd.example1.MainActivity">
    <!--標題欄-->
    <include layout="@layout/layout_title" />
    <!--FrameLayout-->
    <FrameLayout
        android:id="@+id/fl_content"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1" />

    <!--底部的RadioGroup-->
    <RadioGroup
        android:id="@+id/rg_main"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="#11000000"
        android:gravity="center_vertical"
        android:orientation="horizontal"
        android:padding="5dp">


        <RadioButton
            android:id="@+id/rb_common_frame"
            style="@style/bottom_tag_style"
            android:drawableTop="@drawable/rb_common_frame_drawable_selector"
            android:text="常用框架"
            />

        <RadioButton
            android:id="@+id/rb_thirdparty"
            style="@style/bottom_tag_style"
            android:drawableTop="@drawable/rb_thirdparty_drawable_selector"
            android:text="第三方"

            />
        <RadioButton
            android:id="@+id/rb_custom"
            style="@style/bottom_tag_style"
            android:drawableTop="@drawable/rb_custom_drawable_selector"
            android:text="自定義控件" />
        <RadioButton
            android:id="@+id/rb_other"
            style="@style/bottom_tag_style"
            android:drawableTop="@drawable/rb_other_drawable_selector"
            android:text="其他" />

    </RadioGroup>
</LinearLayout>

bottom_tag_style文件

   <style name="bottom_tag_style" >
        <!-- Customize your theme here. -->
        <item name="android:layout_width">wrap_content</item>
        <item name="android:layout_height">wrap_content</item>
        <item name="android:layout_weight">1</item>
        <item name="android:button">@android:color/transparent</item>
        <item name="android:drawablePadding">3dp</item>
        <item name="android:gravity">center</item>
        <item name="android:textColor">@drawable/bottom_textcolor_drawable_selector</item>
        <item name="android:textSize">10sp</item>
    </style>

bottom_textcolor_drawable_selector

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_checked="false" android:color="#363636"/>
    <item android:state_checked="true" android:color="#3097FD"/>
</selector>

rb_common_frame_drawable_selector文件勇垛,另外三個片段一樣

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:state_checked="false" android:drawable="@drawable/ic_tab_video"/>
<item android:state_checked="true" android:drawable="@drawable/ic_tab_video_press"/>
</selector>

這個例子RadioButton的字體的顏色是通過Selector變化的脖母,也可以像下面一樣,第二種方式實現(xiàn)闲孤。

二谆级、RadioButton 加ViewPager 填充片段實現(xiàn)(可點擊可滑動)

1、首先同樣需要一個主Activity讼积,四個片段(一般應用底部都會分為四個部分肥照,可根據(jù)實際需求添加)

底部導航欄的圖片,每個主題對應深色和淺色兩張圖片勤众,分別用來給RedioButton填充背景舆绎,作為待選擇和被選擇的效果。

主Activity布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".activity.MainActivity">
    <!-- 這里可根據(jù)需求添加頂部title布局這個例子就省略了 -->

    <android.support.v4.view.ViewPager
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:id="@+id/viewpager"
        android:background="@color/background"
        >
    </android.support.v4.view.ViewPager>

    <RadioGroup
        android:layout_width="match_parent"
        android:layout_height="60dp"
        android:layout_weight="0"
        android:orientation="horizontal"
        android:gravity="center"
        android:background="#E3E2E1"
        android:id="@+id/radiogroup"
        >
        <RadioButton
            style="@style/MyRadioButtonStyle"
            android:id="@+id/r1"
            android:drawableTop="@drawable/radiobtn_selector1"
            android:text="@string/workbank"
            android:checked="true"
            android:textColor="@color/blue"
            />
        <RadioButton
            style="@style/MyRadioButtonStyle"
            android:id="@+id/r2"
            android:drawableTop="@drawable/radiobtn_selector2"
            android:text="@string/jobfair"
            android:textColor="@color/gray"
            />
        <RadioButton
            style="@style/MyRadioButtonStyle"
            android:id="@+id/r3"
            android:drawableTop="@drawable/radiobtn_selector3"
            android:text="@string/news"
            android:textColor="@color/gray"
            />
        <RadioButton
            style="@style/MyRadioButtonStyle"
            android:id="@+id/r4"
            android:drawableTop="@drawable/radiobtn_selector4"
            android:text="@string/mine"
            android:textColor="@color/gray"
            />
    </RadioGroup>
</LinearLayout>

style如下:

 <style name="MyRadioButtonStyle">
        <item name="android:button">@null</item>
        <item name="android:layout_width">wrap_content</item>
        <item name="android:layout_height">wrap_content</item>
        <item name="android:layout_weight">1</item>
        <item name="android:gravity">center</item>
    </style>

Selecter文件如下(需要幾個導航欄就創(chuàng)建幾個)

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_checked="true" android:drawable="@drawable/nav1_01"></item>
    <item android:state_checked="false" android:drawable="@drawable/nav1_02"></item>
</selector>

RedioButton 和 Selecter的用法應該不用多說了们颜,主要是再Style樣式里面的配置吕朵。
RedioButton 圖片下發(fā)的字體顏色還可以用這種方式實現(xiàn)猎醇,就是先再在xml中設置一個默認顏色,然后在Activty中根據(jù)選中狀態(tài)努溃,修改字體顏色硫嘶。
Selecter 給選中和未被選中時兩個不同顏色的圖片即可,注意是state_checked

主Activity中的代碼

/**
 * author wanggd
 *
 * 主入口Activity
 */
public class MainActivity extends FragmentActivity
        implements RadioGroup.OnCheckedChangeListener,ViewPager.OnPageChangeListener{
    private RadioButton r1, r2, r3, r4;
    private RadioGroup radioGroup;
    private ViewPager pager;
    private List<Fragment> fraglist;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initView();
    }
    private void initView() {
        fraglist = new ArrayList<>();
        fraglist.add(new WorkBankFragment());
        fraglist.add(new JobFairFragment());
        fraglist.add(new NewsFragment());
        fraglist.add(new MineFragment());
        r1 = (RadioButton) findViewById(R.id.r1);
        r2 = (RadioButton) findViewById(R.id.r2);
        r3 = (RadioButton) findViewById(R.id.r3);
        r4 = (RadioButton) findViewById(R.id.r4);
        radioGroup = (RadioGroup) findViewById(R.id.radiogroup);
        radioGroup.setOnCheckedChangeListener(this);
        pager = (ViewPager) findViewById(R.id.viewpager);
        pager.setPageTransformer(true, new DepthPageTransformer());
        pager.addOnPageChangeListener(this);
        pager.setAdapter(new MyAdapter(getSupportFragmentManager()));
    }
    @Override
    public void onCheckedChanged(RadioGroup group, int checkedId) {
        if (checkedId == R.id.r1) {
            checkLogin();
            r1.setTextColor(ContextCompat.getColor(this, R.color.blue));
            r2.setTextColor(ContextCompat.getColor(this, R.color.gray));
            r3.setTextColor(ContextCompat.getColor(this, R.color.gray));
            r4.setTextColor(ContextCompat.getColor(this, R.color.gray));
            pager.setCurrentItem(0);
        } else if (checkedId == R.id.r2) {
            checkLogin();
            r2.setTextColor(ContextCompat.getColor(this, R.color.blue));
            r1.setTextColor(ContextCompat.getColor(this, R.color.gray));
            r3.setTextColor(ContextCompat.getColor(this, R.color.gray));
            r4.setTextColor(ContextCompat.getColor(this, R.color.gray));
            pager.setCurrentItem(1);
        } else if (checkedId == R.id.r3) {
            checkLogin();
            r3.setTextColor(ContextCompat.getColor(this, R.color.blue));
            r1.setTextColor(ContextCompat.getColor(this, R.color.gray));
            r2.setTextColor(ContextCompat.getColor(this, R.color.gray));
            r4.setTextColor(ContextCompat.getColor(this, R.color.gray));
            pager.setCurrentItem(2);
        } else if (checkedId == R.id.r4) {
            checkLogin();
            r4.setTextColor(ContextCompat.getColor(this, R.color.blue));
            r1.setTextColor(ContextCompat.getColor(this, R.color.gray));
            r3.setTextColor(ContextCompat.getColor(this, R.color.gray));
            r2.setTextColor(ContextCompat.getColor(this, R.color.gray));
            pager.setCurrentItem(3);
        }
    }
    @Override
    public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
    }
    @Override
    public void onPageSelected(int position) {
        switch (position) {
            case 0:
                r1.setChecked(true);
                break;
            case 1:
                r2.setChecked(true);
                break;
            case 2:
                r3.setChecked(true);
                break;
            case 3:
                r4.setChecked(true);
                break;
            default:
                break;
        }
    }
    @Override
    public void onPageScrollStateChanged(int state) {
    }
    private class MyAdapter extends FragmentStatePagerAdapter {

        public MyAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int position) {
            return fraglist.get(position);
        }

        @Override
        public int getCount() {
            return fraglist.size();
        }

        @Override
        public void destroyItem(ViewGroup container, int position, Object object) {
            //這個地方注釋掉可以防止ViewPager加載卡頓的問題
            //super.destroyItem(container, position, object);
        }
    }
}

Fragment 的實現(xiàn) 可以結合第一種方式實現(xiàn)梧税÷偌玻或者如果java基礎薄弱根據(jù)實際需要直接創(chuàng)建不同的Fragment穴墅,然后每個Fragment一個布局文件也可以弊琴,只是會顯得雜亂而已
到此邑闲,就可以實現(xiàn)結合viewpager的滑動導航俊嗽,其實也可以結合上面的例子可柿,將代碼精簡很多雄嚣,這是很久之前寫的代碼了碎紊,作為例子放到這澎媒。

?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末晾蜘,一起剝皮案震驚了整個濱河市邻眷,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌剔交,老刑警劉巖肆饶,帶你破解...
    沈念sama閱讀 212,029評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異岖常,居然都是意外死亡驯镊,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,395評論 3 385
  • 文/潘曉璐 我一進店門竭鞍,熙熙樓的掌柜王于貴愁眉苦臉地迎上來板惑,“玉大人,你說我怎么就攤上這事偎快》氤耍” “怎么了?”我有些...
    開封第一講書人閱讀 157,570評論 0 348
  • 文/不壞的土叔 我叫張陵晒夹,是天一觀的道長裆馒。 經常有香客問我,道長丐怯,這世上最難降的妖魔是什么喷好? 我笑而不...
    開封第一講書人閱讀 56,535評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮读跷,結果婚禮上梗搅,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好无切,可當我...
    茶點故事閱讀 65,650評論 6 386
  • 文/花漫 我一把揭開白布蟀俊。 她就那樣靜靜地躺著,像睡著了一般订雾。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上矛洞,一...
    開封第一講書人閱讀 49,850評論 1 290
  • 那天洼哎,我揣著相機與錄音,去河邊找鬼沼本。 笑死噩峦,一個胖子當著我的面吹牛,可吹牛的內容都是我干的抽兆。 我是一名探鬼主播识补,決...
    沈念sama閱讀 39,006評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼辫红!你這毒婦竟也來了凭涂?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,747評論 0 268
  • 序言:老撾萬榮一對情侶失蹤贴妻,失蹤者是張志新(化名)和其女友劉穎切油,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體名惩,經...
    沈念sama閱讀 44,207評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡澎胡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,536評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了娩鹉。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片攻谁。...
    茶點故事閱讀 38,683評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖弯予,靈堂內的尸體忽然破棺而出戚宦,到底是詐尸還是另有隱情,我是刑警寧澤熙涤,帶...
    沈念sama閱讀 34,342評論 4 330
  • 正文 年R本政府宣布阁苞,位于F島的核電站,受9級特大地震影響祠挫,放射性物質發(fā)生泄漏那槽。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,964評論 3 315
  • 文/蒙蒙 一等舔、第九天 我趴在偏房一處隱蔽的房頂上張望骚灸。 院中可真熱鬧,春花似錦慌植、人聲如沸甚牲。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,772評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽丈钙。三九已至非驮,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間雏赦,已是汗流浹背劫笙。 一陣腳步聲響...
    開封第一講書人閱讀 32,004評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留星岗,地道東北人填大。 一個月前我還...
    沈念sama閱讀 46,401評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像俏橘,于是被迫代替她去往敵國和親允华。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,566評論 2 349

推薦閱讀更多精彩內容

  • 內容 抽屜菜單 ListView WebView SwitchButton 按鈕 點贊按鈕 進度條 TabLayo...
    小狼W閱讀 1,613評論 0 10
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,799評論 25 707
  • 內容抽屜菜單ListViewWebViewSwitchButton按鈕點贊按鈕進度條TabLayout圖標下拉刷新...
    皇小弟閱讀 46,729評論 22 665
  • 今夜寥掐,一定有什么是我錯過的 比如靴寂,窗外搖曳的樹影 又比如,窗臺吹亂的落花 那我未錯過的呢 幾頁書之外 必定是一扇打...
    東方云鶴閱讀 187評論 0 0
  • 姓名:陳權 公司:青檸養(yǎng)車 【知~學習】 《財富自由》音頻打卡第2天 《輕課口語》打卡第418天 【行~實踐】 一...
    水青檸閱讀 131評論 0 0