-
Notifications
You must be signed in to change notification settings - Fork 7
/
obsah.tex
761 lines (390 loc) · 123 KB
/
obsah.tex
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
%===============================================================================
\chapter{Úvod}
Hry pro mobilní zařízení, zejména telefony a tablety, v~současnosti zažívají nebývalý rozmach. Dalo by se říct, že jsou nyní tam, kde byly před patnácti až dvaceti lety hry počítačové. Zatímco typický tým vyvíjející hru pro počítač nebo herní konzoli čítá desítky, ale často i stovky členů, hru pro mobilní telefon je stále možné stvořit jen v~několika málo lidech.
Neustále narůstající výkon dovoluje přiblížit se kvalitě počítačových her. Těmi se nově díky mobilním zařízením může hráč zabavit například na cestách anebo při čekání na autobus. Velká většina lidí s~sebou neustále nosí výkonný počítač ve velmi malém balení, který je navíc vybaven množstvím sensorů a často i soustavným připojením k~internetu. To nabízí prostor pro velké množství inovací a je dobře, že se takové inovativní přístupy skutečně objevují.
Tato práce se pokusí přispět svým dílem navržením nové metody ovládání leteckých her pro mobilní zařízení. Nejdříve bude popsána platforma na které bude implementace probíhat, dále budou vysvětleny principy, díky kterým letí a manévruje skutečné letadlo. Tyto principy budou poté zjednodušeny pro použití v~prostředí her. Dále budou shrnuty současné možnosti ovládání herních letadel a navržena nová metoda jejich ovládání, založená na dotyku. Nad touto metodou bude vystavěna plnohodnotná letecká hra, jejíž vývoj bude popsán od prvotního návrhu až po její implementaci a publikování v~internetovém obchodě Google Play. Součástí práce bude také průzkum reakcí hráčů a jeho zhodnocení. Na závěr bude nabídnuta také kapitola shrnující další možný vývoj této hry.
\chapter{Vývoj her na platformě Android}
Andriod je operační systém pro mobilní zařízení, založený na Linuxovém jádře a poskytovaný ve formě otevřeného zdrojového kódu. Poprvé byl představen v~roce 2007, první mobilní telefon s~Androidem se dostal na trh o~rok později. Aktuálně dle statistik za první čtvrtletí roku 2013 má na poli \textit{chytrých mobilních telefonů} (smartphone) Android jakožto operační systém podíl přes 38\% \cite{statCounter} (měřeno na základě navštěvovanosti webových stránek z~mobilních telefonů) se zřejmým vzestupným trendem. Ačkoliv je Android použitelný pro množství různých zařízení (a implementace mimo pole mobilních telefonů se už nyní objevují), bude se tato práce soustřeďovat pouze na jeho využití v~mobilních zařízeních typu mobilního telefonu nebo tabletu. Informace v~následujících kapitolách budou vycházet z~oficiálního vývojářského portálu \cite{androidDevelopers}, knihy o~vývoji her pro Android \cite{libgdxKniha} a z~vlastních zkušeností a poznatků. Kapitola \ref{sec:libgdx} pak bude čerpat z~dokumentace knihovny Libgdx \cite{libgdx}.
\section{Architektura systému}
Architekturu systému lze rozdělit do několika úrovní. Na té nejvyšší jsou aplikace psané v~jazyce Java, které používají systémové funkce a rozhraní. Distribuují se v~balíčcích v~tzv. \textit{mezikódu} (Java bytecode). Mezikód je výstup kompilace a jako takový je přenositelný mezi platformami. Pro spuštění však potřebuje interpret, kterým je v~tomto případě virtuální stroj pojmenovaný Dalvik.
Dalvik samotný se nachází na nižší úrovni, tedy mezi programy a jejich součástmi v~nativním kódu, psanými v~jazyce C, C++ či jazyce symbolických instrukcí. Zde můžeme najít především sadu systémových knihoven, od kterých je vyžadován vysoký výkon. Jsou jimi například knihovny pro vykreslování (OpenGL), databáze (SQLite), šifrování (SSL) anebo multimédia. Jejich rozhraní je dostupné jak v~prostředí Javy, tak skrze \textit{JNI} (Java Native Interface). JNI je rozhraní umožňující psát programy v~některém z~výše uvedených jazyků s~možností přímého propojení se součástmi psanými v~Javě. Aplikace jako celek je pak rozdělená do dvou samostatně kompilovaných částí, jejichž spolupráci zajišťuje Dalvik.
Do nativního kódu je vhodné převést především provádění náročných operací v~kritickém čase, například simulaci fyziky ve 3D scéně anebo výpočty pro umělou inteligenci. Není ovšem vhodné, ačkoliv by se to mohlo zdát, implementovat zde celou aplikaci. Výkon nemusí být oproti interpretovanému kódu nutně vyšší, a naopak mohou nastat problémy s~přenositelností a optimalizací pro konkrétní zařízení \cite{jniPerf}. Tu by běžně prováděl Dalvik, zde ale záleží plně na programátorovi. Konkrétní přínosy i slabiny použití JNI budou dále rozvedeny v~kapitole \ref{sec:nativni-vyvoj}.
Na nejnižší úrovni je již zmíněné jádro psané v~jazyce C. To zajišťuje přístup k~hardwaru, multitasking, správu souborového systému a další základní úlohy. Přímo k~jádru však programátor v~drtivé většině případů nepotřebuje přistupovat. Funkce jádra programátor využije jen při výjimečných případech kdy vytváří systémovou aplikaci. Taková aplikace vyžaduje pro svůj běh vyšší oprávnění a běžný uživatel ji nemůže spustit. Své uplatnění ale nachází při modifikacích systému, kde jakožto předinstalované aplikace získávají vyšší oprávnění automaticky. Jelikož je Android otevřená platforma, je možné její součásti libovolně upravovat a takto vzniklý systém distribuovat.
\section{Verze systému a cílová zařízení}
Vývoj operačního systému vede firma Google, která vydává zdrojové kódy nových verzí. Ty se ovšem přímo do zařízení dostávají jen velmi zřídka (jde o~referenční modely telefonů a tabletů které si nechává vyrábět přímo Google). Typicky přebírají kód jednotliví výrobci a upravují či doplňují jej tak, aby fungoval pro konkrétní modely (nekompatibilita hrozí především u~specifického hardwaru jako je 3D displej a další). Často také výrobce přidává dodatečné aplikace, ať už pro plné využití možností zařízení nebo čistě pro reklamní účely. Jak bude výsledný systém vypadat, jaké funkce bude obsahovat a jaká zařízení bude podporovat je plně v~zodpovědnosti výrobců. Právě kvůli tomu jsou nejčastěji aktualizovány modely z~vyšší cenové třídy a levnější zařízení bývají opomíjena.
Důsledkem otevřenosti Androidu je jeho roztříštěnost co se verzí týče. Aktuální rozložení verzí \textit{API} (programátorské rozhraní, Application Programming Interface) k~dubnu 2013 je na obrázku \ref{fig:api-verze}, data pocházejí z~oficiálního zdroje \cite{statsDroid} a jsou měřena podle toho, jak se zařízení s~danými verzemi připojují k~obchodu Google Play. Statistika tedy nezobrazuje podíly verzí prodaných zařízení, nýbrž těch, která alespoň jednou navštívila oficiální obchod s~aplikacemi. Z~hlediska vývojáře má tato metodika mnohem vyšší vypovídací schopnost. Je vidět, že více než třetinový podíl má verze 10 (v~uživatelském značení jde o~Android verze 2.3.x) která je i nyní, po téměř dvou letech od prvního vydání, stále nasazována do nově prodávaných zařízení. Jde však především o~zařízení nejnižší cenové a výkonnostní třídy.
\begin{figure}[ht]
\begin{center}
\vspace{\baselineskip}
\includegraphics[width=\textwidth]{fig/api-versions.pdf}
\caption{Rozložení verzí systému Android k~dubnu 2013. Obecné aplikace musejí řešit kompatiblitu s~co nevětším množstvím verzí systému. Hry však používají pouze minimum systémových funkcí, a lze tak často pokrýt více než 94\% zařízení. Zbývající procenta představují nejméně výkonná zařízení s~nejstaršími verzemi systému. Zdroj dat: oficiální portál Android \cite{statsDroid}}
\label{fig:api-verze}
\end{center}
\end{figure}
Je často připomínanou pravdou, že právě zpětná kompatibilita se staršími verzemi systému znamená pro tvůrce aplikací množství práce navíc. V~případě vývoje her však toto neplatí, neboť ve skutečnosti využívají jen malou část API. Zde je z~programové části důležitá především podpora knihovny OpenGL ES ve verzi 2.0. Ta je nyní dostupná na více než 90\% zařízení a můžeme ji tedy považovat za standard. Mnohem důležitější je však technická vybavenost zařízení, zejména pak výkon procesoru, výkon grafického čipu a velikost jeho paměti. Zde jsou parametry často velmi rozdílné a relevantní srovnání či statistiky bohužel chybí. Nezanedbatelným parametrem jsou také rozměry displeje. Vzhledem k~různým hustotám zobrazení není vhodné uvádět pouze rozlišení. Větší vypovídací hodnotu z~hlediska dotykového ovládání má fyzická úhlopříčka displeje, která je v~současnosti nejvíce zastoupená v~rozměrech okolo 4 palců. \cite{statsDroid}
\section{Nativní vývoj her}
\label{sec:nativni-vyvoj}
Při zaměření konkrétně na vývoj grafických her se ukazuje, že hluboké znalosti systému nejsou nezbytné. Android umožňuje navrhovat velmi složitá a propracovaná uživatelská rozhraní, hernímu vývojáři však bude stačit jen nezbytné minimum. Podobně je tomu s~dalšími vymoženostmi systému.
Základním stavebním kamenem každé aplikace je tzv. \textit{Aktivita} (třída \texttt{Activity}). Zjednodušeně lze říct že Aktivita je okno aplikace, typicky zobrazované přes celý prostor displeje (s~výjimkou informační lišty a případných virtuálních kláves), se kterým může uživatel interagovat. Aktivita je rodičem jednotlivých elementů uživatelského rozhraní, případných dialogů, nabídek a dalších prvků. Pro potřeby hry běžně postačuje jediná aktivita jejímž potomkem je speciální oblast do které může vykreslovat OpenGL (oblast je třídy \texttt{GLSurfaceView}). Při vytoření této oblasti systém současně spustí speciální vlákno, které má přístup k~OpenGL kontextu, a může tedy provádět grafická volání. Toto vlákno je poté systémem pravidelně probouzeno tak, aby vykreslování probíhalo v~rychlosti co nejbližší ideálním 60 snímkům za sekundu. S~uživatelským rozhraním systému běžící hra až na výjimky nepracuje, vše probíhá v~oblasti vyhrazené pro vykreslování.
Kromě zobrazování je Aktivita důležitá také pro příjem vstupních událostí. Má možnost \textit{přihlásit} se u~systému ke sledování dotyků na obrazovce, stisků tlačítek, naklonění zařízení, případně stavů dalších sensorů. Tyto události přicházejí v~odděleném vlákně které nemá možnost vykreslování. Vlákno typicky upraví vnitřní stav hry a poté se opět uspí a čeká na další událost. Platí zde samozřejmě obdobné principy jako u~jiných vícevláknových prostředí. Jak vlákno pro vykreslování, tak i vlákno pro zpracování události nesmějí provádět blokující volání a neměly by provádět časově náročné výpočty. Pro takové případy nabízí Android možnost spuštění odděleného vlákna a postupy jak s~takovým vláknem asynchronně komunikovat, aniž by docházelo k~narušení plynulosti běhu aplikace, v~tomto případě poklesu počtu vykreslených snímků za vteřinu.
\subsection{Negativa nativního vývoje}
\label{sec:negativna-nativniho-vyvoje}
V~případě že zvolíme tvorbu hry čistě na základě Androidu bez mezičlánků typu frameworku nebo herního engine, můžeme narazit na některé nepříjemnosti. Především nemůžeme počítat s~jakýmkoliv rozšířením nad možnosti OpenGL ES. Je plně v~kompetenci programátora aby si připravil potřebné datové struktury a pomocné třídy. Systém jako takový sice obsahuje několik základních tříd pro popis vektorů, matic a operace s~nimi, ty ale nebyly navrženy pro práci s~OpenGL a dochází zde k~nesrovnalostem (příkladem může být vnitřní reprezentace matice, která je jednou řádková a jindy sloupcová). Na většinu těchto problémů poukazuje autor knihy o~vývoji her pro Android \cite{libgdxKniha}, který je současně autorem knihovny Libgdx, která tyto nedostatky napravuje.
Jako jedna z~výhod aplikací v~jazyce Java bývá uváděna automatická správa paměti použitím tzv. \textit{garbage collectoru}, tedy součásti virtuálního stroje, která počítá reference na každý alokovaný objekt a v~případě odstranění posledního výskytu tento objekt uvolní z~paměti. Ačkoliv může být tento přístup pohodlný, přináší množství rizik. Vzhledem ke složitosti operačního systému (a často i neznalostí jazyka Java) je pro programátora relativně snadné držet referenci na objekt, který již nepoužívá a o~kterém předpokládá, že již byl z~paměti uvolněn. Tím sice nedochází k~nenávratně ztraceným blokům paměti jako při práci v~nativním kódu, paměťové nároky aplikace však zbytečně rostou. Čím více paměti pak aplikace používá, tím častěji se bude garbage collector obracet právě na ni a její paměť se bude snažit uvolňovat. A~právě při úklidu paměti dochází ke krátkému pozastavení všech vláken dané aplikace. Toto pozastavení běžně trvá jednotky až desítky milisekund a nemusí být pro uživatele viditelné. Zda se ale aplikace pozastaví a jak dlouho tento stav potrvá závisí na velkém množství faktorů které programátor nemá pod kontrolou. Je tedy velmi důležité zvolit vhodnou taktiku práce s~pamětí a té se striktně držet. Jednou z~možností je předběžné vytvoření všech objektů které mohou být potřeba s~tím, že v~průběhu hry již nové objekty nevznikají. Toto je však ideální stav, kterého není snadné dosáhnout. Sofistikovanější správa paměti bude popsána v~kapitole \ref{sec:libgdx}.
S~problematikou paměti souvisí také načítání dat potřebných ve hře, zejména pak geometrie scény a textur. V~případě 3D modelů opět narazíme na neexistenci jakýchkoliv pomocných tříd. Ve vývojářské komunitě navíc není úplná shoda na tom, který z~formátů uložení modelů používat. Ve výsledku je opět na programátorovi, aby zvolil vhodný formát, ten nastudoval a implementoval jeho načítání. To vzhledem ke správě paměti popsané v~předchozí části rozhodně není triviální úkol. (U~mnohých formátů existují více či méně zdařilé implementace jejich načítání, téměř vždy ale diktují v~jakých strukturách budou načtená data uložena, což nemusí být optimální.)
V~případě textur je situace o~něco jednodušší. Jelikož Android běžně využívá rastrová data obvyklých formátů ve svém uživatelském rozhraní, umí takové soubory přirozeně i načítat. Nedostatkem zde mohou být zvýšené paměťové nároky ve chvíli kdy se textura dekóduje ze svého formátu do interní reprezentace (velmi podobné bitmapě) a následně přesouvá do grafické paměti. V~tento okamžik jsou současně alokovány dva až tři objekty reprezentující stejnou texturu, ovšem s~rozdílnou vnitřní reprezentací. S~velkou pravděpodobností bude opakovaně docházet ke spuštění automatického úklidu paměti. V~krajních případech při použití rozměrných textur může dojít i k~ukončení aplikace z~důvodu překročení maximálního množství přidělené paměti.
Zmíněné slabiny nízkoúrovňového přístupu k~tvorbě her jsou jen jedny z~mnoha. Zcela jistě záleží na subjektivním přístupu a zkušenostech, zda se můžou rozvinout do skutečných problémů či nikoliv. Drtivá většina těchto i jiných nedostatků je samozřejmě řešitelná a vývojová prostředí nabízejí množství pomocných nástrojů. V~extrémním případě může programátor zajít až tak daleko, že vytvoří vlastní framework či engine. Je však otázkou, zda by měl své síly soustředit na ladění a obcházení záludností systému nebo použít existující řešení a své síly zaměřit na konkrétní prvky, které odliší jeho hru od konkurence.
\vspace{\baselineskip}
Z~výše uvedeného je zřejmé, že u~většiny grafických aplikací je žádoucí použít existující a ověřený kód na kterém se bude dále stavět. Není podstatné zda půjde o~malou knihovnu, framework či herní engine -- názvosloví zde není ustálené (v~dalším textu se proto budu držet obecného pojmu \textit{knihovna}). Důležitý je přínos pro programátora, úspora času a prostředků, možnost soustředit se na konkrétní produkt a ignorovat podružné problémy. V~neposlední řadě také použití knihovny znamená přenesení odpovědnosti za část aplikace na někoho jiného. To může znamenat zefektivnění vývojového procesu, ale také jeho zásadní zpomalení v~případě že se vývoj knihovny zastaví nebo obsahuje chyby které autor nechce či nemůže opravit.
\section{Komplexní vývojová prostředí}
Jistou ochranou před hrozbou budoucích problémů je využití komerčně nabízených řešení. Dva nejrozšířenější zástupci této kategorie jsou Unity 3D a Unreal Engine, kde oba sami sebe označují za herní engine. V~obou případech jde o~specializovaná vývojová prostředí která se snaží upozadit programování ve smyslu psaní kódu a naopak upřednostňují přímé sestavování scény a definice chování. Vývojář může hru hrát a souběžně v~reálném čase upravovat. K~tomu slouží interaktivní pohled do scény, kde je možné pracovat na úrovni objektů. Samozřejmostí je množství předdefinovaných chování, fyzikálních vlastností, materiálů a často i celých objektů se všemi vlastnostmi (například celé pojízdné auto). Na skutečné programování přichází čas pouze ve chvíli, kdy je potřeba nadefinovat vnitřní logiku hry, případně vytvořit nové chování některého z~objektů. V~takovém případě používá engine vlastní skriptovací jazyk. Jak Unity 3D, tak i Unreal Engine jsou komplexní vývojová prostředí produkující multiplatformní aplikace a Android je pouze jedním z~mnoha výstupů. Vývojář ve skutečnosti přichází s~architekturou Androidu do kontaktu jen minimálně nebo vůbec. I~přesto není vhodné tento přístup opomíjet, neboť s~použitím právě Unity 3D vznikají v~současnosti nejlépe hodnocené a po vizuální stránce nejpropracovanější hry. Záporem je zde vysoká počáteční investice, a to v~řádu minimálně stovek dolarů.
\begin{figure}[ht]
\begin{center}
\vspace{\baselineskip}
\includegraphics[width=\textwidth]{fig/shadowgun.jpg}
\caption{Hra Shadowgun od brněnského studia Madfinger Games, vytvořená v~prostředí Unity 3D, zdroj: www.madfingergames.com}
\label{fig:shadowgun}
\end{center}
\end{figure}
\section{Knihovna Libgdx}
\label{sec:libgdx}
Alternativou k~placeným vývojovým nástrojům jsou menší knihovny s~těsnější vazbou na systém Android. Nejpoužívanější z~nich jsou AndEngine a Libgdx. První z~jmenovaných se nazývá herním engine, druhý se označuje za framework. Oba jsou však k~dispozici zdarma ve formě otevřených zdrojových kódů s~možností komerčního i nekomerčního využití. Jelikož jsou si obě knihovny velmi podobné, soustředím se v~dalším popisu pouze na Libgdx, kterou také použiji při implementaci. Zdrojem bude z~velké části oficiální dokumentace \cite{libgdx} a také již zmiňovaná kniha o~vývoji her pro Android \cite{libgdxKniha}.
Na první pohled vypadá knihovna Libgdx podobně jako dříve zmíněné nástroje -- stejně jako ony je schopná produkovat aplikační balíčky pro více platforem. Původně však vznikala jako knihovna pro tvorbu 2D her pro Androida a až postupem času došlo rozšíření o~podporu dalších platforem. Aktuálně je kromě systému Android podporovaná i tvorba webových appletů a klasických aplikací pro počítače. Jak konkrétně multiplatformní vývoj probíhá bude popsáno v~kapitole \ref{sec:oddeleni-platforem}.
Narozdíl od jiných řešení knihovna neobsahuje žádné vývojové prostředí ani předpřipravená data. Zaměřuje se více na usnadnění vývoje nabízením pomocných algoritmů a tříd. Je pak čistě na programátorovi, zda výhod knihovny využije. Díky tomu má prostor pro detailní optimalizace na nízké úrovni pro dosažení maximálního výkonu hry, stejně jako se může pohybovat na vyšší úrovni, která umožní velmi rychlé prototypování a vytvoření funkční aplikace. Mezi přednosti Libgdx patří vysoko- i nízkoúrovňové rozhraní pro 2D a 3D grafiku, podpora fyziky, propracovaný systém uživatelského rozhraní, načítání množství formátů, práce se zvuky a mnoho dalších. V~neposlední řadě se pak snaží stírat rozdíly mezi jednotlivými zařízeními a verzemi systému Android.
Samozřejmostí je řešení výše zmiňovaných nedostatků při vývoji bez pomocných knihoven. Velká síla Libgdx spočívá v~tom, že je implementována částečně přímo v~Javě a částečně v~nativním kódu. Tím dosahuje vyššího výkonu zejména u~již zmiňovaných fyzikálních simulací, ale především má plnou kontrolu nad alokací paměti a jejím následným uvolňováním. Automatické uvolnění přirozeně probíhá pouze v~části aplikace bežící ve virtuálním stroji, zatímco v~nativním kódu je práce s~pamětí plně v~režii programátora. Vhodným propojením těchto dvou částí lze bezpečně alokovat paměť kterou garbage collector \textit{neuvidí}, a dokonce dosáhnout stavu kdy framework jako takový uvnitř hlavní herní smyčky žádnou paměť nealokuje. To samozřejmě nemůže úplně zamezit spouštění úklidu paměti, příčinou však bude běh ostatních aplikací a systému, nikoliv samotné hry. V~praxi se pak paměť uklízí jen velmi výjimečně, probíhá velmi rychle (garbage collector spravuje jen velmi malé množství paměti), a hra tedy může běžet plynule.
\subsection{Struktura aplikace}
Vstupním bodem aplikace je instance třídy \texttt{Application}, která obsahuje základní herní smyčku. Ta je volaná systémem právě na dříve zmiňované frekvenci blízké 60 snímkům za sekundu. Při tvorbě grafických aplikací se často využívá oddělené volání pro aktualizaci scény a její vykreslení, Android však takové rozdělení nepodporuje (z~důvodu návaznosti na systémový element s~OpenGL kontextem, popsaný v~kapitole \ref{sec:nativni-vyvoj}. Je samozřejmě možné aktualizaci scény a její vykreslení oddělit ručně, například v~případě že není nutné se scénou manipulovat příliš často, ale chceme zachovat plynulost vykreslování. Typické využití může být u~tahových her, kdy se herní logika provádí jen v~okamžiku tahu, zatímco zobrazování probíhá v~každém snímku, například kvůli probíhajícím animacím. Je samozřejmě možné optimalizovat ještě více, animace nepoužívat a vykreslovat jen ve chvíli, kdy se scéna změní. V~takovém případě dochází k~maximální úspoře energie. Aplikace ale může působit značně statickým dojmem a pro akční hry tento přístup není vhodný vůbec.
Zatímco samostatná třída aplikace je vhodná jen pro nejtriviálnější aplikace, u~složitějších bude žádoucí rozdělit aplikaci do několika částí. Zde knihovna nabízí členění do tzv. \textit{obrazovek} (potomci třídy \texttt{Screen}), které mají vlastní životní cyklus. Hlavní třída aplikace pak obstarává pouze přepínání mezi jednotlivými obrazovkami. Ty lze považovat za oddělené logické celky a často je možné je i samostatně vyvíjet, což bude přínosem zejména pro vícečlenný tým vývojářů. Obrazovka jako taková však žádné vyšší funkce nemá a kromě zmíněného zpřehlednění kódu nabízí jen možnost jednoduchého vykreslování texturovaných obdélníků (tzv. \textit{sprite}).
\subsection{Zpracování vstupních událostí}
\label{sec:libgdx-vstup}
Vstupní události jsou závislé na aktuální platformě, a proto jsou v~Libgdx skryty pod jednotným rozhraním, kdy konkrétní implementaci doplňuje sama knihovna právě podle aktuálního prostředí. Programátor díky tomu nemusí řešit, zda bude aplikace ovládaná dotykem či myší, klávesnicí nebo hardwarovými tlačítky mobilního telefonu nebo třeba ovladačem typickým pro herní konzole. Je zřejmé, že všechny typy událostí nebudou podporovány na všech platformách. Knihovna však zavádí vhodné substituce a sjednocení, kde například kliknutí na dotykovém displeji odpovídá kliknutí levého tlačítka myši. Na vyšší úrovni je pak možné detekovat celá gesta jako chycení a táhnutí nebo přiblížení použitím dvou prstů.
Přirozeně lze definovat i vlastní zpracování vstupu. Příkladem může být křížový ovladač vykreslovaný na obrazovku v~případě, že aplikace běží na dotykovém zařízení. Vnitřně pak může takový ovladač \textit{překládat} dotykové události na události stisku klávesy. Ty pak aplikace obslouží jako běžné události a programátor tak velmi snadno získává podporu pro dotyková zařízení, počítačové klávesnice i konzolové ovladače.
Toto chování je umožněno tzv. \textit{vstupními procesory} (třída \texttt{InputProcessor}). Vstupní procesor je objekt obsahující kód pro zpracování vstupních událostí. V~okamžiku zpracování v~procesoru už je známo o~jaký typ události jde. Procesor jako takový tedy žádnou detekci neprovádí. Typicky obsahuje referenci na objekty aplikační logiky a s~těmi nějak pracuje. Na konci svého běhu vrací návratovou hodnotu podle toho, zda událost zpracoval (a měla by tedy zaniknout) anebo ne (což nutně neznamená že na událost nijak nereagoval).
Vstupních procesorů může být v~aplikaci několik, aktivní je ale vždy pouze jeden. Pro komplexní zpracování vstupů se proto zavádí tzv. \textit{multiplexer} (třída \texttt{InputMultiplexer}), který představuje kontejner vstupních procesorů, přičemž rozhraní procesoru také sám implementuje. Při příchodu události ji pouze předává jednotlivým procesorům do té doby, dokud ji některý z~nich nezpracuje.
Tím je možné vytvářet kaskády vstupních procesorů ať už pro zmíněné sjednocení vstupů napříč platformami anebo pro obsluhu událostí ze složitějších scén. Příkladem může být dotykem ovládaná hra využívající doplňkové ovládací prvky na obrazovce. Vhodným řešením by mohl být multiplexer zastřešující vstupní procesor pro doplňkové prvky (přicházející na řadu jako první, tedy může událost zpracovat a zahodit) a procesor pro ovládání samotné hry. Takto oddělené zpracování vede nejen k~přehlednějšímu kódu, ale i k~jednodušší obsluze událostí ve chvílích, kdy existuje více prvků souběžně či postupně reagujících na stejný vsutp.
\subsection{Tvorba uživatelského rozhraní}
\label{sec:libgdx-ui}
Velmi silnou součástí Libgdx z~hlediska tvorby uživatelského rozhraní je knihovna Scene2D.ui \cite{scene2dui}, která je použitelná i samostatně v~obecných projektech v~jazyce Java. Jejím základním prvkem je tabulka (třída \texttt{Table}), pro kterou se definují jednotlivé buňky a jejich rozdělení do řádků. Chování je podobné tabulkám vytvořeným v~jazyce HTML v~kombinaci s~CSS. Buňkám lze nastavovat pevné či poměrné rozměry, vnitřní okraje, pozadí a některé další atributy. Je také možné buňky v~rámci jednoho řádku slučovat; vertikální slučování podporované není, ale lze jej dosáhnout jinou cestou. Specifickým parametrem buněk je pak \textit{rozpínavost} v~rámci řádku a v~rámci sloupce. Vhodnou kombinací je možné vytvořit dynamickou, tzv. \textit{fluidní} tabulku, která se bude přizpůsobovat svému obsahu i rozměrům zobrazovací plochy.
Nad tabulkami jsou dále vystavěny základní prvky uživatelského rozhraní jako tlačítka, dialogy, pole pro zadání textu, obrázky, bloky textu, posouvatelné panely a další. Využití tabulky lze demonstrovat na tlačítku. To je ve skutečnosti tabulka o~třech řádcích a třech sloupcích, kde prostřední buňka obsahuje textový popisek čí obrázek. Všechny jeho buňky pak mají vhodně nastavené pozadí, čímž lze vytvořit například i tlačítka se zakulacenými okraji. Samozřejmě si opět zachovává schopnost přizpůsobit rozměry svému obsahu i okolí. Složitější prvky vznikají obdobně, přičemž často využijí i možnost tabulky do sebe libovolně vnořovat.
Prvky rozhraní také nemusejí být nutně statické. Každý prvek může mít nadefinováno své vlastní chování a v~rámci scény se chovat zcela autonomně. Současně knihovna nabízí množství předdefinovaných efektů, které lze prvkům přiřazovat. Typicky jde o~geometrické transformace nebo práce s~barvami či průhledností. Efekty dostávají jako parametr dobu trvání a krajní hodnoty, mezi kterými v~čase interpolují. Podobně jako při zpracování událostí i zde existují speciální efekty chovající se jako kontejnery efektů běžných. Ty umožňují nechat efekty probíhat současně anebo v~řadě za sebou. Samozřejmostí je opět libovolné vnořování, čímž lze vytvořit velmi komplexní chování.
\vspace{\baselineskip}
Kořenovou komponentou uživatelského rozhraní je tzv. \textit{Stage} (jelikož by český výraz mohl být zavádějící, ponechávám bez překladu a použiji přímo název třídy \texttt{Stage}). Představuje kontejner na výše popsané prvky rozhraní. Ty sice své vykreslování řídí samy, ale předpokládají korektně nastavené parametry vykreslování, především pohledových transformací. K~tomu slouží Stage, která udržuje kameru s~ortogonální projekcí a některé další struktury sloužící k~vnitřní optimalizaci vykreslování. Implicitně Stage pokrývá celou zobrazovací plochu, ale toto chování je možné změnit. Stará se také o~pořadí vykreslování prvků rozhraní, čímž řídí jak se budou prvky překrývat. Na samotnou Stage můžeme také, podobně jako na jednotlivé její prvky, aplikovat některé efekty. Stejně tak je možné vložit do jedné obrazovky několik instancí Stage a provádět navigaci mezi nimi, v~rámci jediné obrazovky.
\begin{figure}[ht]
\begin{center}
\includegraphics[width=\textwidth]{fig/ui-design.jpg}
\caption{Součásti rozhraní hlavního menu hry. Červeně jsou orámovány hlavní komponenty, modře kontejnery a zelenou barvou je vyznačeno odsazení vnořené komponenty od okraje svého rodiče.}
\label{fig:ui-design}
\end{center}
\end{figure}
\subsection{Pomocné nástroje a součásti knihovny}
\label{sec:libgdx-tools}
Knihovna Libgdx obsahuje několik velmi užitečných nástrojů. Některé z~nich jsou přímo součástí kódu knihovny a jsou použitelné při běhu aplikace, jiné jsou distribuovány jako samostatné programy vhodné pro předpřipravení dat do vhodného formátu. Jejich použití či nepoužití je vždy rozhodnutí programátora a knihovna jako taková použití žádné z~níže popsaných součástí nevyžaduje.
\vspace{\baselineskip}
První zajímavý nástroj se jmenuje \textit{Texture Packer}. Slouží k~vytváření tzv. \textit{atlasů textur}, což je jediný obrázkový soubor, který v~sobě obsahuje vhodně seskládané jednotlivé textury. Jde obecně používaný přístup, který značně urychluje načítání textur i jejich vykreslování zejména v~případech, kdy aplikace využívá množství malých textur. Z~pohledu načítání se ušetří přístupy do souborového systému, v~okamžiku vykreslování je pak velká textura uložená v~jediné texturovací jednotce. Pokud by se atlas textur nepoužil bylo by nutné v~průběhu vykreslování jediného snímku texturovací jednotky nejen přepínat, ale vzhledem k~jejich omezenému počtu i měnit k~nim připojené textury. Texture Packer současně s~grafickým výstupem ukládá i soubor ve formátu JSON, který knihovna při použití atlasu automaticky načte a podle informací v~něm obsažených vytvoří pojmenované texturové oblasti (odpovídající dílčím texturám). S~těmi je následně možné v~aplikaci pracovat jako s~běžnými texturami, zatímco popsané optimalizace probíhají automaticky bez přičinění programátora. Užitečnou vlastností nástroje Texture Packer je také možnost spouštění z~příkazové řádky, respektive jeho přímého zapojení do procesu sestavování aplikace. Programátor tedy udržuje adresář s~jednotlivými texturami (jelikož je tento způsob mnohem výhodnější pro případné úpravy), z~toho se při sestavení aplikace vytvoří atlas, knihovna jej načte a programátor v~aplikaci dostává přísutp opět k~jednotlivým texturám.
\begin{figure}[ht]
\begin{center}
\includegraphics[width=\textwidth]{fig/ui-atlas.jpg}
\caption{Rozhraní nástroje Texture Packer, který z~dílčích textur vytváří tzv. \textit{atlas}, uložený v~jediném souboru. V~pracovní oblasti programu na pravé straně je atlas textur použitý ve hře.}
\label{fig:ui-atlas}
\end{center}
\end{figure}
\vspace{\baselineskip}
Dalším užitečnou součástí knihovny je podpora bitmapových písem. K~používání písem a vypisování textů není v~Open GL jednotný přísup. Bitmapová písma se však jeví jako rozumný kompromis mezi vektorově definovanými písmy, které je před vykreslením potřeba rasterizovat, a bitmapami s~pevně uloženými řetězci a texty. Bitmapové písmo se skládá z~obrázkového souboru, kde jsou jednotlivé znaky a symboly uloženy podobně jako textury v~atlasu textur, a definičního souboru ve formátu FNT. Tento soubor však kromě informací pro rozsekání obrázku na jednotlivé znaky obsahuje informace o~tom, jak znaky skládat do slov, o~rozestupech mezi nimi, o~názvu a charakteristikách písma a další. K~vytvoření těchto souborů lze využít některý z~volně dostupných nástrojů (bitmapová písma jsou opět obecně používanou technikou). Knihovna Libgdx tyto soubory načítá a může s~jejich použitím vykreslovat jednořádkový či víceřádkový dynamický text. Taková funkcionalita je velmi výhodná při překladu aplikace do jiných jazyků. V~takovém případě není potřeba udržovat popisky v~grafických souborech (čímž rapidně vzrůstá velikost celé aplikace, ale i nároky na její údržbu a další vývoj), ale je možné použít běžný textový soubor s~řetězci. Přidání podpory dalšího jazyka pak znamená pouhé přeložení sady řetězců, vše ostatní proběhne automaticky, včetně například správného zarování popisků v~uživatelském rozhraní, které plyne z~chování prvků rozhraní popisovaných v~kapitole \ref{sec:libgdx-ui}.
\vspace{\baselineskip}
Specialitou knihovny Libgdx je použití vizuálních témat pro celou aplikaci (základem je třída \texttt{Skin}). Při návrhu uživatelského rozhraní zavádí vrstvu abstrakce, kde každému typu prvku rozhraní přiřazuje definici vizuálního stylu. Definuje se tedy obecný vzhled tlačítek, polí, posuvníků a dalších prvků. Je také možné zvolit ještě jemnější dělení v~rámci typu, například zavést několik skupin tlačítek a jejich vzhled definovat odděleně. Pro popis vizuálních témat slouží soubor ve formátu JSON obsahující konkrétní názvy tříd prvků rozhraní a poté pro každý z~nich sadu atributů jako je barva, pozadí, použité písmo a další. Právě zde lze vhodně využít atlasy textur a bitmapová písma popsaná v~předchozích odstavcích. Po načtení vizuálních stylů do knihovny jsou automaticky aplikovány. Knihovna pozná že je v~rozhraní použitý prvek, jehož třída má přiřazenou definici, a tuto použije. Změnou definičního souboru lze velmi snadno změnit celý vzhled aplikace. Samozřejmě je možné s~aplikací dodávat i několik vizuálních témat a výběr ponechat na uživateli; z~hlediska programátora je toto jen úprava jediného řádku, kde se definiční soubor načítá.
\vspace{\baselineskip}
Poslední popisovanou součástí knihovny je \textit{Assets Manager}, který je možné volně přeložit jako \textit{správce obsahu}. Přestože je jeho použití volitelné, je velmi silně doporučováno. Správce obsahu totiž představuje jednotné místo v~rámci aplikace, přes které se bude přistupovat k~datovým souborům. Těmi můžou být textury, zvuky, různé definiční soubory, a teoreticky jakýkoliv obsah. Správce dostane seznam datových souborů které jsou vyžadovány pro běh aplikace a tyto soubory bude načítat. Jeho rozhraní je vhodně navržené tak, aby bylo použitelné pro vytvoření načítací obrazovky s~možností zobrazení reálného průběhu načítání. Zajímavý je také způsob, jakým ze souborů vznikají datové objekty. Aplikace definuje soubor a cílovou třídu, jejíž instanci chce získat. Pro každou takovou třídu pak musí existovat speciální načítací třída (tzv. \textit{loader}), která umí ze souboru konkrétní objekt vytvořit. Pro všechny základní formáty už knihovna tyto třídy nabízí, stejně jako možnost načítání parametrizovat. To se využije například pro volbu způsobu filtrování textur. Spolu s~načítáním také správce hlídá, zda jsou data ještě používána a může je průběžně uvolňovat.
\section{Knihovna Bullet Physics}
\label{sec:bullet}
Nezbytnou součástí zejména leteckých her bude jistá forma fyzikální simulace. K~tomuto účelu popíši volně dostupnou knihovnu Bullet Physics. Ta může fungovat v~několika režimech, přičemž dva nejvýznamnější jsou režimy tzv. \textit{kolizního světa} (třída \texttt{btCollisionWorld}) a \textit{simulačního světa} (třída \texttt{btDiscreteDynamicsWorld}). Základní principy jednoduché 2D a 3D simulace popisuje již zmiňovaná kniha o~vývoji her \cite{libgdxKniha}, informace specifické pro knihovnu Bullet Physics pak vycházejí z~její oficiální webové stránky \cite{bullet} (manuál ve formátu PDF je součástí stažitelného vývojářského balíčku).
Jak již název napovídá, kolizní svět obsahuje entity reprezentované svými kolizními tvary a signalizuje, pokud dojde k~jejich kontaktu. Na tento kontakt lze následně reagovat v~rámci herní logiky. Entity se ale dále mohou pohybovat a v~kolizi pokračovat. Knihovna zde slouží čistě k~detekci kolizí a nikoliv k~jejich další obsluze. Detekce probíhá ve dvou fázích -- tzv. \textit{široké} a \textit{úzké}. Při široké fázi jsou namísto skutečných obalových těles porovnávány kolize osově zarovnaných obalových kvádrů (metoda \textit{AABB - Axis Aligned Bounding Boxes}). Tímto krokem je možné rychle odstranit velké množství dvojic objektů, které spolu nemohou v~žádném případě kolidovat. V~úzké fázi se následně hledá kolize mezi skutečnými tvary entit. Tento krok je značně náročnější, neboť obalové tvary můžou být jednoduché (základní tvary jako koule, kvádr, plocha), ale také složené z~obecně libovolného množství základních a složených tvarů. Se složitostí tvaru přirozeně roste i náročnost ověření kolize. Z~tohoto důvodu je zavedeno rozdělení do dvou fází, které toto složité porovnání provede jen v~případech, kde může kolize reálně nastat.
Nad kolizním světem je vystavěný svět simulační. V~tomto světě obsahují entity více informací. Kromě kolizního tvaru je nutná znalost hmotnosti, rychlosti a zrychlení v~čase a volitelně dalších vlastností jako například tvrdost či pružnost povrchu. Tyto entity jsou pak součástí plnohodnotné fyzikální simulace, která sama řídí jejich pohyb a případné kolize. V~praxi funguje zapojení simulace do hry nejčastěji tak, že herní logika dodává impulsy (například sílu vzniklou vysunutím klapky letadla) a o~zbytek se stará právě fyzikální knihovna.
Provádění úplné fyzikální simulace je ve srovnání s~pouhou detekcí kolizí řádově složitější. Dokonce natolik, že se v~současných hrách pro mobilní zařízení používá jen minimálně, případně jen pro velmi malé množství entit. Takto lze dosáhnout dobrého dojmu ze hry, kdy hráč vnímá hru jako reálnou, přestože fyzikální simulace probíhá jen u~hlavního objektu a chování ostatních objektů může být pevně přednastaveno. I~v~případě simulace jediného objektu je vždy nutné co nejpřesněji fyzikálně popsat působící síly i objekt samotný. Současně není možné do této simulace zasahovat a zavádět vlastnosti, které by realitě neodpovídaly. Ze zmíněných důvodů plyne, že pro tuto práci bude dále relevantní pouze svět kolizní.
\section{Distribuce aplikací}
\label{sec:distribuce}
Aplikace pro systém Android lze distribuovat několika různými cestami. Tou nejzákladnější je přímá distribuce binární podoby aplikace (v~souboru s~příponou APK). Tato volba však znemnožní jakoukoliv další automatickou aktualizaci (manuální aktualizaci lze stále provést získáním nové verze souboru) a současně nedovoluje mít kontrolu nad tím, kteří uživatelé, na jakých zařízeních a za jakých podmínek, budou aplikaci používat. Navzdory tomu je tato metoda používána, například pri testování prototypu aplikace a jeho kompatibility.
Důmyslnějším způsobem distribuce je využití některého z~internetových obchodů. Těch existuje hned několik s~různě velkou uživatelskou základnou a s~různými podmínkami publikování. Dominantní postavení zde má obchod Google Play, představující oficiální katalog aplikací platformy Android. Jeho hlavní předností je, že je přednastaven v~drtivé většině prodávaných zařízení a má tak možnost oslovit maximální množství potenciálních uživatelů. Vývojáři současně nabídne možnost aplikaci průběžně vzdáleně aktualizovat, řídit která zařízení bude aplikace podporovat, sledovat statistiky instalací a také sbírat zpětnou vazbu od uživatelů. Aplikaci zde lze zveřejňovat bezplatně, ale i za poplatek, přičemž si provozovatel obchodu nárokuje 30\% podíl z~provedené transakce. Zmíněné přednosti jsou však vykoupeny jednorázovým poplatkem za registraci vývojáře, který v~době psaní této práce činí \$25. Dalším nedostatkem obchodu Google Play je nemožnost rozdělit konkrétní uživatele do skupin a těm následně nabídnout jinou verzi aplikace nebo například možnost bezplatného testování placené varianty. Stejně tak by mnoho vývojářů ocenilo možnost vygenerování speciálních poukazů, kterými by mohli zdarma zpřístupňovat propagační verze svých aplikací.
\chapter{Ovládání skutečného a herního letadla}
Tato kapitola popíše principy ovládání letadel a leteckých her. Shrne existující metody a představí novou metodu zaměřenou na ovládání akčních leteckých her v~mobilních zařízeních.
Před návrhem herního ovládání letadla je vhodné se seznámit s~tím, jak létají skutečná letadla. Více než o~konkrétní rovnice zde půjde o~základní principy díky kterým se letadlo udržuje ve vzduchu. Rovnice by totiž nebylo možné vyčíslit bez znalosti konkrétních koeficientů, které jsou pro každou konstrukci odlišné. I~s~nimi by však bylo exaktní matematické popsání letícího letadla velmi obtížné. Následující kapitoly budou čerpat především z~knihy Umění létat \cite{umeniLetat} a studijních materiálů americké vládní agentury pro letectví (NASA) \cite{nasa}.
\section{Síly působící za letu}
Základní pojem pro další popis bude takzvaný \textit{ustálený vodorovný let}. V~tuto chvíli se letadlo pohybuje konstantní rychlostí s~nulovým zrychlením, nestoupá, neklesá a není v~náklonu. Působí na něj síly vyznačené na obrázku \ref{fig:letadlo-sily}. Jsou jimi vzlak, gravitační tíha, tah a odpor. Při ustáleném letu jsou tyto síly v~rovnováze. Ne však všechny navzájem, jak je někdy mylně uváděno, ale pouze dvojice sil působících proti sobě. Pro ilustraci můžeme použít namísto celého letadla pouze křídlový profil, na kterém jsou veličiny více zřejmé.
\begin{figure}[ht]
\begin{center}
\includegraphics[scale=0.25]{fig/airfoil.pdf}
\caption{Síly působící na letadlo při ustáleném vodorovném letu}
\label{fig:letadlo-sily}
\end{center}
\end{figure}
Nejdůležitější silou je vztlak. Ten působí kolmo vzhůru od křídla a při vodorovném letu vyrovnává tíhu letadla. Dochází k~němu díky typickému profilu křídla, kdy na jeho spodní straně vzniká vyšší tlak vzduchu a současně na horní se tlak vzduchu snižuje. Pro zjištění celkového vztlaku křídla je potřeba znát množství parametrů, jako je hustota prostředí či rychlost pohybu křídla. Především ale záleží na koeficientech popisujících tvar křídla, jeho plochu, rozpětí a tvar jeho profilu. Pro zjištění celkového vztlak letadla, bylo by samozřejmě nutné započítat dílčí vztlakové síly na celém jeho povrchu.
Další z~hlavních sil je tah motoru. Ten je tvořen jednou či více vrtulemi. Lopatky vrtule svým tvarem ženou vzduch za sebe (tedy směrem na trup a křídla letadla), čímž vzniká síla táhnoucí vrtuli, a tím i celé letadlo, dopředu. Tažná síla opět závisí na množství parametrů které popisují stavbu vrtule, počet lopatek a jejich aerodynamické vlastnosti, a samozřejmě také na rychlosti jakou se vrtule otáčí. S~vyšší rychlostí stoupá tah a současně se zvyšuje rychlost proudění vzduchu v~oblasti za vrtulí. Tím se nepřímo zvyšuje vztlak křídel, ale také se mění citlivost ovládacích prvků letadla.
Gravitační síla působící na letadlo je zřejmá a vždy směřuje ke středu Země.
Poslední silou je odpor který klade vzduch letícímu letadlu a působí vždy v~opačném směru než je směr jeho pohybu. Je složený ze dvou hlavních složek. Jednou je odpor způsobený tvarem křídla, potažmo celého letadla, a jeho pohybem v~prostředí. Druhou složkou je odpor vznikající na koncích křídel. Jak již bylo zmíněno, u~horní části křídla se nachází podtlak, zatímco u~spodní části je naopak přetlak. V~místě kde je křídlo spojeno s~trupem letadla se tyto tlaky \textit{nepotkávají}, na opačném konci křídla však ano. Dochází k~jejich přirozenému vyrovnávání a vznikají víry, které pokrucují proudění vzduchu a mění směr vztlakové síly křídla. Její pomyslný vektor pak není kolmý ke křídlu, ale mírně sklopený, čímž přispívá k~velikosti odporové síly. Čím vyšší vztlak křídlo tvoří, tím více odporové síly současně vzniká. Obecně je pro zjištění odporové síly důležitá rychlost pohybu a charakteristika prostředí, především pak samotný tvar křídel a celého letadla.
Výsledné chování letadla záleží vždy na poměru zmíněných sil. Pokud by křídla netvořila dostatečný vztlak (z~důvodu konstrukce nebo nízké rychlosti pohybu), letadlo se nevznese, případně spadne. Je však nutné si uvědomit, že s~výjimkou tahu motoru nemá pilot letadla možnost žádné z~uvedených parametrů změnit, neboť jsou dány konstrukcí letadla.
\section{Ovládací prvky a manévry}
\label{sec:let-skutecneho-letadla}
Pro popis jiného než vodorovného letu je potřeba zavést nový pojem, a sice \textit{úhel náběhu}. Jde o~úhel, který svírá tětiva profilu křídla s~proudem vzduchu, který na ni dopadá. Úhel náběhu je vyznačený na obrázku \ref{fig:uhel-nabehu}.
\vspace{\baselineskip}
\begin{figure}[ht]
\begin{center}
\includegraphics[scale=0.4]{fig/angle-of-attack.pdf}
\caption{Úhel náběhu křídla. Jeho výchozí hodnota je dána konstrukčními vlastnostmi letadla a křídel. Za letu jej lze pak regulovat vysouváním či zasouváním klapek a samozřejmě změnou orientace celého letadla vůči směru proudění vzduchu.}
\label{fig:uhel-nabehu}
\end{center}
\end{figure}
Navzdory časté představě ve skutečnosti neplatí, že by směr letu letadla byl vždy totožný s~osou jdoucí jeho trupem. Při dostatečné rychlosti může letadlo letět vodorovným letem, přičemž trup bude skutečně souběžný s~dráhou letu. Vztlak křídla bude v~tomto případě dostatečný právě díky vysoké rychlosti proudění. Pokud je ale cílem nižší letová rychlost (například z~důvodu úspory paliva), bude pro zachování vodrovného směru nutné aby pilot zvýšil úhel náběhu křídel (použije k~tomu výškové kormidlo -- viz dále). Tím zvýší i jejich koeficient vztlaku a zamezí klesání letadla. V~této chvíli se letadlo pohybuje vodorovně, ale jeho příď je výš než záď a míří tedy mírně nahoru (úhel o~který je letadlo nakloněno odpovídá úhlu náběhu). Přibližný vztah mezi úhlem náběhu a vztlakovým koeficientem křídla je znázorněn na obrázku \ref{fig:vztlak}. Přesný tvar průběhu je opět dán konstrukčními vlastnostmi konkrétního křídla.
\begin{figure}[ht]
\begin{center}
\includegraphics[scale=0.5]{fig/lift-coefficient.pdf}
\caption{Závislosti vztlakového koeficientu na úhlu náběhu u~neidentifikovaného křídla. Při úhlu vyšším než $25^{\circ}$ křídlo přestává generovat vztlak a letadlo se začíná propadat. Zdroj: Wikimedia Commons na základě dat z~programu XFOIL}
\label{fig:vztlak}
\end{center}
\end{figure}
Nejvyšší hodnota vztlakového koeficientu je v~případě křídla na obrázku vidět v~oblasti kolem $17^{\circ}$ (opět je nutné připomenout že tento průběh je jen jedním z~mnoha možných a například sportovní letadla mohou efektivně využívat i vyšších úhlů náběhu). Při dalším zvyšování úhlu náběhu se mění proudění kolem horní části křídla z~\textit{laminárního} (proudnice jsou rovnoběžné a na pohled vypadají jako by byly rozdělné do vrstev) na proudění \textit{turbulentní} (proudnice se trhají a vzájemně prolínají, vznikají vzdušné víry). V~důsledku vznikajících turbulencí klesá vztlak křídla a současně roste jeho povrchové tření, čímž stoupá i jeho odpor. Na hranici blízké $25^{\circ}$ pak křídlo ztrácí vztlak úplně a letadlo se začne propadat. Této situaci se často říká \textit{přetažení}.
\vspace{\baselineskip}
Pilot k~řízení letadla používá takzvané \textit{primární ovládací prvky}. Jsou jimi klapky na křídlech a výškové kormidlo, obojí ovládané kniplem, a dále směrové kormidlo ovládané pomocí pedálů. Všechny tyto ovládací prvky ve skutečnosti plní jedinou činnost, a sice že mění úhly náběhu jednotlivých částí letadla. Přestože jsou křídla pevně spojená s~trupem, jsou na nich pohyblivé klapky. Stejně tak vodorovné výškové i svislé směrové kormidlo má tvar křídla, jehož část u~odtokové hrany je pohyblivá.
Pokud nyní přestaneme dosud popisované síly vztahovat k~celému letadlu, ale budeme je uvažovat jako samostatné síly vznikající vždy na konkrétním křídle (za křídla zde považujeme i výškové a směrové kormidlo), můžeme snadno popsat základní letecké manévry.
Proces stoupání či klesání pilot reguluje nastavením klapek výškového kormidla. V~případě stoupání klapky zvedne, čímž sníží úhel náběhu kormidla a tím i jeho vztlak. Záď letadla začne klesat, úhel náběhu hlavních křídel letadla se začne zvyšovat a s~ním i jejich vztlak, potažmo vztlak letadla jako celku. Při klesání je situace analogická, pouze se veličiny pohybují opačným směrem.
U~zatáčecího manévru často panuje přesvědčení, že k~jeho provedení slouží pouze směrové kormidlo, od kterého se už podle názvu očekává že bude řídit směr letu. Z~předchozích odstavců již můžeme tušit že tomu tak není. Pokud by pilot při vodorovném letu použil směrové kormidlo, záď letadla by se sice vůči směru letu natočila, samotný směr letu by se však změnil jen málo. Tím jak se osa letadla natáčí vůči směru letu současně roste proudění kolem křídla které je více vystaveno přímému směru letu. Tím na křídle vzniká větší vztlak a letadlo začíná podélně rotovat, ačkoliv je knipl stále ve výchozí poloze.
Správný průlet zatáčkou probíhá v~náklonu. Pilot nejdříve zatočením kniplu nakloní klapky na křídlech (které se narozdíl od výškového a směrového kormidla pohybují navzájem protichůdně) tak, že na jednom z~křídel se vztlak zvýší, zatímco na druhém se sníží. Letadlo tedy letí v~náklonu a vztlak vznikající na křídlech nepůsobí kolmo vzhůru, ale kolmo k~horním stranám křídel (vztlak vzniká stále stejně, pouze jeho směr již není přímo opačný ke gravitační síle). Můžeme tedy vztlakovou sílu pomyslně rozložit na skutečný vztlak držící letadlo ve vzduchu a složku vodorovnou, která působí jako dostředivá síla a nutí letadlo skutečně zatáčet. Směrové kormidlo při průletu zatáčkou slouží regulaci vzájemné polohy přídě a zádi letadla. Některá letadla mají totiž tendenci v~důsledku různě velkého odporu na křídlech (například pokud letadlo automaticky nenastavuje klapky do vzájemně opačných poloh) \textit{sklouzávat} po přídi nebo zádi do středu pomyslného kužele po jehož vnitřní stěně letadlo zatáčkou prolétává. Výškové kormidlo může zatáčecí manévr doplnit na vzestupný či sestupný.
\section{Stabilita letadla}
Každé letadlo má svou konstrukcí dané stabilizační schopnosti, které se uplatňují aniž by pilot zasahoval do řízení. Stabilitu se dělí na statickou a dynamickou.
Statická stabilita popisuje reakci letadla ve chvíli kdy dojde k~narušení jeho dosavadního směru letu například poryvem větru. U~staticky stabilního letadla vznikne opačná síla snažící se letadlo vrátit do původní polohy. Staticky neutrální letadlo bude pokračovat v~pozměněném směru a u~staticky nestabilního letadla se dokonce objeví síla působící spolu s~rušivým vlivem, která odklon od původního směru ještě více posílí.
Dynamická stabilita popisuje stabilitu letadla v~dlouhodobém horizontu. Dynamicky stabilní letadlo sice bude při vyrovnávání svého směru oscilovat kolem rovnovážné polohy (vodorovného letu), ale po čase se v~něm ustálí. Dynamicky neutrální letadlo bude oscilovat s~pořád stejnou sílou a dynamicky nestabilní letadlo bude od rovnovážné polohy dokonce divergovat.
Letadlo které je staticky i dynamicky stabilní je většinou snáze ovladatelné a vhodné pro začínající piloty. Pokud je však tendence letadla dosáhnout stabilní polohy příliš velká, může to být ke škodě, neboť bude k~ovládání letadla nutné vynaložit vyšší sílu. Ač se to nemusí zdát, není vždy cílem zkonstruovat dokonale stabilní letadlo a v~některých případech může být vhodné, pokud se letadlo chová neutrálně nebo dokonce nestabilně.
\section{Herní ovládání letadla}
V~následujících podkapitolách bude popsáno zapojení výše vysvětlených mechanik letu do leteckých her. Zaměřím se přitom především na takzvané \textit{causual} hry, tedy oddechové hry které od hráče nevyžadují vysokou úroveň zkušeností. U~her a programů profilujících se jako simulátory je zřejmá snaha co nejvíce se přiblížit reálné předloze jak u~letové dynamiky, tak i z~hlediska ovládání letadla. I~přesto lze u~simulátorů pozorovat využití některých zjednodušení zejména z~oblasti ovládání.
\subsection{Chování letadla}
V~předchozích kapitolách byla zmíněna složitost numericky přesného výpočtu všech sil působících na letadlo. Nejen že by bylo potřeba znát až desítky reálných koeficientů pro každý jeden typ letadla, ale požadavky na výpočetní výkon by byly velmi vysoké (tím spíše pokud uvažujeme mobilní zařízení). Proto se v~případě leteckých simulátorů a her používají matematické modely s~různou úrovní věrnosti fyzikální předloze \cite{simulationModels}.
Je pochopitelné, že v~případě simulátorů pro výcvik pilotů je maximální věrnost simulace nezbytná. V~případě leteckých her se však ukazuje, že příliš reálně ovládané letadlo je spíše na obtíž. Hráč causual her neumí a často ani nechce umět plnohodnotně ovládat letadlo. Jeho prvotním cílem je splnění herních úkolů, typicky sestřelení protivníka nebo let na dané místo. Letadlo je v~této kategorii her pouze prostředkem k~dosažení cíle, nikoliv cílem samotným.
Do matematických modelů se proto vkládají zjednodušující předpoklady, jako například stabilita letadla při zatáčení. To potom nemá tendence v~zatáčce sklouzávat do jejího středu (tento jev byl detailněji popsán v~kapitole \ref{sec:let-skutecneho-letadla}). Častým zjednodušením je také nemožnost dosáhnout bodu přetažení (opět viz kap. \ref{sec:let-skutecneho-letadla}). Pomyslný omezovač nedovolí další zvyšování úhlu náběhu a letadlo přetažení nikdy nedosáhne. Tím odpadne část simulace, kdy se letadlo ocitá ve vzduchu s~minimálním nebo žádným vztlakem a mělo by začít padat. Jelikož však běžný hráč nemá nejmenší ponětí o~úhlu náběhu a jeho důsledcích na let, je tohoto zjednodušení často využíváno. Nevýhodou tohoto přístupu může být pro hráče viditelné omezení manévrovacích možností. Pokud se ale stále držíme v~kategorii her které se nesnaží býti skutečnými simulátory, není tento problém nijak závažný. Navíc lze vhodným návrhem hry takové omezení ospravedlnit, například použitím grafického modelu letadla který v~hráči vyvolá dojem těžkopádnosti a nízkého výkonu. Od takového letadla potom podvědomě nebude očekávat velké manévrovací schopnosti a snáze pochopí, když narazí na jejich hranice, které ve skutečnosti slouží primárně ke zjednodušení celého výpočtu.
Podobně lze mlčky předpokládat existenci nějakého inteligentního řízení, které hráči asistuje. To může kontrolovat další ovládací prvky letadla sloužící zejména ke zjednodušení práce pilota v~některých specifických situacích. (Jde mimo jiné o~tzv. \textit{sekundární ovládací prvky}, jejichž popis je však nad rámec této práce.) Takový pomocný systém je zcela jistě reálný a můžeme na něj nahlížet jako na jistou formu autopilota.
\vspace{\baselineskip}
Z~pohledu ovládání letadla se hry v~drtivé většině omezují výhradně na klapky na křídlech a výškové kormidlo. Pokud uvažujeme zmíněné \textit{asistované řízení}, lze těmito dvěma prvky letadlo plnohodnotně (stále pouze z~hlediska hry, nikoliv simulátoru) ovládat. V~některých případech bývá doplněna i možnost ovládání směrového kormidla. U~něj však nastává časté nepochopení jeho funkce a bývá implementováno jako možnost zatáčení letadla bez náklonu při zachování vodorovného letu. Takové letadlo pak zatáčí podobně jako například ponorka. Je otázkou, zda toto pramení z~neznalosti vývojáře hry anebo zda jde o~objednávku samotných hráčů. Průlet zatáčkou je totiž při použití \textit{nepochopeného} směrového kormidla mnohem snazší, dává hráči větší kontrolu nad letadlem a přirozeně snižuje požadavky na hráčovu zručnost. Ve výsledku záleží na herním vývojáři a na návrhu a zaměření hry, zda se bude držet reálného (byť zjednodušeného) ovládání letadla anebo se podvolí poptávce po snadno ovládaném letadle, přestože by tím popíral fyzikální zákonitosti.
V~neposlední řadě u~většiny leteckých her chybí možnost regulovat tah motoru. To je veličina se kterou pilot skutečného letadla v~průběhu letu a především pak manévrů aktivně pracuje. Je ale také jednou z~veličin ovlivňujících celkový vztlak letadla a při neznalosti všech důsledků by pro hráče bylo snadné ztratit nad letadlem kontrolu. Stejně tak se často ignoruje dříve zmiňovaná změna citlivosti ovládacích prvků letadla v~závislosti na tahu motoru. Obvyklým řešením bývá opět předpoklad existence nějakého palubního systému který tah motoru automaticky reguluje. Ve výsledku bývá v~průběhu letu tah konstantní a k~jeho změnám dochází pouze při vzletu nebo přistávání. Často také informace o~otáčkách motoru úplně chybí a jedinou zpětnou vazbou hráči je zvuk který motor vydává.
\subsection{Vstupní metody}
\label{sec:vstupni-metody}
Pokud se podíváme na letecké hry pro počítače, případně pro herní konzole, jsou ovládací možnosti vcelku omezené. Prakticky vždy se omezují na ovládání ve dvou osách, kdy pohyb v~jedné ose řídí klapky na křídlech letadla a pohyb v~druhé ose ovládá výškové kormidlo. Subjektivně je asi nejméně pohodlnou a současně nejméně přesnou metodou ovládání klávesnice počítače. Reakce na řízení jsou z~povahy kláves skokové a nelze ovlivňovat jejich intenzitu. Pozitivem použití klávesnice může být velké množství kláves, které můžou ovládat další prvky letadla. Tyto možnosti však využijí spíše simulátory.
O~mnoho lepší kontrolu nabízí joystick, případně jeho zmenšená varianta na ovladačích současných herních konzolí. Ovládání je opět dvouosé, ovšem s~možností plynule regulovat intenzitu. Nejdůležitější vlastností je ale podobnost se skutečným řízením letadla pomocí kniplu. To umožňuje především v~případě joysticku mnohem vyšší vtáhnutí hráče do hry. Nevýhodou může být omezený počet doplňkových ovládacích prvků, typicky je na ovladači pouze několik málo dalších tlačítek. Pokud by hra nabízela možnost detailnějšího ovládání letadla (například vysouvání podvozku nebo vypouštění světlic pro odlákání nepřátelských střel), mohl by to být problém.
\vspace{\baselineskip}
Rozšíření výkonných mobilních zařízení umožnilo příchod nových metod ovládání. Zcela nepoužitelná byla první generace mobilních telefonů s~klávesnicí, jelikož ty ještě neměly dostatek výkonu a k~rozšíření leteckých her u~nich nedošlo. Další popis se tedy zaměří na zařízení s~velkým dotykovým displejem. Taková zařízení často nemají hardwarová tlačítka nebo taková tlačítka slouží systému a hra je pro své potřeby nemůže využít. Novým prvkem je ale zapojení sensorů. Konkrétně gyroskopu a akcelerometru, které sledují orientaci, respektive zrychlení zařízení v~trojrozměrném prostoru. Hráč ovládá letadlo nakláněním zařízení přičemž se typicky neuvažuje svislá osa. Natočení do stran (pohyb podobný zatáčení volantem) ovládá klapky letadla a naklonění od sebe či k~sobě ovládá výškové kormidlo.
Ovládání her pomocí sensorů se rychle rozšířilo, pravděpodobně díky své inovativnosti, a v~současnosti si udržuje přinejmenším u~leteckých her většinový podíl. Má však také několik nedostatků. Asi nejzásadnějším je výchozí pozice zařízení. Při používání sensorů je potřeba stanovit referenční bod ke kterému naměřené hodnoty vztahujeme. Častým řešením je nastavit bod na začátku každého letu, poté se již měnit přirozeně nemůže. Problém nastává ve chvíli, kdy se hráč hrající hru sám pohybuje ve skutečném světě. Může jít například o~houpání se v~křesle anebo přesun v~nějakém dopravním prostředku. Hry se v~takovém případě stávají nehratelnými a zatím se neobjevila metoda, která by uměla tento jev odstranit.
Druhý nedostatek stojící za zmínku je změna úhlu pohledu při pohybu se zařízením. Displeje mají většinou relativně velký pozorovací úhel, i tak ale může docházet ke snižování viditelnosti pokud je zařízení až moc přikloněno, respektive odkloněno od pohledu hráče. Řešením může být vhodná volba citlivosti tak, aby stačilo zařízení naklánět jen v~\textit{pohledově bezpečném} rozsahu. Stejně tak může být problémové i otáčení displeje. Přestože zde se úhel pohledu nemění, dochází k~otáčení celého obrazu včetně uživatelského rozhraní. Na tuto skutečnost je nutné myslet při návrhu hry tak, aby uživatel špatnou čitelností prvků rozhraní nepřicházel o~důležité informace.
Dalším problémem je relativně pomalá odezva sensorického vstupu a tvar jeho průběhu. Zpoždění není zásadní, ale je pozorovatelné. Hru je navíc možné navrhnout tak, aby pro hráče bylo pochopitelné. Při letu s~velkým těžkopádným letadlem asi nikdo nebude očekávat reakce v~řádu zlomků sekundy. U~rychlých a akčních her by však toto mohlo být omezující. Průběh signálu vznikajícího v~sensoru navíc není ideální a běžně obsahuje výkyvy a šum. Je proto nutné jej filtrovat, čímž může docházet ke ztrátě přesnosti.
Poslední slabinou ovládání pomocí sensorů je společenská vhodnost. S~tím jak se herní zařízení stala mobilními se rozšířila i místa kde může člověk hry hrát. Ne vždy a všude bude okolí tolerovat hráče který před sebou mává mobilním telefonem či tabletem. Vývojář sice nemůže ovlivnit na jakých místech budou hráči jeho hru hrát, může se však přinejmenším zamyslet kdo bude cílovou skupinou a nakolik jsou členové této skupiny ochotní se ve společnosti tímto způsobem zviditelňovat.
\begin{figure}[ht]
\begin{center}
\includegraphics[width=\textwidth]{fig/race-pilot.png}
\caption{Screenshot hry Race Pilot ovládané pomocí sensorů. Šedé čtverce po stranách ovládají směrové kormidlo, které je však implementováno nerealisticky; zdroj: Google Play}
\label{fig:racepilot}
\end{center}
\end{figure}
\vspace{\baselineskip}
Z~pohledu mobilních zařízení je asi nejzásadnější možnost ovládání dotykem. Ta se neomezuje pouze na tlačítka a polohovací páčky a je zde možné navrhnout téměř libovolné ovládací prvky. K~tomu lze využít položení prstu na displej, který odpovídá stisku klávesy, a může tedy řídit dvoustavové veličiny. Současně je možné sledovat pohyb jednoho či více prstů v~nějaké předem dané oblasti a získávat plynulé změny hodnoty (například vzdálenost bodu dotyku od referenčního bodu). Dalším velmi silným prostředkem je vzájemná poloha dvou a více bodů dotyku.
Ve srovnání se sensorickým ovládáním netrpí zmiňovanými nedostatky jako je nutné natáčení a naklánění zařízení a nepoužitelnost v~případě že je hráč v~pohybu. Při návrhu dotykového ovládání je však nutné počítat s~tím, že hráč může svými prsty překrývat značnou část displeje. To se může projevit zejména u~displejů s~menší úhlopříčkou. Stejně tak je vhodné se při návrhu ovládání zamýšlet nad tím, kdo bude typickým hráčem hry. Zcela jistě bude rozdíl mezi prstem dítěte a dospělého jedince, navíc s~přihlédnutím k~vykonávané profesi, a je nutné tomuto ovládací metodu přizpůsobit.
Přestože se zdá že právě dotykové ovládání nabízí široké možnosti pro inovace, herní vývojáři je zatím spíše nevyužívají. V~případě leteckých her drtivě převládá využití sensorů, případně dotykového ovladače na displeji -- bodu který hráč dotykem posouvá a simuluje tak pohyb joysticku.
\chapter{Návrh dotykového ovládání a letecké hry}
V~této části práce popíši nově navrhovanou metodu ovládání a varianty její implementace. Dále rozvedu jak by měla výsledná hra vypadat a jaké budou její herní mechanismy. Stále je potřeba mít na paměti, že se pohybujeme v~oblasti her pro mobilní zařízení. Rozměry obrazovek, tedy ovládacích ploch, budou omezené, stejně jako výkon zařízení. Stejně tak povaha práce s~mobilním zařízením je diametrálně odlišná od povahy práce s~běžným počítačem nebo herní konzolí. Tyto faktory bude nutné uvážit.
Při navrhování hry jsem vycházel zejména z~knih Game Industry, které sepsali čeští herní vývojáři \cite{gameindustry1, gameindustry2}.
\section{Nově navrhovaná metoda ovládání}
\label{sec:nova-metoda-ovladani}
Při uvažování o~inovativním přístupu k~ovládání letadla jsem téměř okamžitě vyřadil vstup pomocí sensorů. Důvody zmíněné v~kapitole \ref{sec:vstupni-metody} mi přijdou příliš omezující. Současně v~této oblasti nejsou velké možnosti k~dalšímu rozšiřování tak, aby pro hráče zůstala zachovaná pohodlnost a intuitivnost ovládání.
Navrhované ovládání tedy bude čistě dotykové. Tím bude možné jej používat kdekoliv a jeho hlavními přednostmi bude vysoká přesnost a rychlá odezva. Díky tomu na něm půjde vystavět i velmi rychlou a akční hru. Oproti existujícímu dotykovému vstupu by mělo hráči nabídnou větší vtažení do děje a snad i navodit pocit jisté jedinečnosti, tedy že hráč používá něco nového a neobvyklého. Proti této myšlence jde částečně požadavek na nízké znalosti a zkušenosti. Hráč by měl princip ovládání pochopit prakticky ihned, v~nejlepším případě pak bez jakékoliv nápovědy nebo předchozího proškolení. Současně musí být ovládání maximálně pohodlné. To totiž opět není cílem hry, ale pouze prostředkem k~jeho dosažení. V~ideálním případě by mělo být pro hráče natolik přirozené, že nad ním nebude muset přemýšlet či ještě lépe vůbec si uvědomovat jeho existenci. Pochopitelně jej také nesmí omezovat a mělo by postihnout všechny podstatné funkce ovládaného objektu, v~tomto případě letadla.
\vspace{\baselineskip}
Pro další popis rozdělím ovládání do jednotlivých os. První z~nich bude osa ovládající klapky na křídlech a tím i náklon celého letadla, tedy rotaci kolem osy definované směrem letu. V~případě joysticku a z~něj vycházejících metod jde o~pohyb ukazatele či ovladače do stran. Navrhovaná metoda ovládání vyžaduje aby existovaly právě dva body ve kterých se hráč dotýká displeje. Tyto se použijí k~sestrojení pomyslné kružnice, jejíž střed bude ležet na středu úsečky definované oběma body. Úhel náklonu letadla pak odpovídá úhlu o~který se spojnice bodů odchyluje od vodorovné osy kružnice. Situace je zachycená na obrázku \ref{fig:ovladani-roll}. Ačkoliv to z~popisu nemusí být zřejmé, na poloměru kružnice ve skutečnosti vůbec nezáleží. Hráč může průběžně vzdálenost bodů měnit a pokud oba body zůstanou na původní úsečce, náklon letadla se nezmění. Vedlejším efektem je pak možnost takto měnit citlivost ovládání. Zatímco u~kružnice o~velkém poloměru je možné dosáhnout velmi jemné regulace náklonu, při přibližování bodů a zmenšování poloměru se citlivost přirozeně zvyšuje. Důležité je, že nezáleží na pozici kružnice v~rámci dotykové plochy a v~každém okamžiku se vytváří tak, aby její střed ležel ve středu spojnice mezi oběma body dotyku. Všechny dvojice protilehlých bodů takové kružnice pak definují manévrovací možnosti v~dané ose rotace. Ty budou bez dalších omezení odpovídat 360°. Pro ovládání letadla však bude vhodnější použít rozsah $-180^{\circ}$ až $180^{\circ}$, kde nulový úhel náklonu bude odpovídat vodorovné ose obrazovky. (Přestože se tato práce soutřeďuje na letecké hry, je jistě možné navrhovanou metodu ovládání použít pro ovládání zcela jiných objektů. V~takovém případě můžou být vhodné rozsahy vstupů, stejně jako neutrální poloha, pochopitelně odlišné.) V~zájmu sjednocení rozsahů vstupních hodnot z~obou os bude posledním krokem převod do rozsahu $-1$ až $1$, kde opět $0$ představuje neutrální hodnotu.
Je vhodné zde také zdůraznit, že rozsah hodnot ovládání nemusí přímo odpovídat silám aplikovaným na ovládaný objekt. V~případě letadla si lze hodnoty ovládání představit jako úroveň přitažení kniplu nebo sešlápnutí pedálu. Jde o~bezrozměrné hodnoty a teprve jejich \textit{dosazením} na vstup konkrétního matematického modelu letadla vzniknou síly na něj aplikované. To odpovídá skutečnosti, kde malé letecké letadlo používá rámcově stejné základní ovládací prvky jako například velký dopravní letoun. A~teprve rozměry a ostatní charakteristiky ovládacích ploch letadla určují, jak moc letadlo změní směr svého letu.
\begin{figure}[ht]
\begin{center}
\includegraphics[scale=0.4]{fig/controls-roll.png}
\caption{Ovládání náklonu letadla. Červená místa značí body dotyku prstů. Intenzita změny náklonu odpovídá úhlu, který svírá úsečka ohraničená těmito body s~vodorovnou osou obrazovky. Body je možné podél vodorovné osy libovolně posouvat bez vlivu na řízení.}
\label{fig:ovladani-roll}
\end{center}
\end{figure}
Druhou z~ovládaných veličin je výškové kormidlo. To bývá u~joysticku a podobných metod reprezentováno přitáhnutím nebo odtlačením ovládací páčky. Princip nastiňuje obrázek \ref{fig:ovladani-pitch}. Opět zde pracujeme se středem úsečky mezi dvěma body dotyku a odečítáme jeho vzdálenost od referenčního bodu, který je na obrázku znázorněn uprostřed displeje. Střed displeje je zvolen pouze pro ilustraci a ve skutečnosti se referenční bod přemístí pokaždé když se displeje začnou dotýkat právě dva prsty. V~okamžik prvotního dotyku se referenční bod nastaví a nemění se dokud se oba prsty nepřestanou displeje dotýkat. Jakmile se znovu dotknou, referenční bod se přesune na novou pozici na které opět po dobu kontaktu prstů s~displejem zůstává. Hráč tak při pouhém položení prstů na displej bez dalšího pohybu nijak nezasahuje do ovládání letadla. Současně tento přístup umožňuje hráči začít letadlo ovládat dotykem v~libovolné části displeje, což může být výhodné zejména u~tabletů a obecně zařízení s~větší úhlopříčkou dotykové plochy.
Funkční rozsah tohoto vstupu bude omezený hranami obrazovky. Zde se nabízí více řešení problému. Je možné jako rozsah definovat celou výšku obrazovky. V~místě prvotního dotyku se pak nastaví neutrální hladina s~tím, že obě oblasti (kladná i záporná) budou mít stále výšku poloviny obrazovky, třebaže by zasahovaly mimo oblast obrazovky. Takový přístup nabízí optimální chování v~situaci, kdy prvotní dotyk nastává právě v~polovině výšky obrazovky. V~ostatních pozicích dochází k~oříznutí rozsahu a tím ke zhoršení manévrovacích možností. Jiným přístupem může být zmenšení výšky oblastí tak, aby žádná z~nich nepřesahovala oblast obrazovky. Obě ovládací oblasti by samozřejmě měly být shodně velké, a rozměr by se tedy řídil podle menší z~nich. Slabinou tohoto přístupu je různá citlivost vstupu v~závislosti na místě prvotního dotyku. Optimální chování je opět v~oblasti kolem středu obrazovky, s~klesající vzdáleností k~jejímu okraji pak roste citlivost až do místa, kde vstup přestává být použitelný.
Konkrétní implementace výškového kormidla může opět záviset na povaze ovládaného objektu. V~případě této práce jsem experimentoval s~více variantami. Více k~tomuto tématu bude popsáno v~kapitole \ref{sec:implementace-prototypu}.
\vspace{\baselineskip}
\begin{figure}[ht]
\begin{center}
\includegraphics[scale=0.4]{fig/controls-pitch.png}
\caption{Ovládání vzestupu a poklesu letadla. Červená místa značí body dotyku prstů a současně definují úsečku. Intenzita změny výšky letadla závisí na vzdálenosti středu této úsečky od vodorovné osy obrazovky. Body je možné podél vodorovné osy libovolně posouvat bez vlivu na řízení.}
\label{fig:ovladani-pitch}
\end{center}
\end{figure}
Kombinací těchto dvou principů vzniká ovládání schopné pokrýt stejný rozsah veličin jako zmiňovaný \textit{klasický dotykový} anebo sensorový vstup. Skládá se přitom ze známých prvků -- ovládání náklonu může hráči evokovat zatáčení volantem, změna výšky pak gesto běžně používané pro svislé posouvání obsahu. Je proto možné předpokládat, že navrhované ovládání bude pro hráče skutečně intuitivní a k~jeho vysvětlení nebude potřeba více než dva nákresy uvedené v~této kapitole.
\vspace{\baselineskip}
Navrhované ovládání by bylo možné dále rozšířit o~možnost ovládání směrového kormidla. To by představovalo třetí ovládací osu, tedy svislou osu kolem které může letadlo rotovat. Veličinu by bylo možné odečítat podobně jako je odečítána poloha výškového kormidla. Šlo by o~vzdálenost středu úsečky tvořené dvěma body dotyku od svislé osy obrazovky, procházející bodem počátečního dotyku. (Výškové kormidlo zde měří vzdálenost od osy vodorovné.) Využití této třetí ovládací osy však nemusí být ideální a navrhované ovládání jako celek by mohlo na hráče působit příliš složitě. Ideálním řešením by bylo tuto variantu implementovat a získat zpětnou vazbu od prvních zkušebních hráčů.
\section{Žánrové zařazení a herní mechanismy}
\label{sec:zanrove-zarazeni}
Hra se bude odehrávat v~prostředí leteckých závodů. Hráč je v~roli pilota akrobatického letadla který navštěvuje jednotlivé závody. Samotný závod pak spočívá v~průletu branek ve stanoveném pořadí (přesná dráha letu se nevyžaduje), vzlet ani přistávání se neuvažuje. Branky můžou být různého typu: obyčejná, jednosměrná nebo branka vyžadující průlet v~předem dané poloze letadla. Právě poslední typ branky je typickým prvkem leteckých závodů, kdy nezáleží pouze na času letu, ale hodnotí se také přesnost se kterou pilot brankou proletí. Rozlišuje se průlet vodorovný, svislý (takzvaný \textit{nožový let}) a průlet vzhůru nohama.
Cílem hráče je proletět všemi brankami v~co nejkratším čase. Při průletu brankou vyžadující konkrétní polohu letadla se kontroluje přesnost průletu. Pokud je nízká, přičte se k~času průletu postih závisející na velikosti odchylky. Průlet brankou se však započítá i v~případě, že bude odchylka maximální. To je rozdíl oproti jednosměrné brance, která se považuje za splněnou pouze při průletu ze správného směru. U~obyčejné branky žádná omezení ani postihy nejsou. U~každého závodu jsou definovány tři časy průletů, které se hráč snaží překonat. Tyto časy vzniknou pravděpodobně v~prvotní fázi odhadem, později se upraví na základě testování a zpětné vazby od hráčů.
Celá hra se pak sestává s~množství závodů, kde každý se může odehrávat v~jiném grafickém prostředí (ne však nutně vždy unikátním). Jednotlivé závody spolu ale nijak nesouvisejí. Na začátku má hráč zpřístupněný jediný závod do kterého může vstoupit, ostatní závody jsou uzamčené. Ve chvíli kdy hráč v~některém ze závodů překoná alespoň jeden z~předdefinovaných časů, považuje se závod za splněný a zpřístupní se jeden další závod. Nezbytným předpokladem pro plynulý průchod všemi závody bude postupně se zvyšující obtížnost. Tak jako porostou zkušenosti a dovednosti hráče, musí stoupat i velikost výzvy před kterou je postaven. V~případě příliš jednoduchých závodů by hráč procházel bez nejmešího odporu a hra by byla nudná. U~příliš obtížných závodů by se naopak hráč mohl zaseknout bez možnosti pokračovat dál. Motivací pro nejlepší možné splnění závodu, tedy překonání nejnižšího z~časů, pak mohou být speciální odměny, jako například nové letadlo nebo jiné dekorativní prvky.
V~počáteční fázi (tedy prvních vydaných verzích hry) není žádoucí zpřístupnit hráči více letadel s~různým výkonem, třebaže by se takové letadlo nabízelo jako vhodná odměna za vynikající výkony. Taková možnost by totiž zvýšila obtížnost odhadu přiměřené náročnosti závodu. Lze jistě předpokládat, že hráči nováčkovi potrvá nějakou dobu, než dostane letadlo plně pod kontrolu a naučí se jej ve všech situacích bezpečně ovládat. V~takové situaci není nutné hráče zatěžovat úvahami o~rychlosti či ovladatelnosti letadla. Naopak bude vhodné nabídnout pouze jediné letadlo a hráče odměňovat právě formou dekorativních úprav, například jiného laku (tedy textury) letadla.
Uvedení různě výkonných a ovladatelných letadel může přijít ve chvíli, kdy bude možné předpokládat jistou zkušenost hráče. Takový hráč už letadlo plně ovládá a hlavná výzva závodů se přesouvá od schopnosti proletět všechny branky (v~téměř libovolném čase) k~cíli proletět je co nejrychleji. Zde bude možné zpřísnit požadavky na rychlost průletu závodem, včetně možnosti definovat časy na první pohled nesplnitelně. To hráče přinutí vrátit se k~dřívějším závodům, dosáhnout v~nich nejlepšího hodnocení (to by v~této chvíli měl zvládnout, neboť jeho zkušenosti vzrostly) a odměnou získat rychlejší letadlo, se kterým již bude možné plnit obtížnější závody.
\vspace{\baselineskip}
Celkově by hra měla působit akčně a rychle, čemuž musí odpovídat i metoda ovládání. Právě u~akrobatických letů je potřeba aby odezva na vstup byla minimální a hráč neustále cítil že má letadlo plně pod kontrolou. Takovouto hru by zcela jistě bylo velmi obtížné ovládat pomocí sensorického vstupu, a je proto dobrým prostředkem pro prezentaci předností nově navrhovaného ovládání.
\section{Rozšiřitelnost hry}
Podstatným faktorem je také následná rozšiřitelnost hry. Aktualizace aplikace se nemusí zaměřovat pouze na opravování chyb, ale mohou současně přidávat další herní obsah. To je z~pohledu hráče velmi žádoucí a právě informaci o~aktualizaci mu může připomenout dávno dohranou hru. Aktualizace se pak stávají menšinovým propagačním nástrojem a je velmi žádoucí s~nimi takto pracovat. Příliš časté vydávání nových verzí může být obtěžující, zejména pokud hráč většinu herního obsahu zkonzumoval a nového se mu nedostává. I~v~případě že má vývojář větší množství nevydaného obsahu, je vhodné jeho zveřejňování dávkovat a snažit se tak udržovat dlouhodobý zájem o~hru.
Konkrétně u~navrhované hry je relativně jednoduché doplnit další herní režimy. Ve chvíli kdy je implementováno ovládání letadla a jeho interakce s~herním světem, je jen otázkou téměř triviální logiky, zda se budou počítat body za prolétnuté branky nebo zda bude cílem proletět trasu v~co nejkratším čase, případně cokoliv jiného.
Podobně je tomu v~případě vytváření nových herních úrovní, tedy závodů. Hra nepočítá s~unikátním prostředím pro každý jeden závod a scenérie se nutně budou opakovat. Nejjednodušším doplněním obsahu tedy může být využití existujícího prostředí a vytvoření pouze nové sady branek. O~něco náročnějším rozšířením by pak bylo doplnění nového prostředí. Tam je náročnost zejména na straně grafických prací. Tímto způsobem je možné připravovat tématické balíčky s~novými úrovněmi k~různým významným příležitostem, což může být opět zajímavý propagační prvek.
Z~tohoto hlediska nebude nutné implementovat všechny herní mechanismy z~předchozí kapitoly ihned. S~ohledem na časové možnosti a zmíněného dávkování obsahu může být dokonce vhodné začít s~jednoduchou hrou a tu následně rozšiřovat. Zpětná vazba od hráčů a informace získané ze statistik užívání pak mohou poskytnout indicie k~tomu, kterým směrem hru dále vyvíjet. Současně se v~případě neúspěchu minimalizuje úsilí vynaložené k~vytvoření hry, a tím tedy i ztráty.
V~neposlední řadě je potřeba uvažovat o~monetizaci hry, tedy o~její ziskovosti. Vzhledem k~výše popsaným mechanismům se nabízí jako vhodné řešení zpoplatnění některých závodů, případně možnost zakoupit si předčasně některé odměny. Opět ale bude potřeba dbát na vyváženost hry a zachování její zábavnosti. Nelze proto hráči ihned nabídnout vykonnější letadlo, třebaže za poplatek. Vhodnější bude počkat do chvíle, kdy by toto letadlo mohl získat přirozeně a teprve poté jej nabídnout ke koupi.
\chapter{Implementace a publikování hry}
V~této části popíši implementaci navrženého ovládání a hry. Veškerá práce probíhala ve vývojovém prostředí Eclipse, doplněné o~moduly specifické pro platformu Android. Tyto doplňky jsou součástí podkladů pro vývojáře a jsou dostupné z~oficiálních webových stránek \cite{androidDevelopers}.
Poslední podkapitola bude věnovaná publikování vytvořené hry nejdříve v~omezené skupině lidí a následně na internetovém obchodě Google Play. Zhodnotím také výsledky testování, průzkumů a poznatky získané sběrem statistik užívání. V~závěru kapitoly také shrnu možnosti dalšího vývoje.
\section{Prototyp a prvotní testování}
\label{sec:implementace-prototypu}
V~počátku práce na aplikaci jsem navrhl a impementoval prototyp založený čistě na platformě Android, bez použití dalších knihoven. Potýkal jsem se s~problémy, které byly popsány v~kapitole \ref{sec:negativna-nativniho-vyvoje}, a zpětně lze prohlásit, že vývoj bez pomocných nástrojů či knihoven bude přinejmenším velmi obtížný.
Cílem prototypu byla prvotní implementace navržného ovládání a validace jeho funkcionality. Obecně by mělo jít o~maximálně zjednodušenou verzi aplikace, kde bude možné velmi rychle doplnit libovolnou funkcionalitu, především z~hlediska vstupních metod. Přestože měl být prototyp takřka jednoúčelovou aplikací, jeho vnitřní struktura je v~mnohém podobná struktuře aplikace používající knihovnu Libgdx. To je pravděpodobně z~velké části dáno přímo platformou Android, která se návrhem svého rozhraní (a samozřejmě i všemi ukázkovými příklady) snaží vést programátora ke správně navržené struktuře aplikace. Správná struktura v~tomto kontextu znamená aplikaci, která je vnitřně přehledná a efektivní, její součásti pak znovupoužitelné a dobře testovatelné. Tyto požadavky nemají v~případě prototypu velkou prioritu (snad s~výjimkou znovupoužitelnosti součástí), přesto je vhodné zmínit, že jsem jich při práci na prototypu bez velkého úsilí dosáhl anebo se jim alespoň přiblížil.
Z~pohledu uživatele je prototyp velmi omezenou verzí hry. Neobsahuje uživatelské rozhraní, vizuální stránka je strohá a stejně tak nejsou implementovány ani herní mechanismy týkající se závodů. Dalo by se říct že jde pouze o~letadlo nekonečně se pohybující v~prostoru, neboť detekce kolizí, ať už s~terénem anebo průlet kruhy, v~prototypu chybí také. Pro představu o~prvotní verzi vizte obrázek \ref{fig:prototyp-1}. Později do hry přibylo více vizuálních prvků, především model ostrova pro experimenty s~terénem. Pro lepší pocit ze hry při jejím testování jsem také nahradil hlínu texturou vody. Závěrečná podoba prototypu je na obrázku \ref{fig:prototyp-2}. Kromě zmíněného ostrova bylo v~prostoru rozmístěno i několik kruhů, na kterých bylo možné testovat ovladatelnost letadla zvolenou metodou.
\vspace{\baselineskip}
\begin{figure}[ht]
\begin{center}
\includegraphics[width=\textwidth]{fig/ingame-proto-1.jpg}
\caption{Jedna z~prvních verzí prototypu vyvíjeného bez použití pomocných knihoven. Primárním cílem bylo testování ovládání. Vizuální stránka proto byla zpočátku velmi slabá.}
\label{fig:prototyp-1}
\end{center}
\end{figure}
Ovládacích metod obsahoval prototyp hned několik. V~zájmu rychlé implementace se k~jejich výběru používá nativní menu platformy. V~základu je zvolená již popsaná nová metoda ovládání a dále je možno přepnout na ovládání pomocí akcelerometru nebo virtuálního joysticku na obrazovce. Takto mohlo dojít k~přímému porovnání ovládacích metod. Současně vznikaly i varianty jednotlivých metod s~různou citlivostí nebo lehce odlišným chováním. To se týká především navrhované metody, kde existuje více přístupů, jak jsem popsal v~kapitole \ref{sec:nova-metoda-ovladani}. Ukázalo se, že je vhodnější použít variantu, kdy mají ovládací oblasti pevně danou velikost v~závislosti na rozlišení obrazovky, přičemž případný přesah mimo zobrazovací oblast se ignoruje. Zjednodušeně lze říct, že tato metoda udržuje konstantní citlivost na úkor možnosti dosáhnout vždy plného rozsahu ovladatelnosti. Ideální místo prvotního dotyku tedy bude přesně v~polovině výšky obrazovky. Tuto informaci bude potřeba vhodným způsobem předat uživateli a více tento problém popíši v~kapitole \ref{sec:implementace-ui}.
Druhá popisovaná varianta měnila rozměry obládacích oblastí (a tím i citlivost) v~závislosti na poloze prvotního dotyku tak, aby nikdy nepřesáhly okraje obrazovky. Tato verze se ukázala jako méně vhodná. Původní předpoklad, že hráč bude mít v~průběhu celého letu trvale položené prsty na obrazovce, se ukázal jako nesprávný. Hráči mají naopak tendence prsty zvedat a opětovně se obrazovky dotýkat, což mění refrenční hladinu pro ovládání. V~případě této varianty se navíc měnila i citlivost a hráč ztrácel přehled o~tom, jakou odezvu pohyb jeho prstů vyvolá. Ovládání se tím stávalo složité a nepřirozené, proto jsem tuto variantu zavrhl.
\begin{figure}[ht]
\begin{center}
\includegraphics[width=\textwidth]{fig/ingame-proto-2.jpg}
\caption{Závěrečná podoba prototypu obsahovala několik variant ovládání. I~v~grafickém zpracování nastal posun, neboť testující hráče příliš zjednodušené prostředí rušilo od hlavního cíle, tedy testování ovládání.}
\label{fig:prototyp-2}
\end{center}
\end{figure}
Podobně jsem experimentoval i s~ovládáním směrového kormidla, jakožto třetí ovládané osy, jak bylo opět popsáno v~kapitole \ref{sec:nova-metoda-ovladani}. Tato varianta, byť návrhově čistá a konzistentní s~ostatními osami, se také ukázala jako nevhodná. Problémem zde nebyla nepřirozenost, ale především komplexnost ovládání. Opět připomínám, že hra necílí na profesní piloty či letecké modeláře, ale na běžné hráče. Ti bohužel v~obecném měřítku nejsou schopní ovládat letadlo ve více než dvou osách. Na druhou stranu se ukázalo, že někteří jednotlivci toto ovládání považují za vhodné, a dokonce by jej preferovali před ovládáním dvouosým. Lze o~něm proto uvažovat jako o~možném volitelném rozšíření.
\vspace{\baselineskip}
Zmíněné testování probíhalo v~průběhu vývoje prototypu a jednotlivých metod na vesměs náhodném vzorku testerů. Nebylo však nijak organizované a neexistují z~něj žádné záznamy, právě kvůli plynulému vývoji nezřetelným iteracím. Ne vždy tak bylo možné nechat stejnou osobu srovnat varianty před a po úpravě. V~této fázi by byl jistě prostor pro zlepšení. Detailnější testování jsem provedl později před zveřejněním finální hry a blíže jej popíši v~kapitole \ref{sec:verejne-testovani}. Pro úplnost doplním, že vstup pomocí akcelerometru nebo virtuálního joysticku byl v~drtivé většině případů označen za méně pohodlný a vzhledem k~povaze hry také méně vhodný.
\section{Struktura aplikace}
Tato kapitola popíše strukturu aplikace, která má v~případě použití knihovny Libgdx jistá specifika. Budou také představeny některé diagramy tříd, zachycující nejdůležitější vazby mezi herními objekty. Úplny diagram tříd tato práce neobsahuje, neboť by od čtenáře vyžadoval hlubokou znalost použité knihovny a pro běžného čtenáře by nebyl příliš popisný.
\subsection{Oddělení platforem}
\label{sec:oddeleni-platforem}
Jak již bylo naznačeno v~popisu knihovny Libgdx v~kapitole \ref{sec:libgdx}, jednou z~jejích výhod je přenositelnost mezi platformami. Pro vytvářenou hru budou relevantní platformy Android a PC (bez rozdílu operačního systému). V~době psaní této práce existuje v~knihovně Libgdx také podpora platformy iOS, která se nachází na mobilních zařízeních firmy Apple. Tato podpora je však ve velmi ranném stádiu a proces sestavení aplikace vyžaduje proprietální software, vývojářskou licenci a fyzické zařízení této platformy. Kvůli těmto okolnostem jsem podporu iOS vyloučil. Z~povahy používané knihovny by ovšem bylo přidání podpory této platofrmy do již hotové hry triviální záležitostí.
\vspace{\baselineskip}
Pro další popis budu využívat pojem \textit{projekt} tak, jak je chápán z~hlediska vývojového prostředí, zde programu Eclipse. Jeden projekt lze sestavit do jedné spustitelné aplikace či knihovny. Doporučeným postupem je zde vyčlenění veškeré aplikační logiky do jednoho projektu, ze kterého sestavením vznikne knihovna. Tuto knihovnu využívají ostatní projekty, vždy jeden pro každou cílovou platofrmu. Existence projektu pro platformu Android je vyžadována. Z~historických a praktických důvodů jsou právě zde uloženy veškeré datové soubory budoucí aplikace a varianty pro ostatní platformy se na tento obsah pouze odkazují (ke kopírování dat dochází až při sestavování konkrétních aplikací). Projekty pro jednotlivé platformy ale typicky plní úlohu triviálního zavaděče, který pouze připraví prostředí pro běh a následně předá řízení hlavní aplikační logice. Podoba těchto zavaděčů je pevně daná a v~průběhu vývoje aplikace se již prakticky nemění.
Ukázalo se jako velmi výhodné vedle platformy Android použít i platformu PC a provádět maximální možné množství vývoje právě zde. Narozdíl od mobilní aplikace, kde probíhá typický proces sestavení a následné instalace do zařízení, zde dochází k~sestavení a spuštění přímo v~počítači vývojáře. To probíhá řádově v~jednotkách sekund, tedy zásadně rychleji než na mobilním zařízení. Díky knihovně Libgdx jsou přitom rozdíly mezi platformami z~pohledu vyvíjené hry minimální a lze takto vyvíjet celé uživatelské rozhraní i veškeré vykreslování 3D scény. Problém nastává při implementace ovládání, které pro svoji funkčnost vyžaduje současný dotyk dvou prstů. Tuto část bylo tedy nutné vyvíjet přímo na skutečném mobilním zařízení. Díky důmyslnému systému práce se vstupem (který je popsán v~kapitole \ref{sec:libgdx-vstup}) lze nahradit dotykový vstup vstupem z~klávesnice. Takové řešení sice není ideální (řízení letadla je skokové), ale pro vývoj částí se vstupem přímo nesouvisejících je více než postačující.
Dalším důvodem k~vytvoření projektu pro platformu PC může být také tvorba propagačních materiálů, typicky videí. V~případě mobilních zařízení je snímání obrazovky a současné ukládání ve formě videa velmi omezené nebo úplně nemožné. Je to dáno jak výkonovými možnostmi, tak i potřebou speciálního oprávnění takové snímací aplikace. U~běžného počítače by v~současnosti požadavky na výkon neměly představovat zásadní problém a stejně tak je snadno dostupný i software pro záznam. Opět díky sjednocenému chování napříč platformami je možné vytvořit video přímo ze hry, aniž by neznalý divák rozeznal rozdíl. Stejně jako při vývoji zde může nastat problém se vstupem. Je však možné při hraní na mobilním zařízení veškerý vstup zaznamenávat a následně jej v~počítači přehrávat do běžící aplikace. Tímto způsobem lze velmi efektivně vizualizovat i místa dotyku a vytvořit například názornou ukázku, jak může hráč svými dotyky letadlo ovládat.
\subsection{Objektová struktura aplikace}
Nyní se v~popisu přesunu od prototypu k~finální aplikaci vystavené nad knihovnou Libgdx. V~dřívějších kapitolách již bylo zmíněno, že knihovna umožňuje rozčlenění aplikace do tzv. \textit{obrazovek}. Vyvíjená hra této možnosti využívá, jak je vidět z~obrázku \ref{fig:classes-main}. Do diagramu jsou zaneseny jen základní třídy aplikace a jejich vazba na třídy z~knihovny. Každá z~obrazovek má pak vazby další, které v~zájmu přehlednosti v~diagramu znázorněny nejsou.
Základní třídou aplikace je \texttt{FlightGame}, která po spuštění vytváří jednotlivé obrazovky a zajišťuje mechanismy pro jejich přepínání. Současně vytváří a zpřístupňuje objekty \texttt{AssetManager} a \texttt{LevelsManager} pro načítání obecných datových souborů a definičních souborů pro jednotlivé závody.
První obrazovka která se po spuštění aplikace zobrazí je \texttt{LoadingScreen}. Ta obsahuje triviální animaci informující uživatele o~načítání hry. Tato obrazovka musí být velmi jednoduchá, aby v~době mezi spuštěním aplikace a jejím zobrazením nenabýval uživatel dojmu, že aplikace přestala pracovat. Ihned po zobrazení načítací obrazovky se volá \texttt{AssetManager}, který načte podklady pro ostatní obrazovky, včetně samotné hry. Tato funkcionalita je již součástí Libgdx. V~závěru načítání je volán \texttt{LevelsManager}, který ze souborů ve formátu JSON načte definice jednotlivých závodů, aby na jejich základě mohla obrazovka hlavního menu vytvořit navigační rozhraní. Přepnutí na obrazovku samotného závodu pak v~rámci jednoho spuštění aplikace probíhá vždy téměř okamžitě. Proběhne totiž pouze přesunutí již načtených herních objektů na správné pozice a žádným dalším náročným operacím nedochází.
\vspace{\baselineskip}
\begin{figure}[ht]
\begin{center}
\includegraphics[width=\textwidth]{fig/classes-main.eps}
\caption{Diagram klíčových tříd aplikace. Vstupním bodem je třída \texttt{FlightGame}, která vytváří a udržuje jednotlivé obrazovky, potomky třídy \texttt{BaseScreen}.}
\label{fig:classes-main}
\end{center}
\end{figure}
Datová reprezentace světa je znázorněná na obrázku \ref{fig:classes-simulation}. Diagram opět některé souvislosti v~zájmu přehledosti zjednodušuje. Klíčovou třídou je zde \texttt{GameWorld}, reprezentující právě herní svět. Z~hierarchie dědičnosti lze vyčíst, že třída rozšiřuje třídy \texttt{BulletWorld} a \texttt{BaseWorld}. Druhá zmíněná, tedy \texttt{BaseWorld}, představuje základní svět obsahující vykreslitelné entity. (Samotné vykreslování však provádí třída \texttt{GameRenderer}, spravovaná třídou herní obrazovky.) Každá entita vzniká ze svého \textit{konstruktoru} (ve smyslu třídy \texttt{Constructor}). Konstruktor definuje 3D model entity a v~případě, že má být entita součástí kontroly kolizí, definuje i její kolizní tvar. Entita je pak instancí svého konstruktoru na určitém místě ve scéně (přesněji řečeno s~určitou transformací). Tento princip umožňuje sdílet zdrojová data všech entit stejného typu a snižovat tak paměťové i výkonové nároky.
Základní svět \texttt{BaseWorld} tedy podporuje pouze vykreslení entit. Jeho potomek \\\texttt{BulletWorld} rozšiřuje toto chování o~možnost provádění detekce kolizí. K~tomu rozšiřuje i konstruktor na \texttt{BulletConstructor}, obsahující právě informaci o~kolizním tvaru a povaze tělesa jak byla popsána v~kapitole \ref{sec:bullet}. Stejně tak rozšiřuje entitu na \texttt{BulletEntity}, se kterou může knihovna detekci provádět. Poslední rozšíření světa je právě na nejdůležitější třídu \texttt{GameWorld}, která nad běžící simulací spouští samotnou hru ve smyslu dodání sémantiky kolizím objektů. Z~tohoto důvodu zavádí třídu \texttt{GateEntity}, která reprezentuje entitu, s~níž je kolize žádoucí, tedy branku nebo kruh k~prolétnutí.
Třídy spadající pod knihovnu Bullet Physics obsahují ve svém názvu prefix \texttt{bt}. Pro použití této knihovny není nutná detailní znalost její vnitřní struktury, proto i já popíši tyto třídy jen velmi zběžně. Základní třídou je \texttt{btCollisionShape}, tedy kolizní tvar objektu. Podobně jako vytvořením instance konstruktoru vzniká entita, tak vytvořením instance kolizního \textit{tvaru} vzniká kolizní \textit{objekt}, který je součástí entity a reprezentuje jej třída \texttt{btCollisionObject}. Z~hlediska výkonnosti detekce kolizí je důležité, že opět objekty vzniklé ze stejného tvaru sdílejí stejné datové struktury. Dále fyzikální svět obsahuje instanci třídy \texttt{btCollisionWorld}, což je knihovní reprezentace prostředí, ve kterém detekce probíhá. (Z~důvodů kompatibility rozhraní je do třídy \texttt{BulletWorld} pouze vložen a třída z~něj přímo nedědí.) Toto prostředí představuje zjednodušenou verzi fyzikálního světa, kde je možné pouze testovat zda dochází ke kolizi některých objektů. Fyzikální simulace ve smyslu například volného pádu těles nebo jejich vzájemná interakce zde neprobíhá. Knihovna Bullet Physics toto samozřejmě nabízí, ovšem za cenu značného nárůstu požadavků na výkon. Vyvíjená hra je navíc záměrně navržena tak, aby plnohodnotná simulace probíhat nemusela.
\begin{figure}[ht!]
\begin{center}
\includegraphics[width=\textwidth]{fig/classes-simulation.eps}
\caption{Diagram tříd reprezentujících herní svět. Hlavní třída \texttt{GameWorld} je po vytvoření předána obrazovce, která řídí průběh hry. Herní svět obsahuje entity reagující na kolize a také speciální objekt letadla \texttt{Airplane}, který implementuje autonomní chování v~závislosti na dotykovém vstupu.}
\label{fig:classes-simulation}
\end{center}
\end{figure}
\section{Herní logika a průběh simulace}
Řídicí funkci z~hlediska herní logiky má herní obrazovka, tedy instance třídy \texttt{GameScreen}. Ve chvíli svého vytvoření dostává v~konstruktoru připravený herní svět (tedy \texttt{GameWorld}) a v~něm načtený závod. Obrazovka vytvoří objekt letadla (součástí definice světa je pouze jeho počáteční transformace), nastaví zpracování vstupů tak, aby se dotykové události dostaly na vstup letadla, nechá probíhat simulaci a současně scénu vykresluje pomocí třídy \texttt{GameRenderer}. Samotný pohyb letadla ve scéně probíhá velmi jednoduše, vyčíslováním vztahu pro rovnoměrný přímočarý pohyb, kde jako časový parametr slouží doba uplynulá mezi dvěma vykreslovanými snímky. Díky tomu se bude letadlo pohybovat stále stejně rychle bez ohledu na to, jaký je aktuální počet snímků za sekundu.
Vykreslovací třída je velmi jednoduchá, a proto se jí nebudu hlouběji zabývat. Z~externích souborů načítá shadery ve formátu GLSL, v~každém snímku nastavuje jejich parametry a jejich pomocí vykresluje jednotlivé entity daného světa. K~tomu by jí postačovaly instance \texttt{BaseWorld} a \texttt{BaseEntity}. Takový objekt je však ve hře pouze jediný, a sice tzv. \textit{skybox}, tedy krychle obalující celou scénu, na jejíž vnitřních stěnách je textura nebe. Všechny ostatní objekty jsou zapojeny do fyzikální simulace, kvůli čemuž musejí být v~hierarchii dědičnosti níže.
Z~hlediska vstupů je situace taková, že herní obrazovka nastaví jako jeden ze vstupních procesorů právě \texttt{PlaneInputProcessor}, který je svázán s~objektem letadla. Každý vstup zpracovaný tímto procesorem se převede na intenzitu sil způsobujících rotace v~jednotlivých osách. (Rotace kolem osy Y, anglicky pojmenovávaná \textit{yaw}, je předávána jen pro úplnost; vstup ji však přímo ovlivňovat nemůže.) V~okamžiku příchodu události (který probíhá asynchronně) se tedy do objektu letadla dostává informace o~tom, v~jaké poloze jsou které jeho ovládací prvky, neboť právě taková je sémantika působících sil. Při aktualizaci scény v~každém snímku pak letadlo tyto síly zohledňuje při svém dalším pohybu, upravením jeho směru a případné rotace samotného letadla. Je tedy možné říct, že letadlo ve scéně pohybuje autonomně.
Herní obrazovka před vykreslením každého snímku vyvolá aktualizaci scény, která se skládá právě z~aktualizace pozice letadla a dále z~provedení detekce kolizí. Výsledkem této aktualizace může být informace o~tom, že nastala kolize. V~takovou chvíli herní svět rozliší, zda jde o~kolizi žádoucí (s~brankou či kruhem) nebo nežádoucí (s~ostatními objekty ve scéně) a nastaví příslušné příznaky. Obrazovka pak pouze kontroluje, zda se po provedení aktualizace změnil stav světa, což znamená že letadlo buď prolétlo poslední brankou anebo se vybouralo. Na toto obrazovka reaguje zobrazením příslušných dialogů.
\vspace{\baselineskip}
\begin{figure}[ht!]
\begin{center}
\includegraphics[width=\textwidth]{fig/classes-input.eps}
\caption{Diagram tříd z~hlediska vstupních událostí. \texttt{PlaneInputProcessor} události zpracovává a vytváří hodnoty intenzit, které se ve třídě \texttt{Airplane} aplikují na ovládací prvky letadla.}
\label{fig:classes-input}
\end{center}
\end{figure}
Je zde také vhodné zmínit závislost výkonu aplikace na použitých kolizních tvarech, kterou jsem při vývoji vypozoroval. Dostatečně nízké výkonové nároky tak, aby hra běžela plynule na cílených zařízeních je nezbytností. Výkonovou náročnost hry lze snižovat optimalizací vykreslování scény anebo optimalizací probíhající fyzikální simulace. Narozdíl od způsobu vykreslování má úprava fyzikální simulace přímý vliv na hratelnost výsledné hry. Ve hře která používá fyzikální knihovnu pouze pro detekci kolizí není mnoho prostoru pro optimalizace. Prakticky jedinou proměnnou jsou zde kolizní tvary jednotlivých herních objektů, a právě jejich volba má dopad jak na výkon, tak i na hratelnost. Ve chvíli, kdy jsou kolizní tvary maximálně zjednodušené (koule, krychle, plochy) jsou sice požadavky na výkon nízké, ale kolize může být detekována (nebo naopak nedetekována) velmi nepřesně. Dokonce natolik, že si problému všimne i sám hráč a představuje pro něj chybu. Příkladem může být aproximace kolizního tvaru kruhu pouhým čtvercem. Zde tvar nepokryje některé krajní oblasti kruhu, které však hráč může cíleně využívat pro dokončení závodu v~lepším čase. Knihovna Bullet Physics podporuje kromě základních geometrických tvarů také tvary složené. Právě jejich pomocí jsem implementoval kolizní tvar kruhu, složený ze dvou čtverců, které jsou vzájemně pootočené. Optimálním řešením by jistě bylo použít čtverců mnohem více, případně pak přímo obecný polygon, v~takovém případě pak ale začínají nároky na výkon značně narůstat. Již při použití čtyř čtverců bylo možné pozorovat zpomalení hry (testováno na přístroji Samsung Galaxy Nexus, který se v~době psaní této práce řadí mezi zařízení vyšší třídy). Podobně se ukázalo vhodnější použít pro kolizní tvar letadla jednoduchý kvádr, namísto dvou kvádrů představujících trup a křídla. Oba zmiňované kolizní tvary jsou znázorněny na obrázku
\begin{figure}[ht!]
\begin{center}
\includegraphics[width=\textwidth]{fig/kolizni-tvary.jpg}
\caption{Kolizní tvary některých herních objektů. S~rostoucí složitostí strmě rostou i nároky na výkon. Proto je kruh reprezentován pouze dvěma pootočenými čtverci.}
\label{fig:kolizni-tvary}
\end{center}
\end{figure}
\section{Uživatelské rozhraní}
\label{sec:implementace-ui}
Z~hlediska uživatelského rozhraní lze hru rozdělit na dvě části -- hlavní menu a samotnou hru. Ty jsou implementovány v~již popisovaných třídách \texttt{MainMenu} a \texttt{GameScreen}. Vazby součástí uživatelského rozhraní jsou znázorněny na obrázku \ref{fig:ui-diagram}. Každá z~obrazovek obsahuje jednu instanci potomka třídy \texttt{BaseStage} (konkrétně \texttt{MainMenuStage} a \texttt{GameGuiStage}), kde jsou vytvářeny jednotlivé prvky rozhraní a definováno jejich rozložení. Využity jsou převážně předdefinované komponenty, jejichž vizuální podoba je upravená tak, jak bylo popsáno v~kapitole \ref{sec:libgdx-ui}. Kromě těchto dvou obrazovek hra využívá několik dialogů. Ty jsou ve své výchozí podobě ztvárněny použitím plovoucího okna. Tuto podobu jsem upravil tak, že dialogy okno nepoužívají (lépe řečeno je vykresleno průhlednou barvou) a namísto toho částečně ztmaví celou oblast obrazovky. Toto řešení působí méně technicistním dojmem a lépe zapadá do vizuální podoby celé aplikace.
\vspace{\baselineskip}
\begin{figure}[ht!]
\begin{center}
\includegraphics[width=\textwidth]{fig/ui-diagram.eps}
\caption{Jednotlivé obrazovky a dialogy v~aplikaci. Pro navigaci mezi nimi slouží prvky uživatelského rozhraní nebo volitelně systémová tlačítka zařízení.}
\label{fig:ui-diagram}
\end{center}
\end{figure}
Navigace v~menu je rozdělená do dvou úrovní. Na první úrovni lze vybírat mezi sadami závodů, na úrovni druhé pak mezi jednotlivými závody. Vizuálně výběr realizuje horizontálně posuvná oblast s~náhledy jednotlivých sad či závodů. Položka nejblíže středu obrazovky se považuje za aktuálně vybranou a kliknutím na její miniaturu nebo na tlačítko \textit{PLAY} se sada či závod vybere a spustí. Pro zřetelost vybrané položky se ostatní položky s~rostoucí vzdáleností od středu obrazovky zmenšují. Při posouvání se pak panel vždy zafixuje tak, aby byla aktivní položka právě ve středu obrazovky. Pro dosažení tohoto chování jsem rozšířil knihovní třídu \texttt{ScrollPane}, která podporuje pouze plynulý horizontální či vertikální posun obsahu v~panelu bez dalšího možného nastavení nebo funkcionality. Výsledné chování implementuje třída \texttt{MyScrollPane}. Z~panelu je také odstraněn klasický posuvník, který nahrazuje skupina ikon, jejichž počet je shodný s~počtem položek v~panelu. Ikona reprezentující aktuálně vybranou položku je pak vizuálně odlišená. Situaci ilustruje dříve uvedený obrázek \ref{fig:ui-design}, který současně zachycuje rozdělení obrazovky hlavního menu do jednotlivých komponent.
\vspace{\baselineskip}
V~kapitole \ref{sec:implementace-prototypu} jsem zvolil jako optimální variantu ovládání letadla takovou, kdy nevhodné místo prvotního dotyku může omezit manévrovací schopnosti letadla. Z~tohoto důvodu a současně proto, že je metoda ovládání zcela nová a neznámá, jsem vytvořil speciální dialog pro zaučení nového hráče. Zobrazí se při spuštení prvního závodu ve hře a namísto potvrzovacích tlačítek jsou vykresleny dva žluté kruhy (viz obrázek \ref{fig:ui-tutorial}). Hráč je instruován, aby se současně dotkl obou kruhů, teprve poté se závod zahájí. Kruhy jsou záměrně zarovnány svým středem na střed výšky obrazovky s~cílem naučit hráče tuto pozici prstů jako výchozí. Tento postup se osvědčil. Při testování se ukázalo, že v~drtivé většině případů hráčům stačí pro pochopení ovládání pouze informace zobrazená v~tomto dialogu. Někteří navrhovali, že by se dialog mohl zobrazit před každým závodem. Jiní by preferovali, kdyby se místa optimálního dotyku zobrazovala v~průběhu celého závodu ve chvílích, kdy se hráč obrazovky přestane dotýkat. Především druhý zmíněný námět by mohl pomoci hráčům rychleji se v~ovládání zorientovat například v~případech, že se ke hře vrátili po delší době. Ani jeden z~těchto námětů však v~aktuální verzi zahrnut není.
\begin{figure}[ht]
\begin{center}
\includegraphics[width=\textwidth]{fig/ingame-tutorial.jpg}
\caption{Dialog pro zaučení nového hráče, zobrazovaný u~prvního závodu ve hře. Pro spuštění závodu se musí hráč dotknout současně obou žlutých kruhů. Tím by měl pochopit, že k~ovládání letadla je potřeba vždy dvou prstů.}
\label{fig:ui-tutorial}
\end{center}
\end{figure}
\section{Vývojářské rozhraní}
\label{sec:vyvojarske-rozhrani}
Součástí hry je také speciální vývojářský režim. Ten je pro běžné hráče přirozeně nedostupný a v~publikované verzi je odstraněn. Slouží k~jednoduchým úpravám existujících závodů a k~vytváření závodů nových. Jeho rozhraní je spíše strohé, avšak účelné. Zachyceno je na obrázku \ref{fig:ui-dev}. Vývojářský režim implementuje obrazovka \texttt{GatePlacementScreen}, uživatelské rozhraní pak třída \texttt{OverlayStage}. Obrazovka načte objekty herního světa, podobně jako při spuštění závodu, s~tím rozdílem, že neprobíhá detekce kolizí ani vyhodnocování stavu závodu. Scéna neobsahuje ani letadlo a kamera se může volně pohybovat v~prostoru. K~tomu slouží tlačítka uživatelského rozhraní pro posun o~pevně danou vzdálenost, případně pak speciální vstupní procesor \texttt{FreeMoveInputProcessor} umožňující plynulé rozhlížení a průlet scénou. Je možné přidávat a odebírat jednotlivé kruhy a branky závodu a také nastavit počáteční transformaci letadla.
Z~důvodu zachování přenositelnosti mezi platformami bohužel nelze závod přímo uložit a ihned jej hrát. Aktuálně se závod vypíše do konzole vývojového prostředí jako řetězec ve formátu JSON. Ten je nutné ručně uložit do souboru v~adresáři s~ostatními datovými soubory aplikace. Toto chování vzniká proto, že v~okamžiku sestavování aplikace jsou veškeré její soubory zabaleny do jediného balíčku, který se dále zpřístupňován pouze pro čtení. Alternativně by bylo možné ukládat vytvářené závody do uživatelské paměti zařízení, ovšem tyto by pak byly volně přístupné a upravitelné, a tudíž by nebylo možné zaručit jejich konzistenci. Časy jednotlivých závodů jsou vkládány také ručně, na základě několika zkušebních letů. V~budoucnu budou časy upravovány na základě dat získaných anonymním sběrem statistik.
\begin{figure}[ht]
\begin{center}
\includegraphics[width=\textwidth]{fig/ingame-dev.jpg}
\caption{Rozhraní pro tvorbu a úpravu závodů. Slouží pouze pro vývoj a hráči do něj přístup nemají. Ve scéně se lze pohybovat skokově (tlačítky) nebo volně pomocí dotykového vstupu.}
\label{fig:ui-dev}
\end{center}
\end{figure}
\section{Zveřejnění hry a zpětná vazba}
\label{sec:zverejneni}
Ve chvíli, kdy se hra dostala do finální podoby, bylo možné ji začít veřejně distribuovat. Současně jsem pro dosud bezejmennou hru zvolil název Air Racer. Do té doby jsem testování prováděl osobními schůzkami v~různých stádiích vývoje, kde bylo možné okamžitě reagovat na chyby a jiné nedostatky. Tento přístup současně nabízel mnohem lepší zpětnou vazbu od hráčů, kterou by bylo později obtížné získat. Dalším přínosem bylo také přímé testování hry na různých zařízeních. Zde žádné problémy nenastaly a hra bez problémů běžela na všech zařízeních, se kterými deklarovala kompatibilitu. Hodnotné reakce přinesly také pravidelné konzultace ve firmě CUKETA, s.\,r.\,o.\,, která se herním vývojem profesionálně zabývá.
Posledním krokem před zveřejněním bylo zapojení statistik. Tuto službu poskytuje přímo firma Google formou speciální knihovny. Původně byla navržená pro sledování klasických aplikací, které jsou založeny na výchozích komponentách platformy, jako jsou Aktivity, dialogy a další. Díky poslední verzi jejího rozhraní je však možné sledované události spouštět manuálně, a tím ukládat statistiky i pro hru, která má z~pohledu sledující knihovny jen jedinou obrazovku.
\subsection{Veřejné testování a průzkum}
\label{sec:verejne-testovani}
První veřejnou verzi hry jsem vystavil na internet ve formě samostatného binárního souboru (tedy bez možnosti automatické aktualizace) a nasdílel na některých sociálních sítích. Současně jsem vytvořil dotazník, o~jehož vyplnění jsem hráče požádal. Plné znění dotazníku se nachází v~příloze \ref{ext:pruzkum} této práce. Struktura dotazníku je inspirována metodou \textit{Likert Scale} \cite{likert}. Jejím cílem je nepokládat přímé otázky, ale představovat tvrzení a měřit, jak silně se s~nimi hodnotící ztotožňuje. Tím by mělo být hodnocení jednodušší a spontánnější. Současně se omezuje nežádoucí kreativita hodnotícího a zjednodušuje se i vyhodnocení takového dotazníku. Nejčastěji se používá stupnice o~3, 5 nebo 7 hodnotách. Důležitý je vždy lichý počet možností, umožňující hodnotícímu zaujmout neutrální postoj. Mohou nastat výjimky, kdy je žádoucí použít sudý počet možností, a tím si odpověď vynutit. Tento přístup ale pro vyvíjenou hru vhodný není. Další z~doporučení při tvorbě dotazníku metodou Lickert Scale je nezaujatost tvrzení. Ta by měla být ideálně předkládána tak, aby nebylo zřejmé, která z~možností je správná. Hodnotící hráč pak mnohem méně podléhá společenským a jiným tlakům.
\vspace{\baselineskip}
Dotazník vyplnilo 67 hráčů a z~jeho výsledků dotazníku vyplývá, že první dojem z~aplikace byl pozitivní, hlavní menu hráčům připadá přehledné a je dobře ovladatelné. To potvrzuje, že implementace vlastního chování posuvného panelu v~menu byla úspěšná. Stejně tak se osvědčila ikona ozubeného kola reprezentující dialog s~nastavením. Zajímavostí je, že z~hráčů odpovídajících na dotazník tento dialog otevřela asi jen čtvrtina z~nich (dle údajů ze statistik). Přesto většina odpověděla, že by uměla zapnout či vypnout zvuky ve hře. Příčinou může být buď nedbalost při vyplňování dotazníku anebo fakt, že jde o~silně zažitý prvek uživatelského rozhraní ve hrách, od kterého hráči očekávají chování, aniž by jej ověřovali.
Prokázala se také vhodnost logické stavby menu. Hodnotící ve hře rozpoznali, že má menu více úrovní a že pro zpřístupnění uzamčeného závodu bude pravděpodobně nutné splnit závod předchozí. V~případě dotazu na počet položek závodů ve hře však byly odpovědi sporné. To mohlo být zapříčiněno nevhodně formulovaným tvrzením (\textit{Mám představu o~tom, kolik je ve hře levelů}), které vyžaduje zpřístupnění všech sad závodů. Sady jsou na sebe navázány stejně jako jednotlivé závody a do odemčení do nich nelze ani nahlédnout. Hráč který by chtěl na toto tvrzení korektně odpovědět by tedy musel nejdříve všechny závody zpřístupnit, což však podle statistik neudělal nikdo. Dalším sporným tvrzením byla možnost spuštění závodu. Zde šlo čistě o~experiment -- závod lze spustit jak tlačítkem PLAY, tak i kliknutím na jeho náhled. Téměř třetina odpovídajících využila kliknutí na náhled. Pravděpodobně se tak chovali proto, že jde opět o~chování které lze od posuvného panelu obecně očekávat.
\vspace{\baselineskip}
Další část dotazníku se věnovala samotnému průběhu závodu a chování letadla. Téměř $80\%$ hráčů odpovědělo, že jsou schopní letadlo bez předchozího vysvětlení ovládat (myšleno jiného vysvětlení než dřívě zmiňovaného dialogu před prvním závodem). To potvrzuje intuitivnost navrhovaného ovládání. Většina hráčů považuje ovládání za dostatečně přesné, přičemž lze s~letadlem stále provádět ostré zatáčky. Stejně tak hráči odpovídají, že vědí, jak pohyby jejich prstů ovlivní letadlo. Tím opět potvrzují dobrý návrh ovládání. Současně však hráči nebyli schopni jasně identifikovat, zda chování letadla odpovídá letu skutečného letadla. S~tím souvisí i tvrzení ohledně předvídatelnosti chování letadla, kde jsou odpovědi také vesměs neprůkazné. Již při návrhu hry (v~kapitole \ref{sec:zanrove-zarazeni}) jsem deklaroval, že cílem hry není maximálně věrné napodobení letu skutečného letadla, nýbrž navržení chování tak, aby vyústilo v~co nejvíce zábavnou a obecným hráčům pochopitelnou hru. Z~tohoto pohledu nejsou zmíněné reakce ani pozitivní ani negativní. Bylo by možné říct, že se chování pohybuje na hraně s~lehkým přesahem do nerealistična, což aktuálně považuji za vhodný kompromis. Posledním tvrzením na téma ovladatelnosti byl dotaz na ovládání směrového kormidla. Zde se hráči rozdělili na téměř přesné poloviny, kdy jedna strana preferuje ovládání přímé a druhá inverzní. Tento jev se projevil ještě před veřejným testování a hra již v~době vyplňování dotazníku obsahovala možnost převrácení svislé ovládací osy. Tato volba je u~leteckých her velmi často používaná a měla by uspokojit obě skupiny hráčů. Výchozí chování je nastaveno na inverzní, kde posun prstů ke spodní hraně obrazovky zvedá letadlo nahoru. Toto chování totiž více odpovídá chování kniplu ve skutečném letadle (případně hernímu joysticku).
Poslední část dotazníku se věnovala skladbě závodů a prostředí ve kterém se odehrávají. Letadlo podle většiny hráčů letí spíše pomalu, stejně jako obtížnost závodů hodnotí spíše jako nízkou. Je však vhodné dodat, že statistiky v~období vyplňování dotazníku ukazovaly, že do pokročilejších závodů postoupilo jen malé množství hráčů. Toto tvrzení tak hodnotí spíše subjektivní obtížnost, která však může být zajímavým doplňkem objektivní obtížnosti, kterou lze získat ze statistik na základě četnosti úspěšných dokončení jednotlivých závodů. Drtivé většině odpovídajících hráčů se pak podařilo dokončit alespoň jeden závod a alespoň jednou dokončit závod se ziskem tří hvězdiček. Hráči kladně ohodnotili také navigační šipku, která ukazuje směr k~následujícímu kruhu či brance. Z~dodatečných odpovědí (volitelný vzkaz na konci dotazníku) vyplývá, že by ocenili i informaci o~vzdálenosti následujícího cíle, případně náhled na rozložení branek v~celém závodě. V~závěru bylo představeno několik tvrzení ohledně scenerie závodu. Z~odpovědí lze vyčíst, že hráči prostředí spíše nevnímají, současně by však uvítali kdyby hra obsahovala více dekorativních objektů. Tyto reakce lze přičíst subjektivní prázdnotě prostředí závodu. Při vývoji se ukázalo jako velmi efektivní nechat závod probíhat nad mořem v~teoreticky neomezeném prostoru. Hráči ale zřejmě vnímají jistý nedostatek ukotvení scenerie. Vhodným řešením by mohlo být omezení herního prostoru tak, že by závod probíhal například podél skalního útesu nebo města. Vzhledem k~tématice by pak bylo možné doplnit divácké tribuny, reklamní plochy a další objekty. To může být tématem dalšího rozšiřování hry.
\vspace{\baselineskip}
V~rámci udržení hráčů jsem do veřejně testované verze zabudoval omezení, které hru dovolí hrát pouze dva týdny. Poté bude po spuštění nabízet pouze možnost přechodu do obchodu Google Play, kde bude možné stáhnout finální verzi hry.
\subsection{Zveřejnění na Google Play}
Verze publikovaná na internetovém obchodě Google Play se od veřejně testované prakticky neliší. Opravil jsem několik drobných nedostatků, které se projevovaly pouze na úzkém okruhu zařízení a samozřejmě jsem také odstranil časové omezení. V~souvislosti s~publikováním bylo nutné vytvořit několik variant propagačních materiálů. Jelikož jsem veškeré grafické podklady vytvářel já sám, může být jejich kvalita sporná. Zpětně bych pravděpodobně zvážil spolupráci s~profesionálním grafikem. Přesto se volba jednoduché ikony s~vyobrazeným letadlem ukázala jako vhodná. Obecný vliv prezentačních materiálů na potenciální uživatele však nelze nijak měřit.
Jak již bylo popsáno v~kapitole \ref{sec:distribuce}, pro publikování v~obchodě Google Play je nutné vlastnit vývojářský účet. Vytvoření účtu, stejně jako publikování aplikace pak probíhá na počkání. Není tedy nutné počítat se zpožděním při vydávání aplikace tak, jak je tomu u~jiných platforem. Publikovaná aplikace musí být podepsaná soukromým klíčem vývojáře a stejným klíčem pak musí být podepsány všechny další aktualizace. V~případě ztráty tohoto klíče již nelze binární podobu aplikace nahrané v~obchodě nijak upravovat.
\vspace{\baselineskip}
Po vydání aplikace jsem ji opět propagoval na sociálních sítích, včetně několika komunit specializovaných na hry a platformu Android. Současně jsem žádal hráče, aby hru v~obchodě ohodnotili (dle vlastní vůle), čímž ji učiní teoreticky lákavější pro další potenciální hráče. Hodnocení probíhá udělením jedné až pěti hvězdiček a přidání případného komentáře. Počet hodnotících hráčů je však velmi nízký, dle současných statistik vloží hodnocení méně než jeden hráč z~500, komentář pak vloží asi jeden z~tisíce. Aktuálně hru hodnotilo 87 hráčů, přičemž 61 zvolilo nejvyšší hodnocení, 13 pak hodnocení nejnižší. Při takto nízkém počtu však nemá hodnocení ani většina komentářů téměř žádnou vypovídací hodnotu. Hodnocení hráčů jsem se snažil podpořit herním dialogem, který se jednorázově spustí po splnění všech závodů z~první sady.
I~přes nízkou účast hráčů v~hodnocení se hra v~obchodě prosadila a necelý měsíc po vydání se dostala na pozici deváté nejlepší závodní hry. Graf vývoje pozice v~jednotlivých dnech je na obrázku \ref{fig:pozice-play}. Celkový počet instalací přesahuje 70000, přičemž denní trend nových uživatelů v~současnosti spíše stagnuje.
\begin{figure}[ht]
\begin{center}
\includegraphics[width=\textwidth]{fig/pozice-play.pdf}
\caption{Vývoj pozice hry v~Google Play, v~kategorii závodních her. Po necelém měsíci od vydání je hra na 9. pozici.}
\label{fig:pozice-play}
\end{center}
\end{figure}
Součástí vývojářského rozhraní obchodu Google Play je i sledování vzniklých výjimek. Těch se objevilo vzhledem k~množství instalací jen minimum. Šlo o~nepředvídatelné chyby, které vznikaly jen na některých jednotlivých modelech nebo konkrétních zařízeních. Přestože bývají zaznamenané výjimky běžně svázány s~popisem zařízení na kterém vznikly, v~některých případech tato informace chyběla. Lze předpokládat, že jde o~upravená sestavení platformy Android. Mimo toto jsem narazil také na chybu vyskytující se v~různém množství na celé modelové řadě. Jde o~chybu přímo v~platformě Android, která se však vyskytuje jen u~konkrétního sestavení a verze. V~tomto případě bylo možné přestat těmto typům zařízení hru nabízet a v~obchodě ji označit jako nekompatibilní. V~době před omezením podpory měla tato zařízení téměř 10\% podíl na instalacích. Přestože jde o~nezanedbatelný počet hráčů, jiné řešení dostupné nebylo. Chyba byla nahlášena a v~dalších verzích platformy opravena, nicméně výrobce aktualizaci svých přístrojů nenabídl a jiné řešení než aktualizace platformy pro tento problém známé není.
\section{Možnosti dalšího vývoje}
Přestože jsem v~dřívějších kapitolách často naznačil další možný vývoj, popíši jej nyní detailně v~samostatné kapitole. Za položku s~nejvyšší prioritou aktuálně považuji vizuální stránku hry. Tu by sice bylo možné zařadit v~rámci her nabízených na Google Play do průměru, nicméně je to vlastnost, která produkt prodává. Úplně nejdříve potenciální zákazník vidí ikonu hry, následně propagační obrázky a teprve poté si může přečíst popis a hru případně stáhnout. Z~reakcí na hru lze vyčíst, že s~navrženým způsobem ovládání je většina hráčů spokojená, nicméně jde právě o~tu skupinu hráčů, které neodradil žádný ze zmiňovaných kroků. Lze předpokládat, že existuje početná skupina návštěvníků internetového obchodu, které hra nezaujala právě kvůli grafickému zpracování. Konverzi těchto potenciálních hráčů na hráče skutečné by mělo pomoci právě další zlepšování propagačních materiálů. Jejich součástí je také popis hry v~obchodě, který by bylo vhodné přeložit do více jazyků (aktuálně je pouze anglicky). Stejně tak by bylo možné přeložit všechny texty ve hře, jak také napovídaly některé reakce hráčů.
Součástí vizuální stránky hry je také herní prostředí. V~tomto případě by úpravy cílily převážně na již existující hráče. Ti se v~dotazníku svojí většinou shodli, že by hře prospělo více dekorativních objektů. Spolu s~tím, jak by v~rámci rozšiřování přibývaly ve hře další závody, by mohly začít působit příliš jednotvárně. Herní objekty jako tribuny, reklamní plochy a další by mohly takovou jednotvárnost narušit a současně (ovšem ne nezbytně) zvýšit obtížnost závodů, například jejich vhodným umístěním do trasy letu. Ve větším měřítku pak může jít o~úplnou změnu scenerie závodů. Mohly by se odehrávat například v~poušti, ve městě mezi domy anebo téměř v~libovolné krajině. Bylo by zde nutné dbát na výkonové nároky, které jsou v~současnosti velmi nízké, a případná rozšíření zavádět tak, aby nebylo nutné omezovat podporu některých starších a méně výkonných zařízení. Z~méně náročných úprav pak zmíním například efekty odlesku a rozptylu světla na objektivu kamery (tzv. efekt \textit{lens flare}) nebo viditelné linky vzduchu či páry u~konců křídel letadla, které by kromě vizuálního efektu mohly hráči pomoci lépe si představit, jakým směrem se letadlo aktuálně pohybuje.
\vspace{\baselineskip}
Z~hlediska ovládání zopakuji již zméněnou možnost ovládání směrového kormidla. Jelikož by pravděpodobně většina hráčů tuto změnu nepřivítala, bylo by možné takové rozšíření nabízet volitelně. Bylo by však velmi zajímavé zjišťovat další možnosti navržené metody ovládání. Důležitým aspektem u~úprav tohoto typu v~již publikované hře by bylo zachování zpětné kompatibility herních výsledků. Změna ovládání, stejně jako změna parametrů letadla, může zpřístupnit nové a efektivnější letové trasy anebo naopak ztížit dokončení již existujících závodů. Případ, kdy hráč dostává možnost dokončit závod rychleji, je přirozeně lepší variantou. I~ten by však mohl narušit možnost vzájemného srovnávání výkonů hráčů, například ve sledovaných statistikách.
Podle některých ohlasů by hráči uvítali také lepši přehled o~konkrétních časech závodů, například ve smyslu možnosti zobrazení časů dřívějších letů. Pokud bych tuto myšlenku měl rozvést dál, pravděpodobně bych navrhl možnost zobrazení tzv. \textit{ducha}. Šlo by o~částečně průhledné letadlo, které by představovalo hráčův dřívější let. Díky tomu by bylo zřejmé, kde se hráč od dřívější trasy odchyluje a namísto přímého srovnání časů by hra nabízela srovnání vizuální.
S~možností vzájemného srovnání souvisí i další možnost rožšíření -- online žebříčky a závody pro více hráčů. Zmíněnou možnost zobrazení duchů by bylo možné rozšířit o~sdílení těchto záznamů závodů mezi hráči. Ti by pak vedle předdefinovaných časů závodů mohli překonávat globální rekordy skutčných hráčů. Toto rozšíření by ovšem znamenalo řádově více práce, neboť by bylo nutné zajistit server ukládající tato data a jejich synchronizaci. Nelze také opomenout nutnost zabezpečení takových přenosů, aby nebylo možné výsledky závodů podvrhnout.
Jinou cestou rozšiřování hry pro více hráčů by mohlo být lokální hraní s~využitím technologií jako Bluetooth nebo Wi-Fi. Zde by nároky na synchronizaci a bezpečnost spojení nebyly tak vysoké a současně by bylo možné nabídnout společné hraní v~reálném čase. To by umožnilo také zavedení nových přístupů k~závodům, které by vyžadovaly například spolupráci obou hráčů nebo naopak jejich vzájemné soupeření, v~extrémním případě i formou vzdušných soubojů (které hráči v~dotazníku také několikrát zmínili).
Pro úplnost zmíním také přidávání dalších závodů, přestože jde o~rozšíření naprosto zřejmé. Tímto způsobem by bylo možné udržovat v~aktivitě existující hráče. Opět zde ale hrozí jistá monotónnost a bylo by vhodné tato rozšíření kombinovat právě se zmíněnými grafickými doplňky. Zajímavým, byť opět realizačně náročným rozšířením, by byla možnost vytváření vlastních závodů samotnými hráči. Základní prvky pro tuto funkcionalitu už hra obsahuje, popsány byly v~kapitole \ref{sec:vyvojarske-rozhrani}. Po nezbytných úpravách by se tato dosud vývojářská obrazovka mohla stát plnohodnotnou součástí hry. Hráči by mohli vytvořené závody sdílet a soupeřit o~nejlepší časy, ať už globálně či lokálně. Opět by ale platily dříve uvedené předpoklady o~zabezpečení takovéto funkcionality proti zneužití.
\vspace{\baselineskip}
Při úvahách o~dalším rozšiřování hry je vhodné uvážit také ekonomickou stránku věci. Zatímco některé z~navrhovaných úprav jsou jednoduché, jiné by vyžadovaly například provoz webového serveru. V~takovou chvíli by nutně vznikaly náklady, které by vzhledem k~počtu hráčů zřejmě nebyly nízké. Přestože náklady na dosavadní vývoj hry představovaly pouze strávený čas, i tento představuje omezené prostředky, které by měly vést k~zisku (byť ne nezbytně materiálnímu). U~her pro mobilní zařízení bývá obecným zvykem vydat hru ve dvou variantách -- bezplatné a zpoplatněné. Bezplatná verze hry může sloužit jako demo a současně často obsahuje omezenou nabídku funkcí. V~tomto případě by bylo možné vydat hru obohacenou o~některé z~výše uvedených rozšíření ve zpoplatněné verzi. Přestože by si placenou verzi zakoupil zřejmě jen zlomek stávajících hráčů, bylo by možné uvažovat o~alespoň částečnému návratu investic do vývoje či provozu hry. Současně by ale vznikaly další problémy, kterými bezplatná verze netrpí, jako například pirátství. To může u~žádaných her může dosahovat až k~90\% nelegálně získaných kopií \cite{pirati}. Další výzvou by tedy bylo nastavení vhodného obchodního modelu tak, aby se tyto ztráty minimalizovaly. Příkladem takového modelu může být nákup přímo v~aplikaci (tzv. \textit{in-app payments} nebo \textit{in-app transactions}), kdy se samotná aplikace se základní funkcionalitou distribuuje bezplatně a dokupování dalšího obsahu a možností probíhá přímo v~aplikaci.
\chapter{Závěr}
\hyphenation{Libgdx}
\hyphenation{Google}
Tento text shrnul možnosti vývoje 3D her na platformě Android s~využitím knihovny Libgdx. Dále se věnoval popisu letu skutečného letadla a z~něj vycházející zjednodušené simulace letu používané ve hrách. Srovnává také existující metody ovládání leteckých her a navrhuje novou metodu, která závažnými nedostatky oproti ostatním netrpí. Navrhovaná metoda je založená na dotykovém vstupu za použití dvou prstů a nabízí velmi rychlé a přesné ovládání, čímž posiluje dojem, že má hráč letadlo plně pod kontrolou.
Dále tato práce popsala návrh a implementaci hry, která toto ovládání využívá. Jde o~hru, kde hráč s~akrobatickým letadlem prolétává brankami a snaží se doletět do cíle v~co nejkratším čase. Hra je navržená tak, aby co nejvíce těžila z~představené ovládací metody, podpořila tak její myšlenku a v~ideálním případě i další rozšíření na poli mobilních leteckých her. V~průběhu vývoje vznikl prototyp hry, který ověřil základní principy ovládání a umožnil srovnání s~již používanými metodami. Následně vznikla plnohodnotná implementace hry s~využitím právě knihovny Libgdx.
Hra byla průběžně konzultována s~firmou CUKETA, s.\,r.\,o.\,, která se vývojem her profesionálně zabývá. Před zveřejněním hry proběhl také dotazníkový průzkum, který ukázal, že hráči ovládací metodu i samotnou hru hodnotí velmi pozitivně.
Po získání a zpracování zpětné vazby byla hra publikována na internetový obchod Google Play pod názvem Air Racer\footnote{\url{https://play.google.com/store/apps/details?id=cz.davidsabata.airracer}}. Tam se v~době dokončování této práce umístila na pozici 9. nejlepší bezplatné aplikace v~kategorii závodních her a nainstalovalo si ji více než 70000 hráčů. Hra je kompatibilní s~více než 94\% mobilních zařízení platformy Android (dle dříve uvedených statistik).