Jetpack ViewModel組件其他使用方式-之四

前言

前面介紹ViewModel時,說到ViewModel常與livedata聯(lián)合使用迫摔;但是ViewModel不單單是為了livedata而創(chuàng)建的,它可以幫助我們保存UI相關(guān)的數(shù)據(jù)泥从,維護(hù)與UI一直的生命周期并排除config change造成的影響句占;這里再介紹一種ViewModel的使用場景;如sdk中要實(shí)現(xiàn)無UI接口躯嫉,你不知道具體的接入方的邏輯是怎么樣的纱烘,比如要通過一個flag來控制不同的邏輯,但是因?yàn)榻尤敕降慕缑娌皇悄銓?shí)現(xiàn)的祈餐,你怎么添加這個flag呢擂啥; 就是通過ViewModel,實(shí)際上將你的flag標(biāo)志和接入方的實(shí)例對象綁定起來帆阳;同時前面說過HolderFragment的思路是值得借鑒的哺壶,我們在本文中也會看看如何使用

ViewModel使用

創(chuàng)建ViewModel

public class MyViewModel extends ViewModel {
    private boolean mflag;

    public boolean isflag() {
        return mflag;
    }

    public void setFlag(boolean mFlag) {
        this.mflag = mFlag;
    }
} 

將ViewModel實(shí)例與接入方界面綁定

MyViewModel myViewModel = ViewModelProviders.of(activity).get(MyViewModel.class);
MyViewModel myViewModel = ViewModelProviders.of(fragment).get(MyViewModel.class);

這樣就可以通過創(chuàng)建viewmodel示例對象來操作與activity/fragment相關(guān)的flag;

局限性

但是這種方法是有局限性的;我們看下ViewModelProviders山宾, ViewModelStores的接口

/**
     * Creates a {@link ViewModelProvider}, which retains ViewModels while a scope of given Activity
     * is alive. More detailed explanation is in {@link ViewModel}.
     * <p>
     * It uses {@link ViewModelProvider.AndroidViewModelFactory} to instantiate new ViewModels.
     *
     * @param activity an activity, in whose scope ViewModels should be retained
     * @return a ViewModelProvider instance
     */
    @NonNull
    @MainThread
    public static ViewModelProvider of(@NonNull FragmentActivity activity) {
        return of(activity, null);
    }

    /**
     * Creates a {@link ViewModelProvider}, which retains ViewModels while a scope of given
     * {@code fragment} is alive. More detailed explanation is in {@link ViewModel}.
     * <p>
     * It uses the given {@link Factory} to instantiate new ViewModels.
     *
     * @param fragment a fragment, in whose scope ViewModels should be retained
     * @param factory  a {@code Factory} to instantiate new ViewModels
     * @return a ViewModelProvider instance
     */
    @NonNull
    @MainThread
    public static ViewModelProvider of(@NonNull Fragment fragment, @Nullable Factory factory) {
        Application application = checkApplication(checkActivity(fragment));
        if (factory == null) {
            factory = ViewModelProvider.AndroidViewModelFactory.getInstance(application);
        }
        return new ViewModelProvider(ViewModelStores.of(fragment), factory);
    }

    /**
     * Creates a {@link ViewModelProvider}, which retains ViewModels while a scope of given Activity
     * is alive. More detailed explanation is in {@link ViewModel}.
     * <p>
     * It uses the given {@link Factory} to instantiate new ViewModels.
     *
     * @param activity an activity, in whose scope ViewModels should be retained
     * @param factory  a {@code Factory} to instantiate new ViewModels
     * @return a ViewModelProvider instance
     */
    @NonNull
    @MainThread
    public static ViewModelProvider of(@NonNull FragmentActivity activity,
            @Nullable Factory factory) {
        Application application = checkApplication(activity);
        if (factory == null) {
            factory = ViewModelProvider.AndroidViewModelFactory.getInstance(application);
        }
        return new ViewModelProvider(ViewModelStores.of(activity), factory);
    }
    /**
     * Returns the {@link ViewModelStore} of the given activity.
     *
     * @param activity an activity whose {@code ViewModelStore} is requested
     * @return a {@code ViewModelStore}
     */
    @NonNull
    @MainThread
    public static ViewModelStore of(@NonNull FragmentActivity activity) {
        if (activity instanceof ViewModelStoreOwner) {
            return ((ViewModelStoreOwner) activity).getViewModelStore();
        }
        return holderFragmentFor(activity).getViewModelStore();
    }

    /**
     * Returns the {@link ViewModelStore} of the given fragment.
     *
     * @param fragment a fragment whose {@code ViewModelStore} is requested
     * @return a {@code ViewModelStore}
     */
    @NonNull
    @MainThread
    public static ViewModelStore of(@NonNull Fragment fragment) {
        if (fragment instanceof ViewModelStoreOwner) {
            return ((ViewModelStoreOwner) fragment).getViewModelStore();
        }
        return holderFragmentFor(fragment).getViewModelStore();
    }

