// Create the tabs that make up the user interface. mTabLayout = (TabLayout) findViewById(R.id.tabs); finalinttabCount= UiDataModel.getUiDataModel().getTabCount(); finalbooleanshowTabLabel= getResources().getBoolean(R.bool.showTabLabel); finalbooleanshowTabHorizontally= getResources().getBoolean(R.bool.showTabHorizontally); for (inti=0; i < tabCount; i++) { final UiDataModel.TabtabModel= UiDataModel.getUiDataModel().getTab(i); final@StringResintlabelResId= tabModel.getLabelResId();
final TabLayout.Tabtab= mTabLayout.newTab() .setTag(tabModel) .setIcon(tabModel.getIconResId()) .setContentDescription(labelResId);
if (showTabLabel) { tab.setText(labelResId); tab.setCustomView(R.layout.tab_item);
// Bind the icon to the TextView. finalDrawableicon= tab.getIcon(); if (showTabHorizontally) { // Remove the icon so it doesn't affect the minimum TabLayout height. tab.setIcon(null); text.setCompoundDrawablesRelativeWithIntrinsicBounds(icon, null, null, null); } else { text.setCompoundDrawablesRelativeWithIntrinsicBounds(null, icon, null, null); } }
/** Identifies each of the primary tabs within the application. */ publicenumTab { ALARMS(AlarmClockFragment.class, R.drawable.ic_tab_alarm, R.string.menu_alarm), CLOCKS(ClockFragment.class, R.drawable.ic_tab_clock, R.string.menu_clock), TIMERS(TimerFragment.class, R.drawable.ic_tab_timer, R.string.menu_timer), STOPWATCH(StopwatchFragment.class, R.drawable.ic_tab_stopwatch, R.string.menu_stopwatch);
// Customize the view pager. mFragmentTabPagerAdapter = newFragmentTabPagerAdapter(this); mFragmentTabPager = (ViewPager) findViewById(R.id.desk_clock_pager); // Keep all four tabs to minimize jank. mFragmentTabPager.setOffscreenPageLimit(3); // Set Accessibility Delegate to null so view pager doesn't intercept movements and // prevent the fab from being selected. mFragmentTabPager.setAccessibilityDelegate(null); // Mirror changes made to the selected page of the view pager into UiDataModel. mFragmentTabPager.addOnPageChangeListener(newPageChangeWatcher()); mFragmentTabPager.setAdapter(mFragmentTabPagerAdapter);
// Mirror changes made to the selected tab into UiDataModel. mTabLayout.addOnTabSelectedListener(newTabLayout.OnTabSelectedListener() { @Override publicvoidonTabSelected(TabLayout.Tab tab) { UiDataModel.getUiDataModel().setSelectedTab((UiDataModel.Tab) tab.getTag()); }
/** * This adapter produces the DeskClockFragments that are the content of the DeskClock tabs. The * adapter presents the tabs in LTR and RTL order depending on the text layout direction for the * current locale. To prevent issues when switching between LTR and RTL, fragments are registered * with the manager using position-independent tags, which is an important departure from * FragmentPagerAdapter. */ finalclassFragmentTabPagerAdapterextendsPagerAdapter {
privatefinal DeskClock mDeskClock;
/** The manager into which fragments are added. */ privatefinal FragmentManager mFragmentManager;
/** A fragment cache that can be accessed before {@link #instantiateItem} is called. */ privatefinal Map<UiDataModel.Tab, DeskClockFragment> mFragmentCache;
/** The active fragment transaction if one exists. */ private FragmentTransaction mCurrentTransaction;
/** The current fragment displayed to the user. */ private Fragment mCurrentPrimaryItem;
/** * @param position the left-to-right index of the fragment to be returned * @return the fragment displayed at the given {@code position} */ DeskClockFragment getDeskClockFragment(int position) { // Fetch the tab the UiDataModel reports for the position. final UiDataModel.Tabtab= UiDataModel.getUiDataModel().getTabAt(position);
// First check the local cache for the fragment. DeskClockFragmentfragment= mFragmentCache.get(tab); if (fragment != null) { return fragment; }
// Next check the fragment manager; relevant when app is rebuilt after locale changes // because this adapter will be new and mFragmentCache will be empty, but the fragment // manager will retain the Fragments built on original application launch. fragment = (DeskClockFragment) mFragmentManager.findFragmentByTag(tab.name()); if (fragment != null) { fragment.setFabContainer(mDeskClock); mFragmentCache.put(tab, fragment); return fragment; }
// Otherwise, build the fragment from scratch. finalStringfragmentClassName= tab.getFragmentClassName(); fragment = (DeskClockFragment) Fragment.instantiate(mDeskClock, fragmentClassName); fragment.setFabContainer(mDeskClock); mFragmentCache.put(tab, fragment); return fragment; }
@Override publicvoidstartUpdate(ViewGroup container) { if (container.getId() == View.NO_ID) { thrownewIllegalStateException("ViewPager with adapter " + this + " has no id"); } }
@Override public Object instantiateItem(ViewGroup container, int position) { if (mCurrentTransaction == null) { mCurrentTransaction = mFragmentManager.beginTransaction(); }
// Use the fragment located in the fragment manager if one exists. final UiDataModel.Tabtab= UiDataModel.getUiDataModel().getTabAt(position); Fragmentfragment= mFragmentManager.findFragmentByTag(tab.name()); if (fragment != null) { mCurrentTransaction.attach(fragment); } else { fragment = getDeskClockFragment(position); mCurrentTransaction.add(container.getId(), fragment, tab.name()); }
if (fragment != mCurrentPrimaryItem) { FragmentCompat.setMenuVisibility(fragment, false); FragmentCompat.setUserVisibleHint(fragment, false); }
/* Initialize the current thread as a looper. This gives you a chance to create handlers that then reference this looper, before actually starting the loop. Be sure to call {@link #loop()} after calling this method, and end it by calling{@link #quit()}. */ publicstaticvoidprepare() { prepare(true); }
privatestaticvoidprepare(boolean quitAllowed) { if (sThreadLocal.get() != null) { thrownewRuntimeException("Only one Looper may be created per thread"); } sThreadLocal.set(newLooper(quitAllowed)); } ... privateLooper(boolean quitAllowed) { mQueue = newMessageQueue(quitAllowed); mThread = Thread.currentThread(); }
/** Return the Looper object associated with the current thread. Returns null if the calling thread is not associated with a Looper.*/ publicstatic@Nullable Looper myLooper() { return sThreadLocal.get(); }
/** * Run the message queue in this thread. Be sure to call * {@link #quit()} to end the loop. */ publicstaticvoidloop() { finalLooperme= myLooper(); if (me == null) { thrownewRuntimeException("No Looper; Looper.prepare() wasn't called on this thread."); } finalMessageQueuequeue= me.mQueue; ... for (;;) { Messagemsg= queue.next(); // might block if (msg == null) { // No message indicates that the message queue is quitting. return; } ... try { msg.target.dispatchMessage(msg); end = (slowDispatchThresholdMs == 0) ? 0 : SystemClock.uptimeMillis(); } finally { if (traceTag != 0) { Trace.traceEnd(traceTag); } } ... msg.recycleUnchecked(); } }
/** * Field number for <code>get</code> and <code>set</code> indicating the * month. This is a calendar-specific value. The first month of * the year in the Gregorian and Julian calendars is * <code>JANUARY</code> which is 0; the last depends on the number * of months in a year. * * @see #JANUARY * @see #FEBRUARY * @see #MARCH * @see #APRIL * @see #MAY * @see #JUNE * @see #JULY * @see #AUGUST * @see #SEPTEMBER * @see #OCTOBER * @see #NOVEMBER * @see #DECEMBER * @see #UNDECIMBER */
SQLiteOpenHelper versions the database files. The version number is the int argument passed to the constructor. In the database file, the version number is stored in PRAGMA user_version.
onCreate() is only run when the database file did not exist and was just created. If onCreate()returns successfully (doesn’t throw an exception), the database is assumed to be created with the requested version number. As an implication, you should not catch SQLExceptions in onCreate()yourself.
onUpgrade() is only called when the database file exists but the stored version number is lower than requested in constructor. The onUpgrade() should update the table schema to the requested version.
When changing the table schema in code (onCreate()), you should make sure the database is updated. Two main approaches:
Delete the old database file so that onCreate() is run again. This is often preferred at development time where you have control over the installed versions and data loss is not an issue. Some ways to to delete the database file:
Uninstall the application. Use the application manager or adb uninstall your.package.name from shell.
Clear application data. Use the application manager.
Increment the database version so that onUpgrade() is invoked. This is slightly more complicated as more code is needed.
For development time schema upgrades where data loss is not an issue, you can just use execSQL("DROP TABLE IF EXISTS <tablename>") in to remove your existing tables and call onCreate() to recreate the database.
For released versions, you should implement data migration in onUpgrade() so your users don’t lose their data.
I had the same error after renaming/refactoring. What I did was add the applicationId property attribute to my build.gradle file, and set its value to the application package. Like this:
The Jack toolchain is now considered deprecated according to this post and work is being done to natively support Java 8 features as part of the Android build system in the coming weeks according to the post.
The post also mentions that there should be little to no work migrating from Jack to the new method in case you still wanted to try enabling Java 8 features with Jack.
UPDATE 2 Preview Built-in Support
You can now try out the new built-in support for Java 8 using the latest Android Studio preview 2.4 preview 6.
For more information on how to enable it or migrate from Jack or Retrolambda see the documentation.
Error:(28, 0) Could not find method implementation() for arguments [com.android.support:appcompat-v7:25.3.1] on object of type org.gradle.api.internal.artifacts.dsl.dependencies.DefaultDependencyHandler. Please install the Android Support Repository from the Android SDK Manager. <a href="openAndroidSdkManager">Open Android SDK Manager</a>