Ohm-Management - Projektarbeit B-ME
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Gruppenbericht.tex 60KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813
  1. % !TeX spellcheck = de_DE
  2. % ---
  3. % Preamble
  4. % ---
  5. \documentclass[12pt]{report}
  6. \usepackage{defaultPreamble}
  7. \bibfilename{Gruppenbericht.bib}
  8. %
  9. % ---
  10. % Document Body
  11. % ---
  12. \begin{document}
  13. \pagenumbering{arabic}
  14. %
  15. % Deckblatt
  16. \setcounter{page}{0} % setze seitenzähler auf 0 -> alle Seitenzahlen <1 werden nicht angezeigt
  17. {\centering
  18. \clear % Select ClearSans for the title page
  19. \vspace*{5\baselineskip}
  20. %
  21. {\large Technische Hochschule}\\
  22. \vspace{0.3ex}
  23. {\large Georg-Simon Ohm Nürnberg}\\
  24. \vspace{0.3ex}
  25. {\large Fakultät Elektrotechnik Feinwerktechnik Informationstechnik}\\
  26. \vspace{2.5ex}
  27. %
  28. {\large Projekt OHM News}\\
  29. \vspace{3.5ex}
  30. %
  31. {\Large\b{Projektstudienarbeit}}\\
  32. \vspace{0.3ex}
  33. {\Large\b{B-ME 5}}\\
  34. \vspace{4.5ex}
  35. {\Large\b{OHM News - See only what matters}}\\
  36. \vspace{3.5ex}
  37. %
  38. {\large Wintersemester 2018/2019}
  39. } % Need 1 empty line above for centering
  40. %
  41. \newpage % Seitenumbruch nach Titelblatt
  42. \paragraph{\Large Projekt-Roadmap}$~~$\\
  43. %
  44. \begin{table}[h]
  45. \centering%\small
  46. \begin{tabularx}{\textwidth}{|X|X|X|}
  47. \hline
  48. \rule{0pt}{14pt}\b{Tätigkeit} & \b{Dokument} & \b{Beteiligung} \\
  49. \hline
  50. \rule{0pt}{14pt}Konzeption & Projektidee & Alle \\
  51. Organisation und Pflege & Trello & Erik \\
  52. Organisation und Pflege & Bitrix24 & Erik \\
  53. Protokollierung der Treffen & Handschriftlich & Erik, Vivianne \\
  54. Transkription der Protokolle & LaTeX Protokolle & Erik \\
  55. Organisation und Pflege & Wiki & Vivianne \\
  56. Erstformulierung & Pflichtenheft & Alle \\
  57. Erste Entwürfe des App UI & Scribbles, Skizzen & Alle \\
  58. Namensüberlegungen & Notizen zur Namensfindung & Alle \\
  59. Corporate Design & Styleguide & Xenia \\
  60. Entwicklung & Erstellung der Mockups & Xenia, Edwina \\
  61. Entwicklung & Frontenddesign mit Bootstrap & Xenia, Edwina \\
  62. Entwicklung & Single-Page mit Vue.js & Xenia, Edwina \\
  63. Entwicklung & ServiceWorker & Erik \\
  64. Entwicklung & Manifest mit Json & Erik \\
  65. Entwicklung & Meta-Tags mit HTML & Erik \\
  66. Entwicklung & Erstellung der API-Schnittstelle & Senta, Edwina, Vivianne, \mbox{Xenia} \\
  67. Dokumentation RESTful API & Digital mit Swagger.io & Erik \\
  68. Entwicklung & Server mit Node.js/Express.js & Vivianne \\
  69. Entwicklung & Datenbank mit MongoDB & Senta \\
  70. Entwicklung & DB-Anbindung mit Mongoose & Senta \\
  71. Designentwürfe Plakat & Plakatgestaltung & Vivianne \\
  72. \rule{0pt}{14pt}Design und Ausformulierung & Projektpräsentation & Edwina, Vivianne \\
  73. \hline
  74. \end{tabularx}
  75. \caption{Projekt-Roadmap Tabelle}
  76. \end{table}
  77. %
  78. % Inhaltsverzeichnis
  79. \newpage
  80. \tableofcontents
  81. %
  82. % Bericht
  83. \newpage % Seitenumbruch nach Titelblatt
  84. \section*{Einleitung}
  85. \phantomsection\addcontentsline{toc}{section}{Einleitung}
  86. {\color{red} \bfseries \LARGE ToDo}
  87. %
  88. \section*{Vorwort zur Projektidee}
  89. \phantomsection\addcontentsline{toc}{section}{Vorwort zur Projektidee}
  90. Die erste Konzeptversion zu der Projektidee von dem Projekt ''Ohm Management App'', dass im Verlauf der ersten Projektphase in ''OHM-News - See only what matters'' umbenannt wurde, wurde von Erik Römmelt erstellt. Nach gemeinsamer Überarbeitung durch Prof.Dr. Matthias Hopf, dem zukünftig betreuenden Professor und Erik Römmelt wurde die zu Projektbeginn gegebene Version des Projektkonzeptes ausformuliert.
  91. %
  92. \section*{Projektidee}
  93. \phantomsection\addcontentsline{toc}{section}{Projektidee}
  94. Die Projektidee basiert darauf, dass eine plattformunabhängige App für Studierende an der TH Nürnberg entwickelt werden soll, die Kurznachrichten filtert und in einem Nachrichtenfeed anzeigt. Das Filtern funktioniert durch Tags. Es können beim Nachrichten erstellen mehrere, jedoch mindestens eine eingegeben werden. Diese werden bei einer Überschneidung bei der Suche durch eine logische UND-Verknüpfung eingeschränkt. Angezeigt werden die Nachrichten abhängig von ihrer Priorität und Aktualität. \\
  95. Dazu können in einer Dateiablage von Professoren und Befugten wichtige Links oder andere Dateien gesammelt werden oder als Referenz darauf verwiesen werden. Damit es den Status ''Befugte'' gibt, muss es zudem eine Rollenverteilung geben, wem welche Rechte zu stehen oder beispielsweise wer nur Nachrichten empfangen kann.
  96. Zudem ist eine Speicherung einzelner Nachrichten aus dem Feed möglich, welche im Nachhinein in den Bookmarks oder auch Lesezeichen einsehbar sind.\\
  97. Des Weiteren können Push-Benachrichtigungen angezeigt werden, sodass das Risiko, wichtige Nachrichten zu übersehen, minimiert wird.
  98. Zuletzt gibt es noch ein Nutzerprofil. Hier können Daten zur Person eingetragen und angezeigt werden wie zum Beispiel das Profilbild, deren Status an der technischen Hochschule oder die Matrikelnummer.
  99. Für den Optimalfall einer Übernahme der App von der Hochschule soll modulorientiert und wartungsaufwandsarm gearbeitet werden, um die spätere Verwaltung zu erleichtern. Auch wird Wert gelegt auf eine nutzerfreundliche und möglichst intuitive Bedienoberfläche.
  100. %
  101. \subsection*{Technische Umsetzung}
  102. \phantomsection\addcontentsline{toc}{subsection}{Technische Umsetzung}
  103. Zur Umsetzung des Projektes fiel die Zielsetzung auf die Entwicklung einer progressiven Web-App. Aufgrund ihrer Plattformunabhängigkeit und Kompatibilität mit den TH-Servern (VPN und Eduroam) und laut Erfahrungen von Professor Dr. Hopf stellt dies eine gute Lösung und sichere Wahl dar. Ebenfalls, was die Datenschutzvorkehrungen der Hochschule betrifft. \\
  104. Es werden die Codesprachen HTML, CSS und JavaScript verwendet und als Frameworks Vue.js, Bootstrap und node.js. \\
  105. Als Datenbank dient MongoDB und das zugehörige Framework mongoose.
  106. %
  107. \section*{Projektorganisation}
  108. \phantomsection\addcontentsline{toc}{section}{Projektorganisation}
  109. Wöchentlich fand ein Treffen, zum Austausch und zur Besprechung zurückliegender und anstehender Aufgaben, statt. An diesem Treffen nahmen in der Regel alle Mitglieder dieser Projektgruppe und der Projektbetreuer Prof.Dr. Matthias Hopf teil. Konnten Teammitglieder, gleich welcher Gründe, an einem Treffen nicht teilnehmen, so haben alle sich unverzüglich über Absprachen und den neu anstehenden Aufgaben selbstständig erkundigt.
  110. %
  111. \subsection*{Trello}
  112. \phantomsection\addcontentsline{toc}{subsection}{Trello}
  113. Zu Beginn des Projekts wurde entschieden, dass zur Verbesserung des Arbeitsablaufs ein Organisationstool hilfreich wäre. So fiel ist Trello in Gespräch gekommen.\\
  114. Trello ist ein graphisch sehr gut aufbereitetes Projektmanagement-Tool in Form von Kanban-Boards. In der kostenfreien Version, ist es möglich ''unbegrenzt [viele] Boards, Listen, Karten, Mitglieder, Checklisten und Anhänge'' hinzuzufügen und zu nutzen\cite{trello:price}. Zur typografischen Gestaltung der Karten, um zum Beispiel eine Liste einzufügen ist die Nutzung von einigen Markdown-Markup Elementen möglich.\\
  115. Dem Projekt zugute kommend ist die sehr einfache Bedienbarkeit, Plattform unabhängige Verfügbarkeit, sowie die gute Übersichtlichkeit der Boards. Nachteilig ist, dass es nur sehr eingeschränkt die Zeitkomponente abbilden kann. Dies ist nur über die sogenannten ''Plugins'' möglich, wobei in der kostenfreien Version nur ein Plugin je Board zugleich genutzt werden kann. So kann einem Board entweder die Funktion ''Gantt-Diagramme nutzen'' oder ''Storypoints den Karten hinzufügen'' verwendet werden.\\
  116. Diese Einschränkung führte zur der Entscheidung gegen die Nutzung von Trello im Team.
  117. %
  118. \subsection*{Bitrix24}
  119. \phantomsection\addcontentsline{toc}{subsection}{Bitrix24}
  120. Nach dem Ausschluss von Trello begann die Umsicht nach einem anderen Tool, dass kostenfrei mit einer Gruppe von fünf Personen genutzt werden kann. Und es sollte die Funktionen Gantt-Diagramme erstellen, Aufgaben an Mitglieder verteilen und den Gewichtungs-Punkte oder ''Storypoints'' an Aufgaben zuweisen erfüllen.\\
  121. Bitrix24 hat diese Anforderungen erfüllt, war jedoch bereits auf den ersten Blick deutlich unübersichtlicher und komplexer in der Handhabung. Da es sich bei Bitrix24 in erster Linie um ein umfangreiches CRM-System handelt, mit vielen weiteren Funktionen wie ''Aufgabenverwaltung für Gruppen'', ''Projektplanung und -management'' und ''Team-Chat''\cite{bitrix:feature}.\\
  122. Auch nach der Deaktivierung und Ausblendung sämtlicher Funktionen, die die Nutzung von Bitrix24 komplizierter machten, blieb die Benutzeroberfläche noch unübersichtlich. Was die Umsetzung der Aufgaben als Kanban-Karten mit Gewichtungs-Punkten und der zeitlichen Darstellung der Aufgaben in einem Gantt-Diagramm betrifft, so war dies sehr angenehm und vollends zufriedenstellend möglich.\\
  123. So wurden zahlreiche organisatorische Fristen, sowie Aufgaben organisatorischer und entwicklungs-technischer Natur in Bitrix24 erfasst und anschließend in einem wöchentlichen Treffen vorgestellt. Im Weiteren wurde die Nutzung und Pflege des Tools an alle Mitglieder übertragen.\\
  124. Es stellte sich am Ende des ersten Projektabschnittes heraus, dass das Tool Bitrix24 aufgrund der komplexen und unübersichtliche Benutzeroberfläche kaum genutzt wurde.
  125. %
  126. \section*{Entwicklerblog}
  127. \phantomsection\addcontentsline{toc}{section}{Entwicklerblog}
  128. Die Wiki Blogeinträge in dem hochschulinternen github Repository dienen dazu, die Projektfortschritte transparent zu gestalten und eine kurze prägnante Zusammenfassung für Professoren und Studierende zu gewährleisten. Einzusehen sind die Beiträge auf der git efi Seite \cite{gitefi}.
  129. %
  130. \subsection*{Protokollierung}
  131. \phantomsection\addcontentsline{toc}{subsection}{Protokollierung}
  132. Für die Protokollierung wurden in regelmäßigen Intervallen Dokumentationen über die Tätigkeiten aller Teammitglieder geführt und diese folglich in Schrift festgehalten. Enthalten sind die Aufgabenaufteilung und die bisherige Umsetzung aller Projektteilnehmer in einer bestimmten Zeit, welche anschließend gebündelt zusammengefasst wurden.
  133. %
  134. \subsection*{Blogentwurf}
  135. \phantomsection\addcontentsline{toc}{subsection}{Blogentwurf}
  136. Da das git efi nicht nur für Professoren sondern auch Studierenden der Fakultät efi zugänglich ist, werden die Vorgänge des Projekts in kurzen Sätzen und möglichst verständlich wiedergegeben. So können sich auch Studierende außerhalb des Fachgebietes ein ungefähres Bild von dem Fortschritt der OHM News-App machen. Nach dem Umformulieren und Reduzieren auf die Kernaussagen wurden Einträge in kontinuierlichen Abständen auf dem Hochschul github\cite{gitefi} zum Abrufen hochgeladen. Diese berufen sich auf etwa ein Monats-Abstände oder kürzer. Bei neuen Einträgen wird Professor Dr. Hopf informiert, sodass er stets den Verlauf des Projektes nachvollziehen kann. In zeitlich chronologischer Reihenfolge aufgelistet, vereinfacht dies den Überblick. Insgesamt wurden im Laufe des OHM News-App Projektes sechs Wiki Blogeinträge erstellt.
  137. %
  138. \section*{Pflichtenheft}
  139. \phantomsection\addcontentsline{toc}{section}{Pflichtenheft}
  140. Das Pflichtenheft hat die Gruppe gemeinsam zusammengestellt. Dabei sollte der Datenzugriff per MongoDB erzeugt werden, alternativ könnte auch eine Schnittstelle zur Verfügung gestellt werden. Die Grafische UI der PWA sollte einen Login, ein Menü, eine Suche, ein Profil, einen Filter nach Hashtags, ein Dashboard, Einstellungen, das Nachrichtfeld und ein Formular zum Verfassen von Nachrichten umfassen. Ein Abhängigkeitsdiagramm soll erstellt werden. Die Anwendungslogik sollte im Bereich des Routings, der Darstellung und Speicherung von Nachrichten und dem LDAP-Login umgesetzt werden. Codetests sollten durchgeführt werden und das Rollenkonzept der Nutzer sollte festgelegt werden.\\
  141. Des Weiteren sollte ein Anwendungsname gefunden und ein Logo erstellt werden. Ein UX Konzept sollte aufgestellt werden. Wenn es zeitlich noch möglich wäre, hätte man die Funktionen umsetzen sollen um die Datenbankabrufe zu filtern und Kanälen zu folgen. Ebenso hätte man die Anzeigekriterien festlegen können.
  142. %
  143. \section*{Konzepterstellung}
  144. \phantomsection\addcontentsline{toc}{section}{Konzepterstellung}
  145. Der erste Schritt zur Definition eines Layoutentwurfs war die Erstellung von Scribbles. Unter einem Scribble versteht man einen groben ersten Entwurf, der zur Ideenfindung beitragen soll. Nach Festlegung der grundsätzlichen Projektidee, zeichnete jedes unserer Teammitglieder Scribbles für verschiedene Seiten der App, die wir im Anschluss zusammen diskutierten. Wichtig war uns hierbei, dass jedes Teammitglied erst eigene Ideen entwickelte, um so viele verschiedene Aspekte zu sammeln.\\
  146. Wir versuchten alle Ideen zu vereinen und einigten uns auf einen Layoutentwurf, welcher im Anschluss in einem detaillierten Screendesign umgesetzt wurde. In Abbildung \ref{layout} ist das Screendesign für die Startseite zu sehen. Wir legten vor allem den Fokus auf die Startseite, da diese der erste Baustein der App werden sollte. Die Navigation auf die anderen Seiten erfolgt über eine Leiste am unteren Displayrand. Die weiteren Seiten legten wir zu Beginn fest, arbeiteten diese allerdings erst mit der Zeit detaillierter aus. Ein genauer Aufbau der App wird unter 9.1. beschrieben. \\
  147. \begin{figure}[H]
  148. \centering
  149. \includegraphics[width=6cm]{Layout}
  150. \caption{Layout}
  151. \label{layout}
  152. \end{figure}
  153. %
  154. \section*{Namensfindung}
  155. \phantomsection\addcontentsline{toc}{section}{Namensfindung}
  156. Zu Anfang haben wir uns für den Titel OHM-Management entschieden, mit dem Wissen, dass es höchstwahrscheinlich nicht dabei bleibt, da das Wort Management mit unserer angestrebten Funktionalität der App nur einschränkend passte.
  157. Die Verwendung des Wortstammes Kommunikation war für unsere Benennung auch nicht zutreffend, insofern der Austausch von Information nur in eine Richtung erfolgt.
  158. Zusätzlich gestaltete sich die Namensfindung etwas schwierig, da wir sozusagen eine App für die TH-Nürnberg entwickelten, die man auch in Verbindung mit dem OHM-Zeichen kennt. Diesbezüglich wollten wir uns bei der Benennung auch an dieses anlehnen. Um dies tun zu können, mussten wir uns erst einmal informieren, ob es ähnliche Namen bereits gibt die man aus diesem Grund dementsprechend nicht verwenden kann.
  159. %
  160. \subsection*{Recherche}
  161. \phantomsection\addcontentsline{toc}{subsection}{Recherche}
  162. Bevor also der erste Schritt in Richtung Namensfindung gemacht werden konnte, recherchierten wir über bereits vorhandene Namen, die von der TH-Nürnberg genutzt werden. Neben Namen wie InfoTHek, OHMdoc und OHM-Shop, welche für unser Thema eher irrelevant wirkten, fanden wir auch den Titel OHM-News. Dies war ein Online Journal, welches jedoch, wie sich später rausgestellt hatte, eingestellt wurde.
  163. %
  164. \subsection*{Brainstorming}
  165. \phantomsection\addcontentsline{toc}{subsection}{Brainstorming}
  166. In der Zwischenzeit setzten wir jedoch die Ideensammlung weiter fort und präsentierten in der Gruppe einzelne Namensvorschläge. Hieraus entstand eine Liste an kreativen Gedanken. Unsere Favoriten :
  167. \begin{itemize}
  168. \item \b{ON AIR}
  169. \begin{itemize}
  170. \item \b{O}HM \b{N}ews \b{A}pp for \b{I}nformation and \b{R}ediscovery/\b{R}ecommendations
  171. \item \b{O}HM \b{N}ews Stream for \b{A}ctivities, \b{I}nformation and \b{R}ecommendations
  172. \end{itemize}
  173. %
  174. \item \b{ON STREAM}
  175. %
  176. \begin{itemize}
  177. \item \b{O}HM \b{N}ews \b{Stream}
  178. \end{itemize}
  179. %
  180. \item \b{OHM News}
  181. %
  182. \item \b{TH i s}
  183. %
  184. \item \b{INFOhm}
  185. \end{itemize}
  186. Alle Vorschläge waren sehr durchdacht und kreativ. Jedoch hatten einige nicht ganz die Wirkung die wir uns erhofften, denn wir waren der Meinung, dass man das Wort On Air eher mit einem Radiosender verbindet. Ebenfalls der Nachteil bei der Idee On Stream war es, dass man stark an Streaming Dienste wie zum Beispiel Netflix erinnert wird.
  187. %
  188. \subsection*{Ergebnis}
  189. \phantomsection\addcontentsline{toc}{subsection}{Ergebnis}
  190. Nach langem Überlegen kamen wir auf den Namen OHM News zurück und dieser überzeugte uns im Endeffekt alle. Dieser Titel war recht kurz, er ist leicht zu verstehen ohne viel nachzudenken, man verbindet ihn auf Anhieb mit der TH-Nürnberg auf Grund des Wortes OHM und er verweist auf die Funktionalität die unsere App bereitstellt, die da wären Neuigkeiten, sprich News, für die Studenten der Hochschule bereitzustellen.
  191. %
  192. \subsection*{Claim}
  193. \phantomsection\addcontentsline{toc}{subsection}{Claim}
  194. Parallel zur Namensfindung, befassten wir uns auch mit dem Claim unseres Produktes. Da ein Claim die Funktion unserer App auf den Punkt bringen soll, und dabei kurz, leicht zu merken und aussagekräftig sein sollte, entschieden wir uns für eine Formulierung der Art ''nur relevante Themen werden angezeigt''. Nach einer kurzen Denkphase kamen wir auf den Claim ''See only what matters'', beziehungsweise ''Only see what matters'' vor. Dies kam bei uns allen sehr gut an, jedoch wussten wir nicht, welche von den beiden Optionen die erhoffte Bedeutung am besten hervorbringt. Sie haben zwar die selbe Bedeutung, trotz alle dem empfanden wir, dass die Betonung des Wortes „Only“ auf zwei verschiedene Faktoren des Satzes liegt. Nach reichlicher Überlegung entschieden wir uns also für die Formulierung ''See only what matters'' als Claim für unsere App.
  195. %
  196. \section*{Corporate Design}
  197. \phantomsection\addcontentsline{toc}{section}{Corporate Design}
  198. Um das ganze Konzept stimmig zu gestalten, wurde ein Corporate Design erstellt.
  199. Wichtigste Grundlage hierfür ist das Farbkonzept. Da die App im Hochschulrahmen verwendet werden soll, wurde für die Grundfarbe der Blauton des Hochschullogos gewählt, umso eine Zugehörigkeit zu symbolisieren. Ergänzend zu diesem Blauton werden noch zwei dezente Grautöne verwendet, wie in Abbildung \ref{farbschema} zu sehen ist. \\
  200. %
  201. \begin{figure}[H]
  202. \centering
  203. \includegraphics[width=12cm]{Farbschema_CD}
  204. \caption{Farbkonzept}
  205. \label{farbschema}
  206. \end{figure}
  207. \noindent Für die Schrift wird in der gesamten App die von Google Material Design vorgegebene Schriftart Roboto verwendet. Diese ist eine serifenlosen Schriftart und deshalb durchweg gut lesbar.
  208. Auch ein Logo ist wichtig, um einen gewissen wiedererkennungswert für die App zu schaffen. Da wir uns in der Gruppe für den App-Namen ''Ohm-News'' entschieden haben, war der erster Gedanke eine Zeitung für das Logo herzunehmen (siehe Abbildung \ref{logo1}). Die Grafik war allerdings schwierig in das Produkt einzubinden, weil sie sehr detailreich ist. \\
  209. %
  210. \begin{figure}[H]
  211. \centering
  212. \includegraphics[width=4cm]{Logo1_CD}
  213. \caption{Erster Logoentwurf}
  214. \label{logo1}
  215. \end{figure}
  216. \noindent Ein zweiter etwas minimalistischer Entwurf ist in Abbildung \ref{logo2} zu sehen, für welchen sich das Projektteam schlussendlich auch entschied. Es zeigt ein Globussymbol mit dem Hochschullogo im Zentrum, um auch hier wieder die Verbindung zur Hochschule schaffen. \\
  217. %
  218. \begin{figure}[H]
  219. \centering
  220. \includegraphics[width=4cm]{Logo2_CD}
  221. \caption{Finales Logo}
  222. \label{logo2}
  223. \end{figure}
  224. %
  225. \noindent Das Logo wurde dann auch zu einem App Icon erweitert. In Abbildung \ref{icons} ist das Icon für Android sowie IOS Geräte zu sehen. Das Android Icon wurde analog den Material-Design Richtlinien gestaltet und besitzt deshalb einen leichten Schatten. Bei dem IOS Icon wurde das Logo lediglich auf einen einfarbigen Hintergrund gesetzt. \\
  226. %
  227. \begin{figure}[H]
  228. \centering
  229. \includegraphics[width=10cm]{Icons_CD}
  230. \caption{App Icons}
  231. \label{icons}
  232. \end{figure}
  233. %
  234. \section*{Erstellung der Mockups}
  235. \phantomsection\addcontentsline{toc}{section}{Erstellung der Mockups}
  236. Auf Basis des Screendesigns wurden erste Mockups erstellt, welche mittels HTML und CSS umgesetzt wurden. Um die Elemente entsprechend den Material-Design Richtlinien zu gestalten, wurde das CSS-Framework Bootstrap mit eingebunden, das eine Auswahl an vordefinierten Gestaltungselementen und Stilen in Form von CSS-Stylesheets bereitstellt. Dazu gehören Elemente wie Buttons, Formulare, Navigationszeilen und viele mehr. Wir verwendeten Bootstrap Dateien, die auf Material Design abgestimmt sind. So wird ein einheitliches Design gewährleistet und die Styles müssen nicht selbst erstellt werden. Ein weitere Vorteil von Bootstrap, ist, dass es ein responsives Webdesign unterstützt und man deshalb plattformunabhängig entwickeln kann. Dafür sorgt ein Grid-Layout, welches den Benutzerbildschirm in 12 Spalten teilt und die Elemente, je nach verfügbarer Auflösung, anzeigt. Im ersten Projektabschnitt legten wir den Fokus besonders auf eine mobile Ansicht. Eine Desktopansicht wird ebenfalls unterstüzt, kann allerdings noch stark optimiert werden. \cite{bootstrap} \\
  237. Durch die Ergänzung von eigenen CSS-Dateien konnten Änderungen an den vordefinierten Stilen vorgenommen werden. Sowohl unsere als auch die von Bootstrap bereitgestellten Dateien, wurden auf Grundlage von Less-Stylesheets erstellt. Dabei wird auf effiziente Weise CSS generiert. Variablen sowie Funktionen sind möglich und Vererbung wird durch \mbox{verschachtelte} Selektoren dargestellt. \cite{less} \\
  238. Wir habe für jeden Menüpunkt eine HTML-Seite erstellt, welche im Unterpunkt genauer beschrieben werden.
  239. %
  240. \subsection*{Aufbau der App}
  241. \phantomsection\addcontentsline{toc}{subsection}{Aufbau der App}
  242. %
  243. \begin{figure}[H]
  244. \begin{minipage}[t]{5cm}
  245. \vspace{0pt}
  246. \centering
  247. \includegraphics[width=4cm]{Homescreen_MU}
  248. \caption{Startseite}
  249. \label{abbildung_homescreen}
  250. \end{minipage}
  251. \hfill
  252. \begin{minipage}[t]{11cm}
  253. \vspace{0pt}
  254. Auf der Startseite der App gelangt der Nutzer als erstes auf einen Newsfeed. Der Newsfeed besteht aus einzelnen Nachrichten in Form von Karten und soll für jeden Nutzer nur \mbox{relevante} Nachrichten anzeigen. Jede Nachricht besitzt einen oder \mbox{mehrere} Tags. Eine Nachricht wird für einen Nutzer als relevant eingestuft, wenn dieser mindestens einen Tag aus der Nachricht abonniert hat. Somit erhält jeder Nutzer eine individuelle Startseite. Zusätzlich befindet sich auf der Nachrichtenkarte oben rechts ein Icon, welches die Nachricht zu einer Fakultät zuordnet. Unten rechts kann das Lesezeichen-Icon geklickt werden, um eine Nachricht zu speichern. Auf diese Weise können die Studierenden wichtige Informationen schnell wiederfinden.\\\\
  255. Im Header ist die Suche zu finden, in der Tags explizit gefiltert werden können. Der \mbox{Header} ist von jeder Seite aus zugängig. Im Footer befindet sich die Navigationszeile, von wo aus der Nutzer auf die weiteren Seiten gelangt.
  256. \end{minipage}
  257. \end{figure}
  258. %
  259. \begin{figure}[H]
  260. \begin{minipage}[t]{5cm}
  261. \vspace{0pt}
  262. \centering
  263. \includegraphics[width=5cm]{Links_MU}
  264. \caption{Infoseite}
  265. \label{abbildung_infoseite}
  266. \end{minipage}
  267. \hfill
  268. \begin{minipage}[t]{11cm}
  269. \vspace{0pt}
  270. Der zweiten Menüpunkt, der durch ein Globus-Icon dargestellt wird, soll eine Informationsseite beinhalten. Das Konzept hierfür ist allerdings noch nicht konkret ausgearbeitet, da wir uns zuerst auf die Mitteilungsfunktionen konzentrieren wollten. Aktuell wird ein Empty State angezeigt.
  271. \end{minipage}
  272. \end{figure}
  273. %
  274. \medskip
  275. \begin{figure}[H]
  276. \begin{minipage}[t]{5cm}
  277. \vspace{0pt}
  278. \centering
  279. \includegraphics[width=5cm]{CreateMessage_MU}
  280. \caption{Formular}
  281. \label{abbildung_createmsg}
  282. \end{minipage}
  283. \hfill
  284. \begin{minipage}[t]{10cm}
  285. \vspace{0pt}
  286. Über das Plus-Icon, in der Mitte der Navigationszeile, gelangt man zu einem Formular. Hier können neue Nachrichten erstellt werden. Nachrichten sollen nur von ausgewählten Nutzergruppen, wie Fakultäten, Studienbüro und ähnlichen gesendet werden können.
  287. Für eine Nachricht wird ein Betreff, mindestens ein Tag und der Nachrichteninhalt benötigt. Es soll eine Auswahl an vordefinierten Tags geben, damit am Ende nicht jeder Verfasser wahllos Tags setzten kann und \mbox{Duplikate} vermieden werden. Dieser Menüpunkt ist der einzige in dem keine Navigationszeile angezeigt wird. Nach Absenden einer Nachricht, gelangt man zurück auf die Startseite. Über den Button 'Abbrechen' wird man auf den vorherigen Menüpunkt zurückgeleitet.
  288. \end{minipage}
  289. \end{figure}
  290. %
  291. \begin{figure}[H]
  292. \begin{minipage}[t]{5cm}
  293. \vspace{0pt}
  294. \centering
  295. \includegraphics[width=5cm]{Gespeicherte_MU}
  296. \caption{Gespeicherte}
  297. \label{abbildung_gespeichert}
  298. \end{minipage}
  299. \hfill
  300. \begin{minipage}[t]{11cm}
  301. \vspace{0pt}
  302. Der nächste Menüpunkt wird über das Lesezeichen-Icon erreicht und zeigt die gespeicherten Nachrichten an. Diese Seite ist ähnlich aufgebaut wie der Newsfeed der Startseite, beinhaltet allerdings nur die vom Nutzer gespeicherten Nachrichten.
  303. \end{minipage}
  304. \end{figure}
  305. %
  306. \medskip
  307. \begin{figure}[H]
  308. \begin{minipage}[t]{5cm}
  309. \vspace{0pt}
  310. \centering
  311. \includegraphics[width=5cm]{Profil_MU}
  312. \caption{Profil}
  313. \label{abbildung_profil}
  314. \end{minipage}
  315. \hfill
  316. \begin{minipage}[t]{11cm}
  317. \vspace{0pt}
  318. Der letzte Menüpunkt ist das Profil. \\
  319. Oben sieht man zuerst das Profilbild, hier verwendeten wir die runde Form anstatt der üblichen Quadratischen. Darunter sieht man dann die im Vorfeld angegebene Information des Benutzers. Am Ende der sogenannten Karte befinden sich zwei Knöpfe, die Anzeigen wie vielen Kanälen man folgt und wie viele Beiträge man bis jetzt gespeichert hat. Auf der Karte rechts oben im Eck findet man den Bearbeitungs-Knopf, in Form eines kleinen Stiftes, dies ermöglicht dem Benutzer, die Seite gemäß seinen Wünschen anzupassen.
  320. \end{minipage}
  321. \end{figure}
  322. %
  323. \section*{Single-Page Entwicklung}
  324. \phantomsection\addcontentsline{toc}{section}{Single-Page Entwicklung}
  325. Neben klassischen Webanwendungen gibt es ebenfalls Single-Page Webanwendungen. Bei klassischen Webapplikationen gibt es mehrere untereinander verlinkte HTML-Dokumente, die dann gemeinsam eine komplette Anwendung ausmachen. Der Nachteil hierbei ist, dass die Seite beim Wechsel durch die erneute Kommunikation mit dem Server neu heruntergeladen werden muss. SPA’s jedoch bestehen nur aus einer HTML-Datei. Dies bewirkt, dass neue oder veränderte Daten dynamisch geladen werden und somit die Kommunikation zwischen Server und Client verringert wird. Es ermöglicht dem Benutzer, nach dem einmaligen laden der Seite, diese auch weiterhin im offline Modus zu steuern. Sollte also die Interaktion mit dem Server unterbrochen werden, werden die zwischengespeicherten Daten einfach aus dem Webbrowser verwendet.\cite{spa:1}\cite{spa:2}
  326. %
  327. \subsection*{Warum Vue.js?}
  328. \phantomsection\addcontentsline{toc}{subsection}{Warum Vue.js?}
  329. Neben dem Vue.js Framework gibt es noch weitere wie zum Beispiel react.js, angular.js, meteor.js und viele mehr. Warum wir uns jedoch für das Vue.js Framework entschieden haben, werden wir Ihnen nun erläutern. Vue.js das jüngste von den drei größten Frameworks, es hat jedoch den Vorteil, dass es aus den Fehlern der Vorgänger lernen konnte und diese schon im Vorfeld überdenken und ausmerzen konnte. Da es auf Sprachen wie HTML, CSS und JavaScript aufbaut, lässt es sich problemlos in schon bestehende Projekte einbauen.
  330. Die Syntax ist hierbei, im Vergleich zu anderen, besser lesbar. Hierfür werden nämlich sogenannte ''Single File Components'' verwendet.\cite{vue:warum} Sogenannte Komponenten sind wiederverwendbare Vue-Instanzen mit einem Namen, die als benutzerdefiniertes Element in einer Vue-Stamminstanz verwendet werden. Wie so eine erstellte Komponente aussieht, werden wir Ihnen im folgenden erörtern.
  331. %
  332. \subsection*{Vue.js in vorhandene Mockups einbinden}
  333. \phantomsection\addcontentsline{toc}{subsection}{Vue.js in vorhandene Mockups einbinden}
  334. Nach reichlicher Recherche und dem einlesen in das uns vorher unbekannte Framework, war es an der Zeit unsere HTML-Seiten in Vue-Komponenten umzuwandern. Wir setzten uns erstmal zusammen und erstellten die Startseite.
  335. Im Folgenden wird der Aufbau der Home-Komponente näher erläutert. In Listing 1 ist der dazugehörige Quellcode zu sehen. Nicht enthalten sind zur Komponente zugehörige Methoden. Die Home-Komponente wird in unserem Projekt als HomeRouter betitelt.
  336. %
  337. \begin{center}
  338. \begin{lstlisting}[caption={HomeRouter-Komponente},captionpos=b]
  339. const HomeRouter = {
  340. template: `
  341. <div id="om-msg-cards">
  342. <MsgCard
  343. v-for="id in messagelist.slice().reverse()"
  344. :key="id"
  345. :msg="messages[id] || {}"
  346. ></MsgCard>
  347. </div>`,
  348. //Restliche Komponente
  349. ...
  350. \end{lstlisting}
  351. \end{center}
  352. %
  353. Im Newsfeed, welcher der Hauptbestand der Startseite ist, wird eine Liste von Nachrichten dynamisch generiert, sodass jeder Nutzer später einmal eine individuelle Ansicht erhält, abhängig von den Tags, die dieser abonniert. \\
  354. Jede Komponente besitzt ein Template in Form eines Strings, welcher HTML-Code \mbox{beinhaltet}. Die Komponente HomeRouter ist verschachtelt und enthält die Komponente MsgCard. Der Quellcode für diese Komponete ist in Listing 2 zu sehen.
  355. %
  356. \begin{center}
  357. \begin{lstlisting}[caption={MsgCard-Komponente},captionpos=b]
  358. Vue.component('MsgCard', {
  359. template: `<div class="om-card card">
  360. <h6 class="msg-head">
  361. <b>{{ msg.subject }}</b>
  362. <img src="favicon.ico" width=20px height=20px>
  363. </h6>
  364. {{ msg.message }}<br><br>
  365. <a href="#">{{ msg.tag }}</a></p>
  366. <div class="om-card-footer"> <div class="om-user-line">
  367. <i class="material-icons">account_circle</i>
  368. Erstellt von {{ msg.user }}</div>
  369. <i class="material-icons">bookmark_border</i>
  370. </div></div>`,
  371. props: ['msg']
  372. });
  373. \end{lstlisting}
  374. \end{center}
  375. %
  376. Eine MsgCard stellt eine einzelne Nachricht in Form einer Karte dar. In der HomeRouter-Komponete geht eine For-Schleife alle vorhandenen Nachrichten-Ids durch. Dabei werden der MsgCard-Komponente alle relevanten Informationen übergeben, die für eine Darstellung benötig wird. Am Ende stellt die App eine fortlaufende Liste mit allen Nachrichten dar.\\\\
  377. Da wir nun ein bisschen geübter im Umgang mit Vue waren, teilten wir uns die restlichen Seiten auf. Dabei gingen wir ähnlich vor, wie bei der Home-Komponente.
  378. %
  379. \subsection*{Routing}
  380. \phantomsection\addcontentsline{toc}{subsection}{Routing}
  381. Vue.js bietet eine Reihe von Funktionen, mit denen Sie wiederverwendbare Webkomponenten erstellen können. Routing ist eine dieser Methoden. Der Benutzer kann somit zwischen den Seiten wechseln, ohne diese erneut zu aktualisieren. Dies ermöglicht dem Benutzer eine einfache und dynamische Navigation durch die Webanwendung. Wir werden Ihnen nun, anhand unseres Codes erklären, wie ein Vue.js-Router funktioniert.\cite{vue:router} \\
  382. Zunächst richten wir hierfür die Navigation über die Navigationsbar ein, die wir mittels einem Router-Link-Element erstellt haben. Dieser gibt an, auf welche der angegebenen JS-Datei zugegriffen wird sobald man auf das ausgewählte Item klickt.
  383. %
  384. \begin{center}
  385. \begin{lstlisting}[caption={Navbar},captionpos=b]
  386. <nav class="nav nav-tabs nav-justified om-nav" v-if="$route.path !=='/createMessage' ">
  387. <router-link to="/home" class="nav-item nav-link">
  388. <i class="material-icons">home</i></router-link>
  389. <router-link to="/files" class="nav-item nav-link">
  390. <i class="material-icons">language</i></router-link>
  391. <router-link to="/createMessage" class="nav-item nav-link outlined">
  392. <i class="material-icons">add_circle</i></router-link>
  393. <router-link to="/bookmark" class="nav-item nav-link">
  394. <i class="material-icons">bookmark</i></router-link>
  395. <router-link to="/profil" class="nav-item nav-link">
  396. <i class="material-icons">person</i></router-link>
  397. </nav>
  398. \end{lstlisting}
  399. \end{center}
  400. %
  401. Im Folgenden werden Komponenten definiert und dann in ein neues Array von Objekten, so genannte Routen, eingefügt. Zu beachten ist die Zuordnung eines URL-Pfads zu der dazugehörigen Komponente. Daraufhin wird ein Vue-Router definiert, der an ein neues Vue-Objekt übergeben wird. Wenn nun beispielsweise ''/home'' geladen ist, wird die Home-Komponente in der Router-Ansicht gerendert und aufgerufen. Die erste Zeile mit dem ''/'' steht für die Startseite, auf die man gelangt, sobald man die App öffnet.\cite{vue:router3} \\
  402. %
  403. \begin{center}
  404. \begin{lstlisting}[caption={Routing},captionpos=b]
  405. const routes = [
  406. { path: "/", component: HomeRouter },
  407. { path: "/home", component: HomeRouter },
  408. { path: "/files", component: FileRouter },
  409. { path: "/createMessage", component: CreateMsgRouter },
  410. { path: "/bookmark", component: BookmarkRouter },
  411. { path: "/profil", component: ProfilRouter },
  412. ];
  413. const router = new VueRouter({
  414. routes
  415. });
  416. var app = new Vue({
  417. router,
  418. el: '#api',
  419. methods: {
  420. ...
  421. },
  422. });
  423. \end{lstlisting}
  424. \end{center}
  425. %
  426. \section*{Service Worker}
  427. \phantomsection\addcontentsline{toc}{section}{Service Worker}
  428. Der ServiceWorker ist eine JavaScript Datei, die üblicherweise im root-Verzeichnis der Webanwendung liegt. Für die Funktionsweise ist die Namensgebung der Datei nicht maßgebend. Damit der ServiceWorker überhaupt funktionieren kann, wird eine SSL-Verbindung benötigt. Ist eine sicher Verbindung gegeben, kann der ServiceWorker (im Weiteren auch mit ''SW'' abgekürzt) nach erfolgreichem Laden der Webseite im Browser des Clients registriert werden. Nachdem der SW zeitlich unabhängig, also asynchron im Browser läuft, muss bei der Entwicklung und Umsetzung des ServiceWorker-Codes darauf geachtet werden, dass in Promises geschrieben wird. Da Promises ein wenig schwieriger nachzuvollziehen sind und insbesondere das Problem der sogenannten Callback-Hell nicht lösen, bietet es dich an den SW in Async/Await Struktur zu schreiben. Diese basiert auf dem asynchronen Prinzip der Promises und lässt sich dennoch lesen wie synchrone Funktionen.\\
  429. Der Code für die Registrierung des ServiceWorkers im Browser:\\
  430. %
  431. \begin{center}
  432. \begin{lstlisting}[caption={Register, Promise/Then-Codestruktur},captionpos=b]
  433. /* Promise / Then */
  434. if ('serviceWorker' in navigator) {
  435. window.addEventListener('load', function() {
  436. navigator.serviceWorker.register('/serviceWorker.js').then(function(registration) {
  437. // Registration was successful
  438. }, function(err) {
  439. // registration failed
  440. });
  441. });
  442. }
  443. \end{lstlisting}
  444. \end{center}
  445. %
  446. \begin{center}
  447. \begin{lstlisting}[caption={Register, Async/Await-Codestruktur},captionpos=b]
  448. /* Async / Await */
  449. const registerServiceWorker = async () => {
  450. try {
  451. const registration = await navigator.serviceWorker.register('/serviceWorker.js', {scope: '/'});
  452. // Registration successful
  453. } catch (err) {
  454. // Registration failed
  455. }
  456. return;
  457. }
  458. if ('serviceWorker' in navigator) {
  459. window.addEventListener('load', registerServiceWorker());
  460. }
  461. \end{lstlisting}
  462. \end{center}
  463. %
  464. Einmal in der Promise.then Schreibweise und einmal in der Async/Await Schreibweise. Beide Codes machen exakt das selbe.\\
  465. Nach der Registrierung folgt die Installation. Das ist der Code dazu:
  466. %
  467. \begin{center}
  468. \begin{lstlisting}[caption={Install, Promise/Then-Codestruktur},captionpos=b]
  469. /* Promise / Then */
  470. const cacheName = 'appname-v1-0';
  471. const filesToCache = [ '/index.html' ];
  472. self.addEventListener('install', function(event) {
  473. console.log('[ServiceWorker] Install');
  474. event.waitUntil(
  475. caches.open(cacheName).then(function(cache) {
  476. console.log('[ServiceWorker] Caching app shell');
  477. return cache.addAll(filesToCache);
  478. })
  479. );
  480. });
  481. \end{lstlisting}
  482. \end{center}
  483. %
  484. \begin{center}
  485. \begin{lstlisting}[caption={Install, Async/Await-Codestruktur},captionpos=b]
  486. /* Async / Await */
  487. const installNewCache = async (event) => {
  488. const cacheName = 'appname-v1-0';
  489. const filesToCache = [ '/index.html' ];
  490. const cacheStatic = await caches.open(cacheName);
  491. cacheStatic.addAll(filesToCache);
  492. console.log('[ServiceWorker] Cache static files.');
  493. return;
  494. }
  495. self.addEventListener('install', event => {
  496. // don't wait
  497. self.skipWaiting();
  498. // cache static files
  499. event.waitUntil(installNewCache());
  500. console.log('[ServiceWorker] Install');
  501. });
  502. \end{lstlisting}
  503. \end{center}
  504. %
  505. Während des 'install' Events wird eine neue Version des ServiceWorkers im Browser installiert, sofern eine Netzwerkverbindung besteht. Der nächste Schritt ist die Aktivierung:
  506. %
  507. \begin{center}
  508. \begin{lstlisting}[caption={Activate, Promise/Then-Codestruktur},captionpos=b]
  509. /* Promise / Then */
  510. self.addEventListener('activate', function(event) {
  511. // ServiceWorker activated
  512. event.waitUntil(
  513. caches.keys().then(function(cacheKeyList) {
  514. return Promise.all(
  515. cacheKeyList.map(function(cacheKeyList) {
  516. if (cacheWhitelist.indexOf(cacheKeyList) === -1) {
  517. // Old Cache removed
  518. return caches.delete(cacheKeyList);
  519. }
  520. })
  521. );
  522. })
  523. );
  524. return self.clients.claim();
  525. });
  526. \end{lstlisting}
  527. \end{center}
  528. %
  529. \begin{center}
  530. \begin{lstlisting}[caption={Activate, Async/Await-Codestruktur},captionpos=b]
  531. /* Async / Await */
  532. const cacheCleanUp = async () => {
  533. const cacheKeyList = await caches.keys();
  534. const deletions = cacheKeyList
  535. .filter(key => key.startsWith(appPrefix) && !cacheKeyList.includes(key))
  536. .map(key => {
  537. caches.delete(key)
  538. // Old Cache removed
  539. });
  540. for (const success of deletions) {
  541. await success;
  542. }
  543. return;
  544. }
  545. self.addEventListener('activate', event => {
  546. event.waitUntil(cacheCleanUp());
  547. clients.claim();
  548. // ServiceWorker activated
  549. });
  550. \end{lstlisting}
  551. \end{center}
  552. %
  553. Hier werden sämtliche alte ServiceWorker Dateien aus dem Cache gelöscht und alte SW deinstalliert und unregistriert. Nun bleibt nur noch ein Schritt aus, das ist der 'fetch':
  554. %
  555. \begin{center}
  556. \begin{lstlisting}[caption={Fetch, Promise/Then-Codestruktur},captionpos=b]
  557. /* Promise / Then */
  558. self.addEventListener('fetch', function(event) {
  559. /* We should only cache GET requests */
  560. if (event.request.method !== 'GET') {
  561. /* If we don't block the event as shown below,
  562. then the request will go to the network as usual. */
  563. console.log('[ServiceWorker] Fetch event ignored.',
  564. event.request.method, event.request.url);
  565. return;
  566. }
  567. event.respondWith(
  568. caches.match(event.request)
  569. .then(function(response) {
  570. return fetch(event.request)
  571. .then(function(response) {
  572. // Check if response is valid, status is 200, response type is basic
  573. // (indicates request is from origin, means that requests to third party
  574. // assets aren't cached as well.
  575. if(!response || response.status !== 200 || response.type !== 'basic') {
  576. return response;
  577. }
  578. caches.open(CACHE_NAME)
  579. .then(function(cache) {
  580. // We have to clone the response here because request bodies can only
  581. // be read once. Placing a response in the cache counts as a read.
  582. cache.put(event.request, response.clone());
  583. });
  584. // Cache hit - return response
  585. if (response) return response;
  586. return response;
  587. });
  588. })
  589. .catch(function(err) {
  590. // Report a lack of connectivity to the user.
  591. });
  592. });
  593. \end{lstlisting}
  594. \end{center}
  595. %
  596. \begin{center}
  597. \begin{lstlisting}[caption={Fetch, Async/Await-Codestruktur},captionpos=b]
  598. /* Async / Await */
  599. self.addEventListener('fetch', event => {
  600. /* We should only cache GET requests */
  601. if (event.request.method !== 'GET') {
  602. /* If we don't block the event as shown below,
  603. then the request will go to the network as usual. */
  604. console.log('[ServiceWorker] Fetch event ignored.',
  605. event.request.method, event.request.url);
  606. return;
  607. }
  608. event.respondWith(async function update() {
  609. try {
  610. var requestURL = new URL(event.request.url);
  611. // Start the network request as soon as possible.
  612. const networkPromise = fetch(requestURL);
  613. const cachedResponse = await caches.match(event.request);
  614. const networkResponse = await networkPromise;
  615. // Check if response is valid, status is 200, response type is basic
  616. // (indicates request is from origin, means that requests to third party
  617. // assets aren't cached as well.
  618. if (!networkResponse || networkResponse.status !== 200
  619. || networkResponse.type !== 'basic') return networkResponse;
  620. const cache = await caches.open(staticCacheKey);
  621. // We have to clone the response here because request bodies can only
  622. // be read once. Placing a response in the cache counts as a read.
  623. cache.put(event.request, networkResponse.clone());
  624. if (cachedResponse) return cachedResponse;
  625. // ServiceWorker fetch
  626. return networkResponse;
  627. } catch (err) {
  628. // Report a lack of connectivity to the user.
  629. }
  630. }());
  631. });
  632. \end{lstlisting}
  633. \end{center}
  634. %
  635. Während des 'fetch' Events werden die Dateien im Cache überprüft und gegebenenfalls erneuert, falls keine Netzwerkverbindung besteht, wird eine Version aus dem Cache angezeigt. Falls diese ebenfalls nicht vorhanden ist, dann kann leider nichts angezeigt werden.\\
  636. Die Bildausschnitte mit dem Code wurden teils von mir erstellt, teils stammen diese von dieser Internetseite\cite{dev:codeSource}\cite{sw:cacheThenNetwork}.
  637. %
  638. \section*{''Add-To-Homescreen''-Funktion}
  639. \phantomsection\addcontentsline{toc}{section}{''Add-To-Homescreen''-Funktion}
  640. Für die Implementierung der ''Add-to-Homescreen''-Funktionalität sind für Android, iOS und die diversen Browser unterschiedliche Schritte nötig.\\
  641. Bei Android genügt es eine manifest.json Datei im root-Verzeichnis der App abzulegen. Diese Manifest-Datei enthält Meta-Informationen von App-Icon, über App-Name, über die Start-URL bis hin zu der Theme-Farbe für den Hintergrund des App-Icons oder dem Navigationsmenü im Browser. Wenn ein Nutzer im Abstand von 5 Minuten (Standard-Einstellung) die Webseite zweimalig aufruft, so wird dieser gefragt, ob er die App zum Homescreen hinzufügen möchte. Ab da kann er diese über das App-Icon als auch über den Browser starten.\\
  642. Für iOS müssen die Meta-Informationen im Bereich des Head-Elements, der index.html eingefügt werden. So wird dort das Aussehen von mehreren App-Icon Versionen definiert. Dies wird benötigt, da die meisten Apple-Geräte unterschiedliche Density (Pixel per Inch) haben. Des Weiteren werden Konfigurationseinstellungen aktiviert oder deaktiviert, als auch das Aussehen des Ladescreens für iOS Geräte festgelegt.\\
  643. Auch die Windows-Mobilgeräte muss in der index.html Datei im Head-Bereich mittels Meta-Informationen mitgeteilt werden, wie der Ladebildschirm oder das Appicon aussehen soll (vgl.\cite{dev:homescreen}).
  644. %
  645. \section*{Erstellung der API-Schnittstelle}
  646. \phantomsection\addcontentsline{toc}{section}{Erstellung der API-Schnittstelle}
  647. Nachdem Server und die Clientseite der PWA funktionsfähig waren, war es noch relevant, dass zwischen diesen beiden kommuniziert werden kann. Dafür verwendet man die API.\cite{api} API steht für ''Application Programming Interface'' und diese Programmierschnittstelle kann wie in diesem Projekt zum Zugriff auf die Datenbank verwendet werden. Dabei handelt es sich in diesem Fall um einen RESTful Webservice.\cite{rest} \\
  648. %
  649. \begin{figure}[H]
  650. \centering
  651. \includegraphics[width=0.3\textwidth]{ajax.png}
  652. \caption{Ajax Modell einer Webanwendung. Quelle: nach Haischt, Daniel; https://de.wikipedia.org/wiki/Ajax\_(Programmierung)\#/media/File:Ajax-vergleich.svg}
  653. \label{ajax}
  654. \end{figure}
  655. %
  656. \noindent Das asynchrone Datenübertragungskonzept AJAX wurde zur Kommunikation zwischen Browser und Server verwendet.\cite{ajax1} \cite{ajax2} \cite{ajax3} Dabei werden wie in Abbildung \ref{ajax} zu sehen ist HTTP Requests von dem Browser an den Server geschickt. Diese werden daraufhin von dem Server verarbeitet und es werden entweder Daten zurückgegeben oder ein HTTP Response mit einer Rückmeldung. Mit der Verwendung von AJAX hat man einige Vorteile, zum Beispiel, dass man die Seite nicht neu laden muss, damit die neuen Inhalte erscheinen und dass der Browser bereits weiter Requests schicken kann, ohne auf die Responses des Servers warten zu müssen. Da die gesamte PWA des Projekts in JavaScript geschrieben ist, ist es zudem vorteilhaft, dass die AJAX-Engine in JavaScript geschrieben wird. Die Daten aus der Datenbank werden im JSON-Format transportiert.\\
  657. Um die Funktionsweise einer derartigen Schnittstelle zu verstehen, wurde der Code einer PWA die Professor Matthias Hopf geschrieben und der Gruppe zur Verfügung gestellt hat, analysiert. Dieser war jedoch etwas zu komplex für einen Einsteiger in das Thema. Deswegen wurde auf simplere Beispiele in Tutorials zurückgegriffen. Diese Beispiele konnten nun auf das eigene Projekt übertragen und getestet werden. Nach einer ausführlichen Erläuterung durch Professor Hopf anhand eines Beispiels im Projektzusammenhang, war ein grundsätzliches Verständnis über die Schnittstelle vorhanden. Nun konnte die konkrete Umsetzung der API beginnen.\\
  658. Die erste Methode die umgesetzt wurde war ''list\_messages''. Diese zeigt nun alle Nachrichten der Datenbank auf der Startseite der PWA an. Dazu musste zuerst vom Server die IDs aller Nachrichten abgefragt werden. Man schreibt eine API Funktion, die ein GET Request an den Server schickt. Dieser sucht nach den IDs per find Anfrage in der Datenbank und antwortet mit einem HTTP Response, der das Array mit den IDs enthält. Auf der Client Seite wird nun für jede ID per GET Request alle Informationen der Nachricht angefragt. Der Server sucht die gefragten Information und schickt diese zurück mit dem dazugehörigen Nachrichtentext an den Browser.\\
  659. %
  660. \begin{center}
  661. \begin{lstlisting}[caption={''list\_messages''-Methode},captionpos=b]
  662. list_messages: function () {
  663. $.ajax({url: "api/ids", method: "GET"})
  664. .done(jd => {
  665. _messagelist.splice(0, _messagelist.length);
  666. _messagelist.push.apply(_messagelist, jd);
  667. console.log("jd: "+jd);
  668. for (var e in jd) {
  669. if (!_messages[jd[e]]) {
  670. get_insert_message(jd[e]);
  671. }
  672. }
  673. }).fail(function (e, f, g) {
  674. console.log("err: " + e + f + g);
  675. });
  676. }
  677. \end{lstlisting}
  678. \end{center}
  679. %
  680. Was für Vorgänge beim Senden einer Nachricht in einer API getätigt werden, ist im Folgenden aufgeführt. Hierfür muss die Mongodb heruntergeladen und angebunden sein.\\
  681. In der Server.js Datei wird durch die ''App.post''-Methode eine Nachricht erstellt. Dabei wird in ''createMsg'' die Funktion aufgerufen, in einen String umgewandelt und die Nachricht anschließend in der Datenbank gespeichert. Mit der POST-Methode können Nachrichten in die Datenbank gespeichert werden und anhand von einer GET-Methode angezeigt werden.
  682. Nun müssen in der createMsg.js alle Werte einsetzbar gestaltet werden. Wichtig dabei ist, sich an das vorgegebene Mongo Schema zu halten. Durch die ajax-Methode ''post'' können große Datenmengen an den Server geschickt werden. Zuletzt soll die Konsole auch einen Error anzeigen, falls der Abruf der Funktion erfolglos war. \\
  683. In Abbildung \ref{sys} kann man diese Vorgänge nachverfolgen. Diese beiden Funktionen haben am Ende funktioniert und können zusammen verwendet werden. \\
  684. \begin{figure}[H]
  685. \centering
  686. \includegraphics[width=0.7\textwidth]{system.png}
  687. \caption{Erstellen einer Nachricht und Anzeigen aller Nachrichten}
  688. \label{sys}
  689. \end{figure}
  690. %
  691. \section*{Aufsetzung des Servers}
  692. \phantomsection\addcontentsline{toc}{section}{Aufsetzung des Servers}
  693. Zunächst muss ein http port eingerichtet werden. Über einen lokalen Server über node.js ist die Ohm News-App abrufbar. Mit der createServer Methode
  694. %
  695. \begin{center}
  696. \begin{lstlisting}[caption={''list\_messages''-Methode},captionpos=b]
  697. http.createServer(app).listen(http_port, function () {
  698. console.log("Express http server listening on port " + http_port);
  699. });
  700. \end{lstlisting}
  701. \end{center}
  702. %
  703. lässt sich ein Server aufsetzen. Hier wird auch der Listener für den Port impliziert und auf der Konsole ausgegeben, wenn die Verbindung erfolgreich war. \\
  704. Um die wichtigsten Fehlermeldungen und Bugs auszugeben, wird für unsere node.js App der Logger ''Morgan'' verwendet. Zusätzlich wird eine try-catch Funktion eingeordnet, da Zertifikat und Schlüssel noch nicht zertifiziert sind (siehe SSL und Keys) und somit vorerst ein Fehler angezeigt wird.
  705. %
  706. \subsection*{SSL und Keys}
  707. \phantomsection\addcontentsline{toc}{subsection}{SSL und Keys}
  708. Um die Applikation für die Nutzer sicher zu gestalten und eine Verschlüsselung sensibler Daten zu gewährleisten, sorgt ein SSL(Secure Socket Layer). Dadurch wird ein sicherer Kanal zwischen Web-Server und Client bereitgestellt und aus der http- wird eine https-Verbindung.
  709. \setlength{\parindent}{0cm}
  710. Übergangsweise werden ein selbst verifizierter Schlüssel und Zertifikat erstellt, welche später durch echte ausgetauscht werden. In einem neu erstellten Ordner ''keys'' werden das Zertifikat und der Schlüssel platziert.
  711. %
  712. \section*{Datenbankanbindung}
  713. \phantomsection\addcontentsline{toc}{section}{Datenbankanbindung}
  714. Für diese Anwendung wird eine Datenbank benötigt, weil die geschriebenen Nachrichten gespeichert werden müssen. Später sollen auch User mit ihren Userinformationen gespeichert werden. Im Falle dieses Projekts wurde sich für eine MongoDB entschieden.\\
  715. MongoDB ist eine dokumentenorientierte NoSQL Datenbank. Diese ist für diese Anwendung sinnvoll, da sie im Vergleich zu SQL Datenbanken flexibler handhabbar ist und somit auch viele Einstellungen noch im Nachhinein geändert werden können.\cite{mongo} \\
  716. Um den Umgang mit MongoDB zu vereinfachen wurde Mongoose, eine Object Data Modeling Bibliothek, verwendet.\cite{mongoose} \\
  717. Da man zu Beginn des Projekts noch nicht mit NoSQL Datenbanken gearbeitet hatte, mussten zuerst Tutorials und Erklärungen gelesen werden. Anhand von Beispieltests mit einer lokalen Datenbank konnte die grundsätzliche Funktionsweise bald verstanden werden. Daraufhin wurde das grobe Schema für eine Nachricht aufgebaut. Eine Nachricht sollte einen Betreff, eine Nachricht, den schreibenden User und die Tags beinhalten wie man in Listing \ref{lst:json} an einer beispielhaften JSON-Datei sehen kann. Die Datenbank musste konfiguriert und gestartet werden.
  718. %
  719. \begin{center}
  720. \lstset{language=xml}
  721. \begin{lstlisting}[caption={Nachricht im JSON-Format},captionpos=b,label=lst:json]
  722. {
  723. "messages": [{
  724. "subject": "Test"
  725. "message": "Hallo Welt!"
  726. "tag": "#OHM_News"
  727. "user": "User1"
  728. }]
  729. }
  730. \end{lstlisting}
  731. \lstsetdefault
  732. \end{center}
  733. %
  734. Die serverseitigen Codeteile zur API und der Code für die Datenbank sind momentan zur Vereinfachung in der server.js Datei. Später sollen diese zur Übersichtbarkeit und Wartbarkeit ausgelagert werden. Das Schema für die Nachricht und die Konfiguration der Datenbank sind zu diesem Zweck bereits in eigenen js-Dateien zu finden.\\
  735. Sobald all das lokal funktioniert hat, wurde die Portierung auf einen Server der Hochschule vorgenommen.
  736. %
  737. \section*{Plakatgestaltung}
  738. \phantomsection\addcontentsline{toc}{section}{Plakatgestaltung}
  739. Zu dem Plakat für die OHM News App wurde eine Ideensammlung in Form von Skizzen angefertigt.
  740. \begin{figure}[H]
  741. \centering
  742. \includegraphics[width=6.5cm, angle=90, scale=1.409]{IMG_Skizze1.jpg}\quad
  743. \includegraphics[width=6.5cm]{IMG_Skizze2.jpg}
  744. \includegraphics[width=6.5cm, angle=270, scale=1.409]{IMG_Skizze3.jpg}\quad
  745. \includegraphics[width=6.5cm, angle=270, scale=1.409]{IMG_Skizze4.jpg}
  746. \end{figure}
  747. %
  748. Die Umsetzung derer folgte in Form von ersten Entwürfen in Illustrator, welche der gesamten Gruppe vorgestellt wurden. \\
  749. Zu den hier unten aufgeführten Plakatentwürfen wurden jeweils zwei Versionen erstellt. Diese zwei kamen jedoch in die engere Auswahl und sollten noch weiter ausgearbeitet werden.
  750. \begin{figure}[H]
  751. \centering
  752. \includegraphics[width=6.5cm]{PDF_Plakat1.pdf}\quad
  753. \includegraphics[width=6.5cm]{PDF_Plakat2.pdf}
  754. \end{figure}
  755. %
  756. Bei der nächsten Vorauswahl an fünf OHM News Plakat-Versionen wurde daraufhin auf zwei reduziert, welche hier zu sehen sind.
  757. \begin{figure}[H]
  758. \centering
  759. \includegraphics[width=6.5cm]{PDF_Plakat3.pdf}\quad
  760. \includegraphics[width=6.5cm]{PDF_Plakat4.pdf}
  761. \end{figure}
  762. %
  763. Diese wurden weiter bearbeitet und Änderungen in Absprache mit der Projektgruppe vor\-genommen. Auch die Meinung von Professor Dr. Hopf beeinflusste die Plakatgestaltung.\\
  764. Einige Ansätze später und nach etlichem Experimentieren stand die Endauswahl bei diesen nächsten zwei Exemplaren. Besonderen Zeitaufwand nahm die Suche nach einer geeigneten Schrift für die zweite Version in Anspruch. Es gestaltete sich als äußerst schwierig, den künstlerischen leicht verspielten Ausdruck des Plakatlayouts mit der Seriösität des Hochschulcharakters in einer Schrift zu vereinen.
  765. \begin{figure}[H]
  766. \centering
  767. \includegraphics[width=6.5cm]{PDF_Plakat5.pdf}\quad
  768. \includegraphics[width=6.5cm]{PDF_Plakat6.pdf}
  769. \end{figure}
  770. %
  771. Zuletzt fiel die Wahl auf das linke mit blauem Verlauf versehene Plakat. Denn es vermittelt genau das, was der Slogan des Projektes aussagt. Der Fokus liegt auf der Nachrichtenbox, also das was wichtig ist, und der Rest verschwimmt im Hintergrund. \\
  772. Bei den wöchentlichen Treffen wurde im Allgemeinen der Fortschritt des Plakatdesigns analysiert und mit konstruktiver Kritik versehen. Dieses Feedback wurde im Rahmen des Möglichen umgesetzt und so kristallisierte sich die finale Version heraus. Nach gründlichem Finetuning und einer Umformatierung von DIN A3 auf DIN A1 wurde die Endfassung per Mail eingeschickt und schließlich ausgedruckt. Im Anhang ist sie in voller Auflösung aufzufinden.
  773. %
  774. \section*{PowerPoint Präsentation}
  775. \phantomsection\addcontentsline{toc}{section}{PowerPoint Präsentation}
  776. Als sich das Projekt für dieses Semester zum Ende hinneigte, war es an der Zeit sich für die Projektpräsentation vorzubereiten. Hierfür erstellte ich für die Gruppe eine PowerPoint Präsentation. Zunächst besprachen wir Gemeinschaftlich, wie diese ungefähr aussehen sollte. Wir waren alle einstimmig der Meinung, dass wir die Präsentation sehr einfach gestalten wollen, ohne zu lange Textpassagen mit einzubinden. Wir wollten es nämlich verhindern, dass das Publikum von zu vielen Faktoren in der Darstellung abgelenkt werden, zum Beispiel durch lesen von langem Fließtext. Durch den Gebrauch von bildlicher Veranschaulichung, versuchten wir es den Zuschauern zu ermöglichen, das Verständnis zu erleichtern.
  777. Auch hatten wir die Idee, eine fiktive Persona zu erstellen, zu dem der Betrachter einen Bezug finden kann und diese durch das vorgetragene Thema begleitet.
  778. %
  779. \subsection*{Gestaltungselemente}
  780. \phantomsection\addcontentsline{toc}{subsection}{Gestaltungselemente}
  781. Da man jedoch, aus Urheberrechtlichen Gründen, nicht einfach wahllos Charaktere aus dem Internet benutzen kann, suchte ich nach einer Seite, auf der man solche Animationen erstellen kann. Ich bin auch direkt fündig geworden, trotz alle dem war es erstmal schwer etwas zu finden das man auch Gratis verwenden konnte.
  782. Dann bin ich auf die Seite www.Powtoon.com gestoßen. Dort fing ich als nächstes an, unseren sogenannten Klaus in all den Stimmungen, die wir für unsere Präsentation brauchten, zu gestalten. \\
  783. Wir entschieden im Vorfeld als Gruppe, dass sich unser Männchen so wenig wie möglich bewegen soll, aus dem gleichen Grund wie auch oben schon erwähnt und zwar um den Fokus nicht zu sehr darauf zu richten und das Publikum nicht zu sehr abzulenken. Nachdem das erledigt war, war es an der Zeit, die eben erstellten Videos von unserer Persona einzeln so zu kürzen und zu beschneiden, dass man sie gut auf die gewünschte Position einsetzen konnte. Als dieses erledigt wurde, besprach ich weitere Vorgehen einzeln mit den anderen Gruppenmitgliedern. Ich Informierte mich darüber, über welches Thema jeder einzelne vorträgt und wie sie sich ihren Präsentationsteil vorstellen. Diese Wünsche und Vorstellung setzte ich dann dementsprechend in unserer Präsentation um.
  784. Falls sie an dem Ergebnis interessiert sind, haben sie die Möglichkeit sich dies im Git-Repository im Ordner ''Präsentation'' anzusehen.
  785. %
  786. \section*{Weiterverwendung für Infoscreens}
  787. \phantomsection\addcontentsline{toc}{section}{Weiterverwendung für Infoscreens}
  788. Am Mittwoch, den 24.10.2018 fand das Info-Treffen mit Frau Fabi statt, welches von Prof.Dr. \mbox{Matthias} Hopf organisiert wurde. Dieses Treffen diente der Besprechung und Erörterung zukünftiger Nutzungsmöglichkeiten der Info-Screens, das sind geplante und teils bereits montierte Bildschirme zur Information-Bereitstellung für die Hochschulangehörigen auf dem Campusgelände. Frau Fabi ist Angehörige der Fakultät efi und in ihrer Funktion für die Dokumentenlenkung und Webredaktion an dem ''Tag''-basiertem Nachrichten-System unserer Webanwendung interessiert, welches zukünftig auch auf den Bildschirmen dargestellt werden könnten. Sie ließ auch das Interesse von Herr Pöhlau, dem Dekan der Fakultät efi bekunden.\\
  789. Außerdem informierte Frau Fabi uns, die Projektmitglieder und unseren Projektbetreuer, dass die Fakultät efi sicher die restlichen Info-Screens montieren werde und derzeit Besprechungen hinsichtlich weiterer Sicherheitsaspekte zu den Montagebereichen am Laufen sind. Ergänzend würden sie sich eine Softwarelösung wünschen, für die Ansteuerung und ''Befüllung'' der Monitore mit den Daten unserer App. Diese wird jedoch nicht mehr im Rahmen unseres Projektes umgesetzt werden. Die Umsetzung einer Softwarelösung wäre als Idee für ein zukünftiges Projekt denkbar, nach Aussage von Prof.Dr. Matthias Hopf.
  790. %
  791. \section*{Soll/Ist Vergleich auf Basis des Projektplanes}
  792. \phantomsection\addcontentsline{toc}{section}{Soll/Ist Vergleich auf Basis des Projektplanes}
  793. Zu Beginn des Semesters hat die Gruppe zusammen ein Pflichtenheft für den ersten Abschnitt des Projekts erstellt. Ziel war es den Datenzugriff zu ermöglichen und eine Schnittstelle zur Verfügung zu stellen. Es besteht eine Datenbankanbindung über den Server der Technischen Hochschule Nürnberg. Darüber hinaus sollten die folgenden Ziele im Bereich der grafischen Benutzerschnittstelle erreicht werden.\\
  794. Die Oberfläche der PWA sollte über eine Login Möglichkeit, eine Menüleiste, eine Suchoption, ein Profil, eine Filterfunktion, ein Dashboard als Hauptseite, ein Einstellungsmenü, Nachrichtenfelder und ein Formular zum Verfassen einer Nachricht verfügen. Die Login Funktion ist noch nicht implementiert und die Filterfunktion soll im nächsten Semester in das Suchfeld und die Profilanzeige eingebaut werden. Die restlichen Funktionen sind in der PWA grafisch umgesetzt worden. Es sollte zusätzlich ein Abhängigkeitsdiagramm erstellt werden. Allerdings war dies aufgrund fehlender komplexer Zusammenhänge noch nicht notwendig. Für die API-Schnittstelle wurde eine Dokumentation erstellt, um den Überblick zu bewahren. Im Bereich der Anwendungslogik wurde das Routing, die Darstellung der Nachrichten, die Speicherung der Nachrichten in der Datenbank und der LDAP-Login besonders behandelt. Das Routing sowie die Erstellung, Speicherung und Darstellung der Nachrichten funktioniert. Die Login Funktion wurde noch nicht näher betrachtet. Codetests wurden bisher noch keine durchgeführt. Das Rollenkonzept der Nutzer wurde bereits besprochen, allerdings noch nicht umgesetzt.\\
  795. Darüber hinaus gibt noch einige nachrangige Punkte. Dazu gehört die Festlegung des Anwendungsnamens, die Erstellung eines Logos, das UX-Konzept, das Filtern der Datenbankabrufe, das Festlegen der Anzeigekriterien und die Funktion einem Kanal zu folgen. Wie oben bereits erwähnt wird das Filtern der Abrufe im nächsten Semester umgesetzt. Auch die Anzeigekriterien und die Möglichkeit einem Kanal zu folgen wird erst zu einem späteren Zeitpunkt möglich sein.
  796. %
  797. \section*{Fazit}
  798. \phantomsection\addcontentsline{toc}{section}{Fazit}
  799. Im ersten Projektabschnitt ist viel Zeit ist in der Konzeptionsphase vergangen, da es ein sehr iterativer Prozess ist und deshalb mehrere Durchläufe bis zum Endentwurf nötig waren. Verschiedene Meinungen mussten mit einbezogen werden. Doch ein gutes Konzept erspart am Ende viel Änderungsarbeit. \\
  800. Für die weitere Erarbeitung der App haben sich die Projektmitglieder auf die Bereiche Front\-end, Datenbank, Sever und Service-Worker aufgeteilt. Diese Aufgaben\-verteilung hat gut funktioniert. Trotzdem ist unser Team sehr oft ins Stocken geraten, da sich die Aufgaben\-bereiche gerade am Anfang oft überschnitten haben. Hinzukommt, dass wir insgesamt \mbox{wenig} Erfahrung in diesen Bereichen hatten. Wir haben uns deshalb zu Beginn intensiv in die Themen eingearbeitet, was länger dauerte als erwartet. Zusätzlich mussten wir auf \mbox{fachliche} Unterstützung zurückgreifen.\\\\
  801. Alles in einem kann man trotzdem von einem erfolgreichen Projektabschnitt sprechen. Ein erster funktionierender Prototyp ist fertig und kann im nächsten Zug erweitert werden. Parallele individuelle Arbeiten sind einfacher möglich, da das Grundgerüst steht. Das Team hat sich in die Materie eingearbeitet und kann im zweiten Projektabschnitt schneller durchstarten. Die Idee und das Konzept für die App sind soweit ausgereift, sodass diese nur noch umgesetzt werden müssen. Abläufe innerhalb der Gruppe sind klarer geworden, allerdings gibt es noch einiges strukturelle Abläufe, die im zweiten Projektabschnitt weiter optimiert werden können.
  802. %
  803. % Literaturverzeichnis
  804. \newpage
  805. \section*{Quellen- und Literaturverzeichnis}
  806. \phantomsection\addcontentsline{toc}{section}{Quellen- und Literaturverzeichnis}
  807. \medskip
  808. %\nocite{*} % Alle Inhalte zur Ausgabe markieren, nicht nur jene die zitiert wurden.
  809. {\setstretch{1.0}%
  810. \printbibliography[heading=none]}
  811. %
  812. \end{document}