Android listview row delete animation -
what best way animate listview row?
i keep trying like:
final animation animation = animationutils.loadanimation(getactivity(), r.anim.translate_up_fade_anim); animation.setanimationlistener(new animationlistener() { @override public void onanimationstart(animation animation) { } @override public void onanimationrepeat(animation animation) { } @override public void onanimationend(animation animation) { datasource.remove(index); adapter.notifydatasetchanged(); } }); view.startanimation(animation);
but animation isn't want. think need animate every other listview row except deleted one, open suggestions.
here's sourcee code (for android 4.1 guess).
http://developer.android.com/shareables/devbytes/listviewremovalanimation.zip
here's link video
http://www.youtube.com/watch?list=plwz5rj2ekkc_xogcruksokkjewfjzrkv0&v=ychnai9kji4
here's blog google engineer
http://www.graphics-geek.blogspot.in.
you have videos know what's happening. know how port previous versions.
example:
listviewremovalanimation.java
public class listviewremovalanimation extends activity { stablearrayadapter madapter; listview mlistview; backgroundcontainer mbackgroundcontainer; boolean mswiping = false; boolean mitempressed = false; hashmap<long, integer> mitemidtopmap = new hashmap<long, integer>(); private static final int swipe_duration = 250; private static final int move_duration = 150; @override protected void oncreate(bundle savedinstancestate) { super.oncreate(savedinstancestate); setcontentview(r.layout.activity_list_view_deletion); mbackgroundcontainer = (backgroundcontainer) findviewbyid(r.id.listviewbackground); mlistview = (listview) findviewbyid(r.id.listview); android.util.log.d("debug", "d=" + mlistview.getdivider()); final arraylist<string> cheeselist = new arraylist<string>(); (int = 0; < cheeses.scheesestrings.length; ++i) { cheeselist.add(cheeses.scheesestrings[i]); } madapter = new stablearrayadapter(this,r.layout.opaque_text_view, cheeselist, mtouchlistener); mlistview.setadapter(madapter); } /** * handle touch events fade/move dragged items swiped out */ private view.ontouchlistener mtouchlistener = new view.ontouchlistener() { float mdownx; private int mswipeslop = -1; @override public boolean ontouch(final view v, motionevent event) { if (mswipeslop < 0) { mswipeslop = viewconfiguration.get(listviewremovalanimation.this). getscaledtouchslop(); } switch (event.getaction()) { case motionevent.action_down: if (mitempressed) { // multi-item swipes not handled return false; } mitempressed = true; mdownx = event.getx(); break; case motionevent.action_cancel: v.setalpha(1); v.settranslationx(0); mitempressed = false; break; case motionevent.action_move: { float x = event.getx() + v.gettranslationx(); float deltax = x - mdownx; float deltaxabs = math.abs(deltax); if (!mswiping) { if (deltaxabs > mswipeslop) { mswiping = true; mlistview.requestdisallowintercepttouchevent(true); mbackgroundcontainer.showbackground(v.gettop(), v.getheight()); } } if (mswiping) { v.settranslationx((x - mdownx)); v.setalpha(1 - deltaxabs / v.getwidth()); } } break; case motionevent.action_up: { // user let go - figure out whether animate view out, or place if (mswiping) { float x = event.getx() + v.gettranslationx(); float deltax = x - mdownx; float deltaxabs = math.abs(deltax); float fractioncovered; float endx; float endalpha; final boolean remove; if (deltaxabs > v.getwidth() / 4) { // greater quarter of width - animate out fractioncovered = deltaxabs / v.getwidth(); endx = deltax < 0 ? -v.getwidth() : v.getwidth(); endalpha = 0; remove = true; } else { // not far enough - animate fractioncovered = 1 - (deltaxabs / v.getwidth()); endx = 0; endalpha = 1; remove = false; } // animate position , alpha of swiped item // note: simplified version of swipe behavior, // purposes of demo animation. real version should use // velocity (via velocitytracker class) send item off or // @ appropriate speed. long duration = (int) ((1 - fractioncovered) * swipe_duration); mlistview.setenabled(false); v.animate().setduration(duration). alpha(endalpha).translationx(endx). withendaction(new runnable() { @override public void run() { // restore animated values v.setalpha(1); v.settranslationx(0); if (remove) { animateremoval(mlistview, v); } else { mbackgroundcontainer.hidebackground(); mswiping = false; mlistview.setenabled(true); } } }); } } mitempressed = false; break; default: return false; } return true; } }; /** * method animates other views in listview container (not including ignoreview) * final positions. called after ignoreview has been removed * adapter, before layout has been run. approach here figure out * now, allow layout run, figure out after * layout, , run animations between of start/end positions. */ private void animateremoval(final listview listview, view viewtoremove) { int firstvisibleposition = listview.getfirstvisibleposition(); (int = 0; < listview.getchildcount(); ++i) { view child = listview.getchildat(i); if (child != viewtoremove) { int position = firstvisibleposition + i; long itemid = madapter.getitemid(position); mitemidtopmap.put(itemid, child.gettop()); } } // delete item adapter int position = mlistview.getpositionforview(viewtoremove); madapter.remove(madapter.getitem(position)); final viewtreeobserver observer = listview.getviewtreeobserver(); observer.addonpredrawlistener(new viewtreeobserver.onpredrawlistener() { public boolean onpredraw() { observer.removeonpredrawlistener(this); boolean firstanimation = true; int firstvisibleposition = listview.getfirstvisibleposition(); (int = 0; < listview.getchildcount(); ++i) { final view child = listview.getchildat(i); int position = firstvisibleposition + i; long itemid = madapter.getitemid(position); integer starttop = mitemidtopmap.get(itemid); int top = child.gettop(); if (starttop != null) { if (starttop != top) { int delta = starttop - top; child.settranslationy(delta); child.animate().setduration(move_duration).translationy(0); if (firstanimation) { child.animate().withendaction(new runnable() { public void run() { mbackgroundcontainer.hidebackground(); mswiping = false; mlistview.setenabled(true); } }); firstanimation = false; } } } else { // animate new views along others. catch did not // exist in start state, must calculate starting position // based on neighboring views. int childheight = child.getheight() + listview.getdividerheight(); starttop = top + (i > 0 ? childheight : -childheight); int delta = starttop - top; child.settranslationy(delta); child.animate().setduration(move_duration).translationy(0); if (firstanimation) { child.animate().withendaction(new runnable() { public void run() { mbackgroundcontainer.hidebackground(); mswiping = false; mlistview.setenabled(true); } }); firstanimation = false; } } } mitemidtopmap.clear(); return true; } }); } }
stablearrayadapter.java
public class stablearrayadapter extends arrayadapter<string> { hashmap<string, integer> midmap = new hashmap<string, integer>(); view.ontouchlistener mtouchlistener; public stablearrayadapter(context context, int textviewresourceid, list<string> objects, view.ontouchlistener listener) { super(context, textviewresourceid, objects); mtouchlistener = listener; (int = 0; < objects.size(); ++i) { midmap.put(objects.get(i), i); } } @override public long getitemid(int position) { string item = getitem(position); return midmap.get(item); } @override public boolean hasstableids() { return true; } @override public view getview(int position, view convertview, viewgroup parent) { view view = super.getview(position, convertview, parent); if (view != convertview) { // add touch listener every new view track swipe motion view.setontouchlistener(mtouchlistener); } return view; } }
cheese.java
public class cheeses { public static final string[] scheesestrings = { "abbaye de belloc", "abbaye du mont des cats", "abertam", "abondance", "ackawi", "acorn", "adelost", "affidelice au chablis", "afuega'l pitu", "airag", "airedale", "aisy cendre", "allgauer emmentaler", "alverca", "ambert", "american cheese", "ami du chambertin", "anejo enchilado", "anneau du vic-bilh", "anthoriro", "appenzell", "aragon", "ardi gasna", "ardrahan", "armenian string", "aromes au gene de marc", "asadero", "asiago", "aubisque pyrenees", "autun", "avaxtskyr", "baby swiss", "babybel", "baguette laonnaise", "bakers", "baladi", "balaton", "bandal", "banon", "barry's bay cheddar", "basing", "basket cheese", "bath cheese", "bavarian bergkase", "baylough", "beaufort", "beauvoorde", "beenleigh blue", "beer cheese", "bel paese", "bergader", "bergere bleue", "berkswell", "beyaz peynir", "bierkase", "bishop kennedy", "blarney", "bleu d'auvergne", "bleu de gex", "bleu de laqueuille", "bleu de septmoncel", "bleu des causses", "blue", "blue castello", "blue rathgore", "blue vein (australian)", "blue vein cheeses", "bocconcini", "bocconcini (australian)", "boeren leidenkaas", "bonchester", "bosworth", "bougon", "boule du roves", "boulette d'avesnes", "boursault", "boursin", "bouyssou", "bra", "braudostur", "breakfast cheese", "brebis du lavort", "brebis du lochois", "brebis du puyfaucon", "bresse bleu", "brick", "brie", "brie de meaux", "brie de melun", "brillat-savarin", "brin", "brin d' amour", "brin d'amour", "brinza (burduf brinza)", "briquette de brebis", "briquette du forez", "broccio", "broccio demi-affine", "brousse du rove", "bruder basil", "brusselae kaas (fromage de bruxelles)", "bryndza", "buchette d'anjou", "buffalo", "burgos", "butte", "butterkase", "button (innes)", "buxton blue", "cabecou", "caboc", "cabrales", "cachaille", "caciocavallo", "caciotta", "caerphilly", "cairnsmore", "calenzana", "cambazola", "camembert de normandie", "canadian cheddar", "canestrato", "cantal", "caprice des dieux", "capricorn goat", "capriole banon", "carre de l'est", "casciotta di urbino", "cashel blue", "castellano", "castelleno", "castelmagno", "castelo branco", "castigliano", "cathelain", "celtic promise", "cendre d'olivet", "cerney", "chabichou", "chabichou du poitou", "chabis de gatine", "chaource", "charolais", "chaumes", "cheddar", "cheddar clothbound", "cheshire", "chevres", "chevrotin des aravis", "chontaleno", "civray", "coeur de camembert au calvados", "coeur de chevre", "colby", "cold pack", "comte", "coolea", "cooleney", "coquetdale", "corleggy", "cornish pepper", "cotherstone", "cotija", "cottage cheese", "cottage cheese (australian)", "cougar gold", "coulommiers", "coverdale", "crayeux de roncq", "cream cheese", "cream havarti", "crema agria", "crema mexicana", "creme fraiche", "crescenza", "croghan", "crottin de chavignol", "crottin du chavignol", "crowdie", "crowley", "cuajada", "curd", "cure nantais", "curworthy", "cwmtawe pecorino", "cypress grove chevre", "danablu (danish blue)", "danbo", "danish fontina", "daralagjazsky", "dauphin", "delice des fiouves", "denhany dorset drum", "derby", "dessertnyj belyj", "devon blue", "devon garland", "dolcelatte", "doolin", "doppelrhamstufel", "dorset blue vinney", "double gloucester", "double worcester", "dreux la feuille", "dry jack", "duddleswell", "dunbarra", "dunlop", "dunsyre blue", "duroblando", "durrus", "dutch mimolette (commissiekaas)", "edam", "edelpilz", "emental grand cru", "emlett", "emmental", "epoisses de bourgogne", "esbareich", "esrom", "etorki", "evansdale farmhouse brie", "evora de l'alentejo", "exmoor blue", "explorateur", "feta", "feta (australian)", "figue", "filetta", "fin-de-siecle", "finlandia swiss", "finn", "fiore sardo", "fleur du maquis", "flor de guia", "flower marie", "folded", "folded cheese mint", "fondant de brebis", "fontainebleau", "fontal", "fontina val d'aosta", "formaggio di capra", "fougerus", "four herb gouda", "fourme d' ambert", "fourme de haute loire", "fourme de montbrison", "fresh jack", "fresh mozzarella", "fresh ricotta", "fresh truffles", "fribourgeois", "friesekaas", "friesian", "friesla", "frinault", "fromage raclette", "fromage corse", "fromage de montagne de savoie", "fromage frais", "fruit cream cheese", "frying cheese", "fynbo", "gabriel", "galette du paludier", "galette lyonnaise", "galloway goat's milk gems", "gammelost", "gaperon l'ail", "garrotxa", "gastanberra", "geitost", "gippsland blue", "gjetost", "gloucester", "golden cross", "gorgonzola", "gornyaltajski", "gospel green", "gouda", "goutu", "gowrie", "grabetto", "graddost", "grafton village cheddar", "grana", "grana padano", "grand vatel", "grataron d' areches", "gratte-paille", "graviera", "greuilh", "greve", "gris de lille", "gruyere", "gubbeen", "guerbigny", "halloumi", "halloumy (australian)", "haloumi-style cheese", "harbourne blue", "havarti", "heidi gruyere", "hereford hop", "herrgardsost", "herriot farmhouse", "herve", "hipi iti", "hubbardston blue cow", "hushallsost", "iberico", "idaho goatster", "idiazabal", "il boschetto al tartufo", "ile d'yeu", "isle of mull", "jarlsberg", "jermi tortes", "jibneh arabieh", "jindi brie", "jubilee blue", "juustoleipa", "kadchgall", "kaseri", "kashta", "kefalotyri", "kenafa", "kernhem", "kervella affine", "kikorangi", "king island cape wickham brie", "king river gold", "klosterkaese", "knockalara", "kugelkase", "l'aveyronnais", "l'ecir de l'aubrac", "la taupiniere", "la vache qui rit", "laguiole", "lairobell", "lajta", "lanark blue", "lancashire", "langres", "lappi", "laruns", "lavistown", "le brin", "le fium orbo", "le lacandou", "le roule", "leafield", "lebbene", "leerdammer", "leicester", "leyden", "limburger", "lincolnshire poacher", "lingot saint bousquet d'orb", "liptauer", "little rydings", "livarot", "llanboidy", "llanglofan farmhouse", "loch arthur farmhouse", "loddiswell avondale", "longhorn", "lou palou", "lou pevre", "lyonnais", "maasdam", "macconais", "mahoe aged gouda", "mahon", "malvern", "mamirolle", "manchego", "manouri", "manur", "marble cheddar", "marbled cheeses", "maredsous", "margotin", "maribo", "maroilles", "mascares", "mascarpone", "mascarpone (australian)", "mascarpone torta", "matocq", "maytag blue", "meira", "menallack farmhouse", "menonita", "meredith blue", "mesost", "metton (cancoillotte)", "meyer vintage gouda", "mihalic peynir", "milleens", "mimolette", "mine-gabhar", "mini baby bells", "mixte", "molbo", "monastery cheeses", "mondseer", "mont d'or lyonnais", "montasio", "monterey jack", "monterey jack dry", "morbier", "morbier cru de montagne", "mothais la feuille", "mozzarella", "mozzarella (australian)", "mozzarella di bufala", "mozzarella fresh, in water", "mozzarella rolls", "munster", "murol", "mycella", "myzithra", "naboulsi", "nantais", "neufchatel", "neufchatel (australian)", "niolo", "nokkelost", "northumberland", "oaxaca", "olde york", "olivet au foin", "olivet bleu", "olivet cendre", "orkney mature cheddar", "orla", "oschtjepka", "ossau fermier", "ossau-iraty", "oszczypek", "oxford blue", "p'tit berrichon", "palet de babligny", "paneer", "panela", "pannerone", "pant ys gawn", "parmesan (parmigiano)", "parmigiano reggiano", "pas de l'escalette", "passendale", "pasteurized processed", "pate de fromage", "patefine fort", "pave d'affinois", "pave d'auge", "pave de chirac", "pave du berry", "pecorino", "pecorino in walnut leaves", "pecorino romano", "peekskill pyramid", "pelardon des cevennes", "pelardon des corbieres", "penamellera", "penbryn", "pencarreg", "perail de brebis", "petit morin", "petit pardou", "petit-suisse", "picodon de chevre", "picos de europa", "piora", "pithtviers au foin", "plateau de herve", "plymouth cheese", "podhalanski", "poivre d'ane", "polkolbin", "pont l'eveque", "port nicholson", "port-salut", "postel", "pouligny-saint-pierre", "pourly", "prastost", "pressato", "prince-jean", "processed cheddar", "provolone", "provolone (australian)", "pyengana cheddar", "pyramide", "quark", "quark (australian)", "quartirolo lombardo", "quatre-vents", "quercy petit", "queso blanco", "queso blanco con frutas --pina y mango", "queso de murcia", "queso del montsec", "queso del tietar", "queso fresco", "queso fresco (adobera)", "queso iberico", "queso jalapeno", "queso majorero", "queso media luna", "queso para frier", "queso quesadilla", "rabacal", "raclette", "ragusano", "raschera", "reblochon", "red leicester", "regal de la dombes", "reggianito", "remedou", "requeson", "richelieu", "ricotta", "ricotta (australian)", "ricotta salata", "ridder", "rigotte", "rocamadour", "rollot", "romano", "romans part dieu", "roncal", "roquefort", "roule", "rouleau de beaulieu", "royalp tilsit", "rubens", "rustinu", "saaland pfarr", "saanenkaese", "saga", "sage derby", "sainte maure", "saint-marcellin", "saint-nectaire", "saint-paulin", "salers", "samso", "san simon", "sancerre", "sap sago", "sardo", "sardo egyptian", "sbrinz", "scamorza", "schabzieger", "schloss", "selles sur cher", "selva", "serat", "seriously strong cheddar", "serra da estrela", "sharpam", "shelburne cheddar", "shropshire blue", "siraz", "sirene", "smoked gouda", "somerset brie", "sonoma jack", "sottocenare al tartufo", "soumaintrain", "sourire lozerien", "spenwood", "sraffordshire organic", "st. agur blue cheese", "stilton", "stinking bishop", "string", "sussex slipcote", "sveciaost", "swaledale", "sweet style swiss", "swiss", "syrian (armenian string)", "tala", "taleggio", "tamie", "tasmania highland chevre log", "taupiniere", "teifi", "telemea", "testouri", "tete de moine", "tetilla", "texas goat cheese", "tibet", "tillamook cheddar", "tilsit", "timboon brie", "toma", "tomme brulee", "tomme d'abondance", "tomme de chevre", "tomme de romans", "tomme de savoie", "tomme des chouans", "tommes", "torta del casar", "toscanello", "touree de l'aubier", "tourmalet", "trappe (veritable)", "trois cornes de vendee", "tronchon", "trou du cru", "truffe", "tupi", "turunmaa", "tymsboro", "tyn grug", "tyning", "ubriaco", "ulloa", "vacherin-fribourgeois", "valencay", "vasterbottenost", "venaco", "vendomois", "vieux corse", "vignotte", "vulscombe", "waimata farmhouse blue", "washed rind cheese (australian)", "waterloo", "weichkaese", "wellington", "wensleydale", "white stilton", "whitestone farmhouse", "wigmore", "woodside cabecou", "xanadu", "xynotyro", "yarg cornish", "yarra valley pyramid", "yorkshire blue", "zamorano", "zanetti grana padano", "zanetti parmigiano reggiano" }; }
backgroundcontainer.java
public class backgroundcontainer extends framelayout { boolean mshowing = false; drawable mshadowedbackground; int mopenareatop, mopenareabottom, mopenareaheight; boolean mupdatebounds = false; public backgroundcontainer(context context) { super(context); init(); } public backgroundcontainer(context context, attributeset attrs) { super(context, attrs); init(); } public backgroundcontainer(context context, attributeset attrs, int defstyle) { super(context, attrs, defstyle); init(); } private void init() { mshadowedbackground = getcontext().getresources().getdrawable(r.drawable.shadowed_background); } public void showbackground(int top, int bottom) { setwillnotdraw(false); mopenareatop = top; mopenareaheight = bottom; mshowing = true; mupdatebounds = true; } public void hidebackground() { setwillnotdraw(true); mshowing = false; } @override protected void ondraw(canvas canvas) { if (mshowing) { if (mupdatebounds) { mshadowedbackground.setbounds(0, 0, getwidth(), mopenareaheight); } canvas.save(); canvas.translate(0, mopenareatop); mshadowedbackground.draw(canvas); canvas.restore(); } } }
layout files
<linearlayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:orientation="vertical" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".listviewanimations" > <view class="com.example.android.listviewremovalanimation.backgroundcontainer" android:layout_width="match_parent" android:layout_height="match_parent" android:id="@+id/listviewbackground"> <listview android:id="@+id/listview" android:divider="@null" android:dividerheight="0dp" android:layout_width="match_parent" android:layout_height="wrap_content" /> </view> </linearlayout>
opaque_text_view.xml
<textview xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="wrap_content" android:background="@drawable/tv_background_with_divider" android:textappearance="?android:attr/textappearancelistitemsmall" android:gravity="center_vertical" android:paddingstart="?android:attr/listpreferreditempaddingstart" android:paddingend="?android:attr/listpreferreditempaddingend" android:minheight="?android:attr/listpreferreditemheightsmall" />
Comments
Post a Comment