我們看到傳入的參數(shù)只支持support的activity/fragment至扰; 前面我們也分析過HolderFragment,說明了其是為了向前適配的塌碌, 我們看下能否使用HolderFragment呢渊胸?

@RestrictTo(RestrictTo.Scope.LIBRARY_GROUP)
public class HolderFragment extends Fragment implements ViewModelStoreOwner

很可惜是不行的旬盯,這個類里面的接口都是隱藏的台妆;

這個就比較奇怪了,明明底層已經(jīng)做了適配胖翰,但不把上層接口暴露出來接剩;這個底層有什么用呢?

原因: 適配是適配了androidx的support activity/fragment萨咳;而不是normal Activity/Fragment(android.app.activity)

參考:
Before androidx FragmentActivity.java
androidx FragmentActivity.java

借鑒HolderFragment

雖然不明白google為什么沒有為normal activity提供接口懊缺;但是既然沒有開放接口,我們也只能另想辦法了培他; 我們可以寫一個類HolderFragment鹃两,然后將其依賴到接入方的normal activity/fragment;然后再通過其得到對應(yīng)的ViewModel中保存的flag; 如下:

public class HolderFragmentForNormalUI extends Fragment implements ViewModelStoreOwner {
    private static final String LOG_TAG = "ViewModelStores";

    private static final HolderFragmentForNormalUI.HolderFragmentManager sHolderFragmentManager = new HolderFragmentForNormalUI.HolderFragmentManager();

    /**
     * @hide
     */
    @RestrictTo(RestrictTo.Scope.LIBRARY_GROUP)
    public static final String HOLDER_TAG =
            "com.bytedance.ttgame.sdk.module.account.utils.HolderFragmentForNormalUI";

    private ViewModelStore mViewModelStore = new ViewModelStore();

    public HolderFragmentForNormalUI() {
        setRetainInstance(true);
    }

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        sHolderFragmentManager.holderFragmentCreated(this);
    }

    @Override
    public void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        mViewModelStore.clear();
    }

    @NonNull
    @Override
    public ViewModelStore getViewModelStore() {
        return mViewModelStore;
    }

    public static HolderFragmentForNormalUI holderFragmentFor(Activity activity) {
        return sHolderFragmentManager.holderFragmentFor(activity);
    }

    public static HolderFragmentForNormalUI holderFragmentFor(Fragment fragment) {
        return sHolderFragmentManager.holderFragmentFor(fragment);
    }

    @SuppressWarnings("WeakerAccess")
    static class HolderFragmentManager {
        private Map<Activity, HolderFragmentForNormalUI> mNotCommittedActivityHolders = new HashMap<>();
        private Map<Fragment, HolderFragmentForNormalUI> mNotCommittedFragmentHolders = new HashMap<>();

        private Application.ActivityLifecycleCallbacks mActivityCallbacks =
                new EmptyActivityLifecycleCallbacksForNormalUI() {
                    @Override
                    public void onActivityDestroyed(Activity activity) {
                        HolderFragmentForNormalUI fragment = mNotCommittedActivityHolders.remove(activity);
                        if (fragment != null) {
                            Log.e(LOG_TAG, "Failed to save a ViewModel for " + activity);
                        }
                    }
                };

        private boolean mActivityCallbacksIsAdded = false;

        void holderFragmentCreated(Fragment holderFragment) {
            mNotCommittedActivityHolders.remove(holderFragment.getActivity());
        }

        private static HolderFragmentForNormalUI findHolderFragment(FragmentManager manager) {
            if (manager.isDestroyed()) {
                throw new IllegalStateException("Can't access ViewModels from onDestroy");
            }

            Fragment fragmentByTag = manager.findFragmentByTag(HOLDER_TAG);
            if (fragmentByTag != null && !(fragmentByTag instanceof HolderFragmentForNormalUI)) {
                throw new IllegalStateException("Unexpected "
                        + "fragment instance was returned by HOLDER_TAG");
            }
            return (HolderFragmentForNormalUI) fragmentByTag;
        }

        private static HolderFragmentForNormalUI createHolderFragment(FragmentManager fragmentManager) {
            HolderFragmentForNormalUI holder = new HolderFragmentForNormalUI();
            fragmentManager.beginTransaction().add(holder, HOLDER_TAG).commitAllowingStateLoss();
            return holder;
        }

        HolderFragmentForNormalUI holderFragmentFor(Activity activity) {
            FragmentManager fm = activity.getFragmentManager();
            HolderFragmentForNormalUI holder = findHolderFragment(fm);
            if (holder != null) {
                return holder;
            }
            holder = mNotCommittedActivityHolders.get(activity);
            if (holder != null) {
                return holder;
            }

            if (!mActivityCallbacksIsAdded) {
                mActivityCallbacksIsAdded = true;
                activity.getApplication().registerActivityLifecycleCallbacks(mActivityCallbacks);
            }
            holder = createHolderFragment(fm);
            mNotCommittedActivityHolders.put(activity, holder);
            return holder;
        }

        HolderFragmentForNormalUI holderFragmentFor(Fragment parentFragment) {
            FragmentManager fm = parentFragment.getChildFragmentManager();
            HolderFragmentForNormalUI holder = findHolderFragment(fm);
            if (holder != null) {
                return holder;
            }
            holder = createHolderFragment(fm);
            return holder;
        }
    }

