1 /* 2 * Copyright (C) 2015 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.internal.widget; 18 19 import android.database.DataSetObservable; 20 import android.database.DataSetObserver; 21 import android.os.Parcelable; 22 import android.view.View; 23 import android.view.ViewGroup; 24 25 /** 26 * Base class providing the adapter to populate pages inside of 27 * a {@link androidx.viewpager.view.ViewPager}. You will most likely want to use a more 28 * specific implementation of this, such as 29 * {@link androidx.fragment.app.FragmentPagerAdapter} or 30 * {@link androidx.fragment.app.FragmentStatePagerAdapter}. 31 * 32 * <p>When you implement a PagerAdapter, you must override the following methods 33 * at minimum:</p> 34 * <ul> 35 * <li>{@link #instantiateItem(android.view.ViewGroup, int)}</li> 36 * <li>{@link #destroyItem(android.view.ViewGroup, int, Object)}</li> 37 * <li>{@link #getCount()}</li> 38 * <li>{@link #isViewFromObject(android.view.View, Object)}</li> 39 * </ul> 40 * 41 * <p>PagerAdapter is more general than the adapters used for 42 * {@link android.widget.AdapterView AdapterViews}. Instead of providing a 43 * View recycling mechanism directly ViewPager uses callbacks to indicate the 44 * steps taken during an update. A PagerAdapter may implement a form of View 45 * recycling if desired or use a more sophisticated method of managing page 46 * Views such as Fragment transactions where each page is represented by its 47 * own Fragment.</p> 48 * 49 * <p>ViewPager associates each page with a key Object instead of working with 50 * Views directly. This key is used to track and uniquely identify a given page 51 * independent of its position in the adapter. A call to the PagerAdapter method 52 * {@link #startUpdate(android.view.ViewGroup)} indicates that the contents of the ViewPager 53 * are about to change. One or more calls to {@link #instantiateItem(android.view.ViewGroup, int)} 54 * and/or {@link #destroyItem(android.view.ViewGroup, int, Object)} will follow, and the end 55 * of an update will be signaled by a call to {@link #finishUpdate(android.view.ViewGroup)}. 56 * By the time {@link #finishUpdate(android.view.ViewGroup) finishUpdate} returns the views 57 * associated with the key objects returned by 58 * {@link #instantiateItem(android.view.ViewGroup, int) instantiateItem} should be added to 59 * the parent ViewGroup passed to these methods and the views associated with 60 * the keys passed to {@link #destroyItem(android.view.ViewGroup, int, Object) destroyItem} 61 * should be removed. The method {@link #isViewFromObject(android.view.View, Object)} identifies 62 * whether a page View is associated with a given key object.</p> 63 * 64 * <p>A very simple PagerAdapter may choose to use the page Views themselves 65 * as key objects, returning them from {@link #instantiateItem(android.view.ViewGroup, int)} 66 * after creation and adding them to the parent ViewGroup. A matching 67 * {@link #destroyItem(android.view.ViewGroup, int, Object)} implementation would remove the 68 * View from the parent ViewGroup and {@link #isViewFromObject(android.view.View, Object)} 69 * could be implemented as <code>return view == object;</code>.</p> 70 * 71 * <p>PagerAdapter supports data set changes. Data set changes must occur on the 72 * main thread and must end with a call to {@link #notifyDataSetChanged()} similar 73 * to AdapterView adapters derived from {@link android.widget.BaseAdapter}. A data 74 * set change may involve pages being added, removed, or changing position. The 75 * ViewPager will keep the current page active provided the adapter implements 76 * the method {@link #getItemPosition(Object)}.</p> 77 */ 78 public abstract class PagerAdapter { 79 private DataSetObservable mObservable = new DataSetObservable(); 80 81 public static final int POSITION_UNCHANGED = -1; 82 public static final int POSITION_NONE = -2; 83 84 /** 85 * Return the number of views available. 86 */ getCount()87 public abstract int getCount(); 88 89 /** 90 * Called when a change in the shown pages is going to start being made. 91 * @param container The containing View which is displaying this adapter's 92 * page views. 93 */ startUpdate(ViewGroup container)94 public void startUpdate(ViewGroup container) { 95 startUpdate((View) container); 96 } 97 98 /** 99 * Create the page for the given position. The adapter is responsible 100 * for adding the view to the container given here, although it only 101 * must ensure this is done by the time it returns from 102 * {@link #finishUpdate(android.view.ViewGroup)}. 103 * 104 * @param container The containing View in which the page will be shown. 105 * @param position The page position to be instantiated. 106 * @return Returns an Object representing the new page. This does not 107 * need to be a View, but can be some other container of the page. 108 */ instantiateItem(ViewGroup container, int position)109 public Object instantiateItem(ViewGroup container, int position) { 110 return instantiateItem((View) container, position); 111 } 112 113 /** 114 * Remove a page for the given position. The adapter is responsible 115 * for removing the view from its container, although it only must ensure 116 * this is done by the time it returns from {@link #finishUpdate(android.view.ViewGroup)}. 117 * 118 * @param container The containing View from which the page will be removed. 119 * @param position The page position to be removed. 120 * @param object The same object that was returned by 121 * {@link #instantiateItem(android.view.View, int)}. 122 */ destroyItem(ViewGroup container, int position, Object object)123 public void destroyItem(ViewGroup container, int position, Object object) { 124 destroyItem((View) container, position, object); 125 } 126 127 /** 128 * Called to inform the adapter of which item is currently considered to 129 * be the "primary", that is the one show to the user as the current page. 130 * 131 * @param container The containing View from which the page will be removed. 132 * @param position The page position that is now the primary. 133 * @param object The same object that was returned by 134 * {@link #instantiateItem(android.view.View, int)}. 135 */ setPrimaryItem(ViewGroup container, int position, Object object)136 public void setPrimaryItem(ViewGroup container, int position, Object object) { 137 setPrimaryItem((View) container, position, object); 138 } 139 140 /** 141 * Called when the a change in the shown pages has been completed. At this 142 * point you must ensure that all of the pages have actually been added or 143 * removed from the container as appropriate. 144 * @param container The containing View which is displaying this adapter's 145 * page views. 146 */ finishUpdate(ViewGroup container)147 public void finishUpdate(ViewGroup container) { 148 finishUpdate((View) container); 149 } 150 151 /** 152 * Called when a change in the shown pages is going to start being made. 153 * @param container The containing View which is displaying this adapter's 154 * page views. 155 * 156 * @deprecated Use {@link #startUpdate(android.view.ViewGroup)} 157 */ startUpdate(View container)158 public void startUpdate(View container) { 159 } 160 161 /** 162 * Create the page for the given position. The adapter is responsible 163 * for adding the view to the container given here, although it only 164 * must ensure this is done by the time it returns from 165 * {@link #finishUpdate(android.view.ViewGroup)}. 166 * 167 * @param container The containing View in which the page will be shown. 168 * @param position The page position to be instantiated. 169 * @return Returns an Object representing the new page. This does not 170 * need to be a View, but can be some other container of the page. 171 * 172 * @deprecated Use {@link #instantiateItem(android.view.ViewGroup, int)} 173 */ instantiateItem(View container, int position)174 public Object instantiateItem(View container, int position) { 175 throw new UnsupportedOperationException( 176 "Required method instantiateItem was not overridden"); 177 } 178 179 /** 180 * Remove a page for the given position. The adapter is responsible 181 * for removing the view from its container, although it only must ensure 182 * this is done by the time it returns from {@link #finishUpdate(android.view.View)}. 183 * 184 * @param container The containing View from which the page will be removed. 185 * @param position The page position to be removed. 186 * @param object The same object that was returned by 187 * {@link #instantiateItem(android.view.View, int)}. 188 * 189 * @deprecated Use {@link #destroyItem(android.view.ViewGroup, int, Object)} 190 */ destroyItem(View container, int position, Object object)191 public void destroyItem(View container, int position, Object object) { 192 throw new UnsupportedOperationException("Required method destroyItem was not overridden"); 193 } 194 195 /** 196 * Called to inform the adapter of which item is currently considered to 197 * be the "primary", that is the one show to the user as the current page. 198 * 199 * @param container The containing View from which the page will be removed. 200 * @param position The page position that is now the primary. 201 * @param object The same object that was returned by 202 * {@link #instantiateItem(android.view.View, int)}. 203 * 204 * @deprecated Use {@link #setPrimaryItem(android.view.ViewGroup, int, Object)} 205 */ setPrimaryItem(View container, int position, Object object)206 public void setPrimaryItem(View container, int position, Object object) { 207 } 208 209 /** 210 * Called when the a change in the shown pages has been completed. At this 211 * point you must ensure that all of the pages have actually been added or 212 * removed from the container as appropriate. 213 * @param container The containing View which is displaying this adapter's 214 * page views. 215 * 216 * @deprecated Use {@link #finishUpdate(android.view.ViewGroup)} 217 */ finishUpdate(View container)218 public void finishUpdate(View container) { 219 } 220 221 /** 222 * Determines whether a page View is associated with a specific key object 223 * as returned by {@link #instantiateItem(android.view.ViewGroup, int)}. This method is 224 * required for a PagerAdapter to function properly. 225 * 226 * @param view Page View to check for association with <code>object</code> 227 * @param object Object to check for association with <code>view</code> 228 * @return true if <code>view</code> is associated with the key object <code>object</code> 229 */ isViewFromObject(View view, Object object)230 public abstract boolean isViewFromObject(View view, Object object); 231 232 /** 233 * Save any instance state associated with this adapter and its pages that should be 234 * restored if the current UI state needs to be reconstructed. 235 * 236 * @return Saved state for this adapter 237 */ saveState()238 public Parcelable saveState() { 239 return null; 240 } 241 242 /** 243 * Restore any instance state associated with this adapter and its pages 244 * that was previously saved by {@link #saveState()}. 245 * 246 * @param state State previously saved by a call to {@link #saveState()} 247 * @param loader A ClassLoader that should be used to instantiate any restored objects 248 */ restoreState(Parcelable state, ClassLoader loader)249 public void restoreState(Parcelable state, ClassLoader loader) { 250 } 251 252 /** 253 * Called when the host view is attempting to determine if an item's position 254 * has changed. Returns {@link #POSITION_UNCHANGED} if the position of the given 255 * item has not changed or {@link #POSITION_NONE} if the item is no longer present 256 * in the adapter. 257 * 258 * <p>The default implementation assumes that items will never 259 * change position and always returns {@link #POSITION_UNCHANGED}. 260 * 261 * @param object Object representing an item, previously returned by a call to 262 * {@link #instantiateItem(android.view.View, int)}. 263 * @return object's new position index from [0, {@link #getCount()}), 264 * {@link #POSITION_UNCHANGED} if the object's position has not changed, 265 * or {@link #POSITION_NONE} if the item is no longer present. 266 */ getItemPosition(Object object)267 public int getItemPosition(Object object) { 268 return POSITION_UNCHANGED; 269 } 270 271 /** 272 * This method should be called by the application if the data backing this adapter has changed 273 * and associated views should update. 274 */ notifyDataSetChanged()275 public void notifyDataSetChanged() { 276 mObservable.notifyChanged(); 277 } 278 279 /** 280 * Register an observer to receive callbacks related to the adapter's data changing. 281 * 282 * @param observer The {@link android.database.DataSetObserver} which will receive callbacks. 283 */ registerDataSetObserver(DataSetObserver observer)284 public void registerDataSetObserver(DataSetObserver observer) { 285 mObservable.registerObserver(observer); 286 } 287 288 /** 289 * Unregister an observer from callbacks related to the adapter's data changing. 290 * 291 * @param observer The {@link android.database.DataSetObserver} which will be unregistered. 292 */ unregisterDataSetObserver(DataSetObserver observer)293 public void unregisterDataSetObserver(DataSetObserver observer) { 294 mObservable.unregisterObserver(observer); 295 } 296 297 /** 298 * This method may be called by the ViewPager to obtain a title string 299 * to describe the specified page. This method may return null 300 * indicating no title for this page. The default implementation returns 301 * null. 302 * 303 * @param position The position of the title requested 304 * @return A title for the requested page 305 */ getPageTitle(int position)306 public CharSequence getPageTitle(int position) { 307 return null; 308 } 309 310 /** 311 * Returns the proportional width of a given page as a percentage of the 312 * ViewPager's measured width from (0.f-1.f] 313 * 314 * @param position The position of the page requested 315 * @return Proportional width for the given page position 316 */ getPageWidth(int position)317 public float getPageWidth(int position) { 318 return 1.f; 319 } 320 } 321