• úvod
  • témata
  • události
  • tržiště
  • diskuze
  • nástěnka
  • přihlásit
    registrace
    ztracené heslo?
    LWEEKAndroid development
    VIRTUALVOID
    VIRTUALVOID --- ---
    DATEL: co sa tyka tej druhej otazky tak s tym ti neporadim. mozno len tak, ze by som si nacital vsetky data in-memory a zmeny robil iba on-demand.
    VIRTUALVOID
    VIRTUALVOID --- ---
    DATEL: ja to robim tak, ze pridam vsetky fragmenty (cez kod, fragmentManager) do kontajneru cez metodu add (plus nejaky ten tag coby stringovy identifikator) a ten ktory chcem aby bol zobrazeny ako prvy dam show. tie ostatne hide.
    vtedy sa to nerecreatuje. chce to teda, ale trochu viac kodu na handlovanie.
    DATEL
    DATEL --- ---
    Díky všem za info. Měl bych další dotaz, respektive nejasnost. V originální dokumentaci o použití fragmentů (http://developer.android.com/training/basics/fragments/fragment-ui.html) je poznámka:

    Note: When you remove or replace a fragment and add the transaction to the back stack, the fragment that is removed is stopped (not destroyed). If the user navigates back to restore the fragment, it restarts. If you do not add the transaction to the back stack, then the fragment is destroyed when removed or replaced.

    Já při výměně fragment dávám do back stacku, ale i přesto když pak vyměňuju fragment, tak se mi na tom původním volá onDestroyView a po "uzavření" nového fragmentu (což je formulář pro novou položku) zavolám getSupportFragmentManager().popBackStack(); a pak se tedy pro původní fragment volá onCreateView atd.

    Další věc, co při konverzi na jednu aktivitu začala zlobit, je uzavírání kurzoru a databáze ve fragmentu pro seznam. Dokud jsem otevíral ten formulář pomocí aktivity, tak to jakžtakž fungovalo (viz. předchozí příspěvky), ale když jsem to předělal, tak mi při vyvolání nového fragmentu (s formulářem) aplikace spadla s výjimkou "attemp to re-open an already-closed object" a týkalo se to právě toho kurzoru. Jenže vůbec netuším, proč. Dal jsem si breakpointu do všech metod a padá to po onStop ve fragmentu se seznamem. To uzavření kurzoru a DB je v onPause. Přitom ten onStop proběhl v pořádku. Do onDestroyView mi debugger už nedošel a snažil se otevírat vnitřní zdrojáky Androidu, které tam nemám. Máte nějaký tip, na co se podívat?
    DRIZDIK
    DRIZDIK --- ---
    Každý přístup je dobrý pro jiné použití. Teď například dělám aplikaci, kde hlavní aktivita spravuje loading v intervalech pomocí loaderu a na ní pouze vyměňuju spoustu fragmentů. Ty pokud mají nějakou akci, tak jí přidají do fronty a aktivita jí odešle na server a o výsledku (změně dat) notifikuje všechny observery, takže si každý fragment může obnovit data. Stejně tak po každý loadu na aktivitě jde notofikace pro observery.

    Ale implementovat všechny metody na aktivitě mi nakonec přišlo většinou jako zbytečný datlování a o nějakým decouplinfu se tam moc nedalo mluvit.
    VIRTUALVOID
    VIRTUALVOID --- ---
    DATEL: skus obe, uvidis co ti bude vyhovovat viac :)
    DATEL
    DATEL --- ---
    DRIZDIK: hm, zajímavé, takže každý fragment máš v podstatě samostatný "blackbox", který funguje kompletně sám o sobě včetně načítání a ukládání dat? Zatím jsem právě všude viděl (knihy, tutoriály i android dokumentace) to, že se má ve fragmentu vytvořit interface, který aktivita musí implementovat a tyhle ukládací věci se pak dělají v aktivitě. Na druhou stranu tvé řešení má výhodu, že fragment je opravdu samostatná jednotka, kterou když přesunu do jiné aktivity, tak by měla víceméně fungovat, aniž bych musel přenášet i metody z jedné aktivity do druhé.

    No, tak teď fakt nevím, kterou cestou se dát :)
    DRIZDIK
    DRIZDIK --- ---
    DATEL:
    já tohle řeším tak, že tyhle data (nejlépe pouze id) předám fragmentu v Argumets bundlu. Fragment se poté sám postará o zobrazení progress, načtení na pozadí vlastním loaderem a zbrazení případně uložení. Snažím se ho jako funkční celek oddělit od aktivity, aby byl použitelnej i na jíném místě.
    Mezi fragmenty komunikuju pomocí Observer patternu a funguje to relativně dobře.

    Rozdělení na je celkem na tobě. Někdo se snaží vše řešit v jednom, někdo odděluje, aby se zachovávali intenty, které ti pak pomůžou pži zabití aktivity obnovit předchozí stav - to je velice důležitý.
    DATEL
    DATEL --- ---
    VIRTUALVOID: super, tohle je mi už jasný, díky :)
    VIRTUALVOID
    VIRTUALVOID --- ---
    DATEL: co zoznam, to aktivita. urob niekde menu/zoznam, ake veci vlastne chces urobit. potom si sprav mockupy...

    alebo inak to skusim napisat: co typ entity to samostatna aktivita+fragmenty
    DATEL
    DATEL --- ---
    VIRTUALVOID: VIRTUALVOID: Ok, tomu rozumím, ale pokud ta aplikace bude mít více různých seznamů, formulářu apod., tak v té hlavní aktivitě by ty akce mohly narůst velice rychle a ten počet 40 překročit. Ono už vlastně i to, že aktivita implementuje události fragmentů např. pro uložení dat po odeslání formuláře, naplňování formulářů z daty atd., způsobí její nabobtnání. Nebo co by vlastně samotný fragment měl všechno řešit? Pouze obsluhu svých View (eventuelně otevírání dalších aktivit/dialogů, např. pro výber datumu)? Veškeré ukládání dat pak předává právě aktivitě.
    VIRTUALVOID
    VIRTUALVOID --- ---
    hlavna aktivita by potom mala riesit cez callbacky operacie daneho fragmentu
    VIRTUALVOID
    VIRTUALVOID --- ---
    DATEL:
    Na jednu stranu pokud je mám všechny v jedné aktivitě, tak bude jednodušší komunikace mezi fragmenty, na druhou stranu, kdyby byla aplikace rozsáhlejší, tak mít všechno v jedné aktivitě je podle mě taky nesmysl.

    nieje to nezmysel. pretoze by si nikdy nemal mat 40 akcii v jednej aktivite, ale nejak logicky to rozdelovat.

    ja som doteraz vzdy siel cestou hlavna aktivita + jej logicky patriace fragmenty.
    v tomto pripade :
    hlavna aktivita :
    - fragment pre zobrazenie v liste
    - fragment pre editaciu/zobrazenie detailu

    na editaciu / detail nepotrebujes dva rozne fragmenty... len ked dany zaznam budes otvarat ako detail tak nastavis textboxy,etc. ako readonly. napr.
    DATEL
    DATEL --- ---
    VIRTUALVOID: díky za tip, jsem začátečník, takže parcelable neznám (respektive zahlédl jsem ho v dokumentaci), tak se na to podívám - předpokládám, že to umožňuje dávat do intentu jakýkoliv objekt, takže půjde nejspíš o implementaci nějakého rozhraní, pomocí kterého se v té třídě Record určí, co se ukládá atd.?

    A jak s těmi fragmenty? Blbý je, že ať kouknu kam chci, tak všechny tutoriály i EN knihy fragmenty nakousnou jen v základu a jako příklad tam dají seznam a vložení nové položky, ale třeba právě editaci již existující a právě přenos dat do fragmentu (eventuelně skrz další aktivitu) se nikde neřeší :(
    VIRTUALVOID
    VIRTUALVOID --- ---
    DATEL: sprav to ako parcelable, nech nemusis puttovat do bundlu objekty na per property basis
    DATEL
    DATEL --- ---
    A teď bych měl dotaz z jiného soudku. Základy jsem studoval ze seriálu na Zdrojáku. Rád bych se zeptal, jakým způsobem řešíte fragmenty ve spojitosti s activity.

    V tom tutoriálu (http://www.zdrojak.cz/clanky/vyvijime-pro-android-fragmenty-a-sqlite-databaze/) je totiž pro variantu "malé" obrazovky použita sólo aktivita pro fragment s formulářem a sólo aktivita pro zobrazení detailu (a samozřejmně hlavní aktivita). Oproti tomu v dokumentaci (http://developer.android.com/training/basics/fragments/communicating.html) používají pouze jednu hlavní aktivitu pro všechny fragmenty.

    Kdybych nepoužil fragmenty, tak je jasné, že pro každou zmíněnou akci budou sólo aktivity. Jakým způsobem je tedy lepší fragmenty začleňovat do aktivity? Na jednu stranu pokud je mám všechny v jedné aktivitě, tak bude jednodušší komunikace mezi fragmenty, na druhou stranu, kdyby byla aplikace rozsáhlejší, tak mít všechno v jedné aktivitě je podle mě taky nesmysl.

    Dá se tedy říci, že pro malé aplikace se dá použití varianta jedné aktivity pro všechny fragmenty a pro větší aplikace (co je větší aplikace?) pak rozdělení na více aktivit?

    Právě řeším v té jednoduché aplikaci to, že fragmenty mám ve více aktivitách a potřebuju při výběru položky ze seznamu (hlavní aktivita) otevřít formulář a naplnit ho daty z DB - což je až ve fragmentu, ale v jiné aktivitě. Takže pokud aktivita pro zobrazení fragmentu s formulářem vypadá takto:
    public class AddRecordActivity extends FragmentActivity implements
        AddRecordFragment.OnAddRecordListener {
        
        public static final String EXTRA_DATE = "date";
        public static final String EXTRA_DAILY_COUNTER = "dailyCounter";
        public static final String EXTRA_MAIN_COUNTER = "mainCounter";
        public static final String EXTRA_DESCRIPTION = "description";
        
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.add_record_container);
        }
    
        @Override
        public void onAddRecord(Record record) {
            Intent result = new Intent();
            result.putExtra(EXTRA_DATE, record.dateInMillis);
            result.putExtra(EXTRA_DAILY_COUNTER, record.dailyCounter);
            result.putExtra(EXTRA_MAIN_COUNTER, record.mainCounter);
            result.putExtra(EXTRA_DESCRIPTION, record.description);
    
            setResult(RESULT_OK, result);
            finish();
        }
        
    }
    

    a v hlavní aktivitě se tato aktivita otevírá takto:
                Intent i = new Intent(this, AddRecordActivity.class);
                i.putExtra(AddRecordActivity.EXTRA_DATE, record.dateInMillis);
                i.putExtra(AddRecordActivity.EXTRA_DAILY_COUNTER, record.dailyCounter);
                i.putExtra(AddRecordActivity.EXTRA_MAIN_COUNTER, record.mainCounter);
                i.putExtra(AddRecordActivity.EXTRA_DESCRIPTION, record.description);
                startActivityForResult(i, REQUEST_UPDATE_RECORD);
    

    tak tam budu muset dělat dalšího prostředníka pro přenesení dat do fragmentu.
    DATEL
    DATEL --- ---
    DRIZDIK: breaknout myslíš hodít si breakpointy pro debugger? No, ona ta výjimka je jen občas, zkouším otevírat dialogy, zavírat je atd. a vyskytne se to jen občas. Hlásí, že se finalizuje kurzor, který nebyl deaktivován nebo uzavřen. Problém je, že se to vyvolá jen někdy. Dle jedné situace jsem myslel, že se při otevření nové aktivity s tím fragmentem a formulářem nevyvolají onPause atd. pro fragment se seznamem a pak, když uložím novou položku, tak se vyvolá update seznamu, kde se vytváří znova kurzor a tom to vyhodí tu výjimku. Jenže už se mi i stalo, že mi to tu samou výjimku do LogCatu vyhodilo i ve chvíli, kdy jsem měl otevřený ten formulář pro novou položku a z něj otevřel ještě dialog pro výběr datumu - tady teda moc nechápu, proč se obnovoval seznam. Blbý je, že to je všechno náhodně, nedá se to přesně replikovat, takže nějaké ladění v debuggeru asi mi moc nepomůže :(
    GORG
    GORG --- ---
    DRIZDIK: Všiml jsem si, že se tam nějaký thread pool zřejmě vytváří... v DDMS jsem viděl právě hromadu těch vznikajících thread-pool- workerů. Nevím, jestli já to přímo můžu omezit, ale ten zmíněný Continuation zřejmě thread pool používá.
    DRIZDIK: Nezkoušel.. zkusím se na to příště zaměřit. Na konci to byl OutOfMemory exception, ale možná tam bylo předtím víc informací.
    GORG
    GORG --- ---
    REDGUY: Ja tam přímo ty thready nevytvářím. To dělá třetí strana formou classy Continuation, což je handler, co mi má volat po dokoneční jejich asynchronní operace. Já tedy přímo v tomhle případě ty thready nevytvářím. Všiml jsem si, že když to zahltím požadavky, dokáže se dostat do stavu, kdy přibývají nové thready cca 50 nových vláken za sekundu, dokonce aniž by přibývaly další požadavky (na další stahování). Zdálo se mi, že to souvisí s nějakým deadlockem na přistup k vnějším proměnným. To se mi, zdá se, podařilo skoro vyřešit, i když mi není přímo jasné, čím přesně mi to podařilo.

    situace se má konkrétně nějak takto:



    rssFeedController.requestAllItems(new Continuation<List<RSSItem>>() {



    @Override

    public void withValue(List<RSSItem> arg0, Exception arg1) {

    // tady je zpracování dat získaných v arg0

    // jejich vynášení ven mimo tuto funkci delalo ty problémy



    }
    DRIZDIK
    DRIZDIK --- ---
    DATEL: Zkoušel jsi breaknout kdy ti to umře? Asi se ti volá saveFragmentViewState až po tom co kurzor zavřeš. Jenže pokud ho neodstraníš z toho adaptéru, tak ten se ještě snaží uložit pozici v listu atd ..
    DRIZDIK
    DRIZDIK --- ---
    GORG: Na tohle určitě používat ThreadPool a ThreadPoolExecutor s nastaveným jedním threadem, nebo prostě minimum, aby se brzy objevili takovéhle problémy tím, že se ti začne ucpávat ta fronta. Docházející paměť se špatně hledá ..
    Kliknutím sem můžete změnit nastavení reklam