使用方式(activity為android.app.Activity)

MyViewModel myViewModel = new ViewModelProvider(HolderFragmentForNormalUI.holderFragmentFor(activity).getViewModelStore(),
                    ViewModelProvider.AndroidViewModelFactory.getInstance(activity.getApplication())).get(SendCodeViewModel.class);

總結(jié)

ViewModel與activity/fragment依賴的關(guān)系可以用來給無法操作的界面添加相應(yīng)屬性舀凛,標(biāo)志位等俊扳;適用于我們抽取無UI接口時,有些地方需要與界面有耦合關(guān)系的情況(當(dāng)然理想情況下這是不合理的猛遍,可惜現(xiàn)實(shí)往往并不總能盡如人意)馋记; jectpack組件的使用方式還有很多,這都有待于我們探索懊烤;

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末梯醒,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子腌紧,更是在濱河造成了極大的恐慌茸习,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,858評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件壁肋,死亡現(xiàn)場離奇詭異号胚,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)墩划,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評論 3 395
  • 文/潘曉璐 我一進(jìn)店門涕刚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人乙帮,你說我怎么就攤上這事杜漠。” “怎么了?”我有些...
    開封第一講書人閱讀 165,282評論 0 356
  • 文/不壞的土叔 我叫張陵驾茴,是天一觀的道長盼樟。 經(jīng)常有香客問我,道長锈至,這世上最難降的妖魔是什么晨缴? 我笑而不...
    開封第一講書人閱讀 58,842評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮峡捡,結(jié)果婚禮上击碗,老公的妹妹穿的比我還像新娘。我一直安慰自己们拙,他們只是感情好稍途,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,857評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著砚婆,像睡著了一般械拍。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上装盯,一...
    開封第一講書人閱讀 51,679評論 1 305
  • 那天坷虑,我揣著相機(jī)與錄音,去河邊找鬼埂奈。 笑死迄损,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的挥转。 我是一名探鬼主播海蔽,決...
    沈念sama閱讀 40,406評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼绑谣!你這毒婦竟也來了党窜?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,311評論 0 276
  • 序言:老撾萬榮一對情侶失蹤借宵,失蹤者是張志新(化名)和其女友劉穎幌衣,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體壤玫,經(jīng)...
    沈念sama閱讀 45,767評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡豁护,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了欲间。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片楚里。...
    茶點(diǎn)故事閱讀 40,090評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖猎贴,靈堂內(nèi)的尸體忽然破棺而出班缎,到底是詐尸還是另有隱情蝴光,我是刑警寧澤,帶...
    沈念sama閱讀 35,785評論 5 346
  • 正文 年R本政府宣布达址,位于F島的核電站蔑祟,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏沉唠。R本人自食惡果不足惜疆虚,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,420評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望满葛。 院中可真熱鬧径簿,春花似錦、人聲如沸纱扭。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽乳蛾。三九已至,卻和暖如春鄙币,著一層夾襖步出監(jiān)牢的瞬間肃叶,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評論 1 271
  • 我被黑心中介騙來泰國打工十嘿, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留因惭,地道東北人。 一個月前我還...
    沈念sama閱讀 48,298評論 3 372
  • 正文 我出身青樓绩衷,卻偏偏與公主長得像蹦魔,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子咳燕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,033評論 2 355

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