diff --git a/android-app/.gitignore b/android-app/.gitignore
new file mode 100644
index 0000000..aa724b7
--- /dev/null
+++ b/android-app/.gitignore
@@ -0,0 +1,15 @@
+*.iml
+.gradle
+/local.properties
+/.idea/caches
+/.idea/libraries
+/.idea/modules.xml
+/.idea/workspace.xml
+/.idea/navEditor.xml
+/.idea/assetWizardSettings.xml
+.DS_Store
+/build
+/captures
+.externalNativeBuild
+.cxx
+local.properties
diff --git a/android-app/.idea/.gitignore b/android-app/.idea/.gitignore
new file mode 100644
index 0000000..26d3352
--- /dev/null
+++ b/android-app/.idea/.gitignore
@@ -0,0 +1,3 @@
+# Default ignored files
+/shelf/
+/workspace.xml
diff --git a/android-app/.idea/.name b/android-app/.idea/.name
new file mode 100644
index 0000000..a781ee6
--- /dev/null
+++ b/android-app/.idea/.name
@@ -0,0 +1 @@
+Futharkboard
\ No newline at end of file
diff --git a/android-app/.idea/compiler.xml b/android-app/.idea/compiler.xml
new file mode 100644
index 0000000..fb7f4a8
--- /dev/null
+++ b/android-app/.idea/compiler.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/.idea/gradle.xml b/android-app/.idea/gradle.xml
new file mode 100644
index 0000000..526b4c2
--- /dev/null
+++ b/android-app/.idea/gradle.xml
@@ -0,0 +1,20 @@
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/.idea/misc.xml b/android-app/.idea/misc.xml
new file mode 100644
index 0000000..495fe5d
--- /dev/null
+++ b/android-app/.idea/misc.xml
@@ -0,0 +1,32 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/app/.gitignore b/android-app/app/.gitignore
new file mode 100644
index 0000000..42afabf
--- /dev/null
+++ b/android-app/app/.gitignore
@@ -0,0 +1 @@
+/build
\ No newline at end of file
diff --git a/android-app/app/build.gradle b/android-app/app/build.gradle
new file mode 100644
index 0000000..021e91c
--- /dev/null
+++ b/android-app/app/build.gradle
@@ -0,0 +1,41 @@
+plugins {
+ id 'com.android.application'
+}
+
+android {
+ compileSdk 31
+
+ defaultConfig {
+ applicationId "de.drmaxnix.futharkboard"
+ minSdk 24
+ targetSdk 31
+ versionCode 1
+ versionName "1.0.0"
+
+ testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
+ }
+
+ buildTypes {
+ release {
+ minifyEnabled false
+ proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
+ }
+ }
+ compileOptions {
+ sourceCompatibility JavaVersion.VERSION_1_8
+ targetCompatibility JavaVersion.VERSION_1_8
+ }
+ buildFeatures {
+ viewBinding true
+ }
+}
+
+dependencies {
+
+ implementation 'androidx.appcompat:appcompat:1.3.1'
+ implementation 'com.google.android.material:material:1.4.0'
+ implementation 'androidx.constraintlayout:constraintlayout:2.1.1'
+ testImplementation 'junit:junit:4.+'
+ androidTestImplementation 'androidx.test.ext:junit:1.1.3'
+ androidTestImplementation 'androidx.test.espresso:espresso-core:3.4.0'
+}
\ No newline at end of file
diff --git a/android-app/app/proguard-rules.pro b/android-app/app/proguard-rules.pro
new file mode 100644
index 0000000..481bb43
--- /dev/null
+++ b/android-app/app/proguard-rules.pro
@@ -0,0 +1,21 @@
+# Add project specific ProGuard rules here.
+# You can control the set of applied configuration files using the
+# proguardFiles setting in build.gradle.
+#
+# For more details, see
+# http://developer.android.com/guide/developing/tools/proguard.html
+
+# If your project uses WebView with JS, uncomment the following
+# and specify the fully qualified class name to the JavaScript interface
+# class:
+#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
+# public *;
+#}
+
+# Uncomment this to preserve the line number information for
+# debugging stack traces.
+#-keepattributes SourceFile,LineNumberTable
+
+# If you keep the line number information, uncomment this to
+# hide the original source file name.
+#-renamesourcefileattribute SourceFile
\ No newline at end of file
diff --git a/android-app/app/release/app-release.apk b/android-app/app/release/app-release.apk
new file mode 100644
index 0000000..a33e5f3
Binary files /dev/null and b/android-app/app/release/app-release.apk differ
diff --git a/android-app/app/release/output-metadata.json b/android-app/app/release/output-metadata.json
new file mode 100644
index 0000000..6e4ec24
--- /dev/null
+++ b/android-app/app/release/output-metadata.json
@@ -0,0 +1,20 @@
+{
+ "version": 3,
+ "artifactType": {
+ "type": "APK",
+ "kind": "Directory"
+ },
+ "applicationId": "de.drmaxnix.futharkboard",
+ "variantName": "release",
+ "elements": [
+ {
+ "type": "SINGLE",
+ "filters": [],
+ "attributes": [],
+ "versionCode": 1,
+ "versionName": "1.0.0",
+ "outputFile": "app-release.apk"
+ }
+ ],
+ "elementType": "File"
+}
\ No newline at end of file
diff --git a/android-app/app/src/main/AndroidManifest.xml b/android-app/app/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..a93f2a6
--- /dev/null
+++ b/android-app/app/src/main/AndroidManifest.xml
@@ -0,0 +1,35 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/app/src/main/ic_launcher-playstore.png b/android-app/app/src/main/ic_launcher-playstore.png
new file mode 100644
index 0000000..17dccf5
Binary files /dev/null and b/android-app/app/src/main/ic_launcher-playstore.png differ
diff --git a/android-app/app/src/main/icon-playstore.png b/android-app/app/src/main/icon-playstore.png
new file mode 100644
index 0000000..17dccf5
Binary files /dev/null and b/android-app/app/src/main/icon-playstore.png differ
diff --git a/android-app/app/src/main/java/de/drmaxnix/futharkboard/IME.java b/android-app/app/src/main/java/de/drmaxnix/futharkboard/IME.java
new file mode 100644
index 0000000..c374f25
--- /dev/null
+++ b/android-app/app/src/main/java/de/drmaxnix/futharkboard/IME.java
@@ -0,0 +1,75 @@
+package de.drmaxnix.futharkboard;
+
+import android.inputmethodservice.InputMethodService;
+import android.inputmethodservice.Keyboard;
+import android.inputmethodservice.KeyboardView;
+import android.text.TextUtils;
+import android.view.View;
+import android.view.inputmethod.InputConnection;
+
+public class IME extends InputMethodService implements KeyboardView.OnKeyboardActionListener {
+ @Override
+ public View onCreateInputView(){
+ // APPLY KEYBOARD LAYOUT //
+ KeyboardView keyboardView = (KeyboardView)getLayoutInflater().inflate(R.layout.keyboard_view, null);
+ Keyboard keyboard = new Keyboard(this, R.xml.futhark);
+ keyboardView.setKeyboard(keyboard);
+ keyboardView.setOnKeyboardActionListener(this);
+ return keyboardView;
+ }
+
+ @Override
+ public void onKey(int primaryCode, int[] keyCodes){
+ // GET INPUT-CONNECTION //
+ InputConnection input_connection = getCurrentInputConnection();
+
+ //make sure we got something
+ if(input_connection == null){
+ return;
+ }
+
+
+ // HANDLE INPUT //
+ switch(primaryCode){
+ case Keyboard.KEYCODE_DELETE:
+ // DELETE KEY //
+ //check if some text is selected
+ CharSequence selectedText = input_connection.getSelectedText(0);
+ if (TextUtils.isEmpty(selectedText)) {
+ //no selection, delete previous character
+ input_connection.deleteSurroundingText(1, 0);
+
+ } else {
+ //delete selection
+ input_connection.commitText("", 1);
+ }
+ break;
+
+ default:
+ // NORMAL CHARACTER INPUT //
+ char code = (char)primaryCode;
+ input_connection.commitText(String.valueOf(code), 1);
+ }
+ }
+
+ @Override
+ public void onPress(int primaryCode) { }
+
+ @Override
+ public void onRelease(int primaryCode) { }
+
+ @Override
+ public void onText(CharSequence text) { }
+
+ @Override
+ public void swipeLeft() { }
+
+ @Override
+ public void swipeRight() { }
+
+ @Override
+ public void swipeDown() { }
+
+ @Override
+ public void swipeUp() { }
+}
diff --git a/android-app/app/src/main/java/de/drmaxnix/futharkboard/Main.java b/android-app/app/src/main/java/de/drmaxnix/futharkboard/Main.java
new file mode 100644
index 0000000..dde04f8
--- /dev/null
+++ b/android-app/app/src/main/java/de/drmaxnix/futharkboard/Main.java
@@ -0,0 +1,52 @@
+package de.drmaxnix.futharkboard;
+
+import android.content.pm.PackageInfo;
+import android.content.pm.PackageManager;
+import android.os.Bundle;
+import android.widget.TextView;
+import com.google.android.material.tabs.TabLayout;
+import androidx.viewpager.widget.ViewPager;
+import androidx.appcompat.app.AppCompatActivity;
+import de.drmaxnix.futharkboard.databinding.ActivityMainBinding;
+import de.drmaxnix.futharkboard.ui.main.SectionsPagerAdapter;
+
+public class Main extends AppCompatActivity {
+ private ActivityMainBinding binding;
+ private TabLayout tabs;
+
+ @Override
+ protected void onCreate(Bundle savedInstanceState){
+ super.onCreate(savedInstanceState);
+
+ binding = ActivityMainBinding.inflate(getLayoutInflater());
+ setContentView(binding.getRoot());
+
+
+ // SET VERSION TEXT //
+ try {
+ //get version of the app
+ PackageInfo pInfo = this.getPackageManager().getPackageInfo(this.getPackageName(), 0);
+ String version = pInfo.versionName;
+
+ //set text
+ TextView version_text = (TextView)findViewById(R.id.version);
+ version_text.setText(version);
+
+ } catch(PackageManager.NameNotFoundException e){
+ e.printStackTrace();
+ }
+
+
+ // INITIALIZE TAB LAYOUT //
+ //get new sections-pager-adapter
+ SectionsPagerAdapter sectionsPagerAdapter = new SectionsPagerAdapter(this, getSupportFragmentManager());
+
+ //connect with view-pager
+ ViewPager viewPager = binding.viewPager;
+ viewPager.setAdapter(sectionsPagerAdapter);
+
+ //connect with tab-selector
+ tabs = binding.tabs;
+ tabs.setupWithViewPager(viewPager);
+ }
+}
\ No newline at end of file
diff --git a/android-app/app/src/main/java/de/drmaxnix/futharkboard/ui/main/SectionsPagerAdapter.java b/android-app/app/src/main/java/de/drmaxnix/futharkboard/ui/main/SectionsPagerAdapter.java
new file mode 100644
index 0000000..51f8ad2
--- /dev/null
+++ b/android-app/app/src/main/java/de/drmaxnix/futharkboard/ui/main/SectionsPagerAdapter.java
@@ -0,0 +1,42 @@
+package de.drmaxnix.futharkboard.ui.main;
+
+import android.content.Context;
+import androidx.annotation.Nullable;
+import androidx.annotation.StringRes;
+import androidx.fragment.app.Fragment;
+import androidx.fragment.app.FragmentManager;
+import androidx.fragment.app.FragmentPagerAdapter;
+import de.drmaxnix.futharkboard.R;
+
+public class SectionsPagerAdapter extends FragmentPagerAdapter {
+ private final Context context;
+
+ public SectionsPagerAdapter(Context context, FragmentManager fm){
+ super(fm);
+ this.context = context;
+ }
+
+ @Override
+ public Fragment getItem(int position){
+ switch(position){
+ case 0:
+ return new Start();
+ }
+
+ return null;
+ }
+
+ @Nullable
+ @Override
+ public CharSequence getPageTitle(int position){
+ @StringRes
+ final int[] TAB_TITLES = new int[]{R.string.tab_name_start};
+
+ return context.getResources().getString(TAB_TITLES[position]);
+ }
+
+ @Override
+ public int getCount(){
+ return 1;
+ }
+}
\ No newline at end of file
diff --git a/android-app/app/src/main/java/de/drmaxnix/futharkboard/ui/main/Start.java b/android-app/app/src/main/java/de/drmaxnix/futharkboard/ui/main/Start.java
new file mode 100644
index 0000000..910007d
--- /dev/null
+++ b/android-app/app/src/main/java/de/drmaxnix/futharkboard/ui/main/Start.java
@@ -0,0 +1,170 @@
+package de.drmaxnix.futharkboard.ui.main;
+
+import android.content.Context;
+import android.content.Intent;
+import android.content.pm.PackageInfo;
+import android.content.pm.PackageManager;
+import android.net.Uri;
+import android.os.Bundle;
+import android.provider.Settings;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.view.ViewGroup;
+import android.view.inputmethod.InputMethodInfo;
+import android.view.inputmethod.InputMethodManager;
+import android.widget.Button;
+import androidx.annotation.NonNull;
+import androidx.annotation.Nullable;
+import androidx.fragment.app.Fragment;
+
+import de.drmaxnix.futharkboard.R;
+
+public class Start extends Fragment {
+ private View view;
+
+ public Start(){
+
+ }
+
+ @Override
+ public void onCreate(Bundle savedInstanceState){
+ super.onCreate(savedInstanceState);
+ }
+
+ @Override
+ public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState){
+ // GENERATE VIEW //
+ View view = inflater.inflate(R.layout.fragment_start, container, false);
+
+ //save
+ this.view = view;
+
+
+ // OPEN-FUTHARKBOARD-WEBSITE-BUTTON ONCLICK CALLBACK //
+ Button open_futharkboard_website = (Button)view.findViewById(R.id.open_futharkboard_website);
+ open_futharkboard_website.setOnClickListener(new View.OnClickListener(){
+ public void onClick(View v){
+ view.getContext().startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse("https://futharkboard.drmaxnix.de")));
+ }
+ });
+
+
+ // OPEN-IME-SETTINGS-BUTTON ONCLICK CALLBACK //
+ Button open_ime_settings = (Button)view.findViewById(R.id.open_ime_settings);
+ open_ime_settings.setOnClickListener(new View.OnClickListener(){
+ public void onClick(View v){
+ view.getContext().startActivity(new Intent(Settings.ACTION_INPUT_METHOD_SETTINGS));
+ }
+ });
+
+
+ // BUGREPORT-MAIL-BUTTON ONCLICK CALLBACK //
+ Button bugreport_mail = (Button)view.findViewById(R.id.bugreport_mail);
+ bugreport_mail.setOnClickListener(new View.OnClickListener(){
+ public void onClick(View v){
+ try {
+ // GET APP'S VERSION //
+ //get version of the app
+ PackageInfo pInfo = view.getContext().getPackageManager().getPackageInfo(view.getContext().getPackageName(), 0);
+ String version = pInfo.versionName;
+
+
+ // SEND MAIL INTENT //
+ Intent intent = new Intent(Intent.ACTION_SENDTO);
+ intent.setData(Uri.fromParts("mailto","futharkboard@drmaxnix.de", null));
+
+ //set extra data
+ intent.putExtra(Intent.EXTRA_SUBJECT, "Bug/suggestion for FutharkBoard v" + version);
+
+ //start intent
+ view.getContext().startActivity(intent);
+
+ } catch(PackageManager.NameNotFoundException e){
+ e.printStackTrace();
+ }
+ }
+ });
+
+
+ // OPEN-GH-BUTTON ONCLICK CALLBACK //
+ Button open_gh = (Button)view.findViewById(R.id.open_gh);
+ open_gh.setOnClickListener(new View.OnClickListener(){
+ public void onClick(View v){
+ view.getContext().startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse("https://github.com/DrMaxNix/futharkboard")));
+ }
+ });
+
+
+ // OPEN-GH-LICENSE-BUTTON ONCLICK CALLBACK //
+ Button open_gh_license = (Button)view.findViewById(R.id.open_gh_license);
+ open_gh_license.setOnClickListener(new View.OnClickListener(){
+ public void onClick(View v){
+ view.getContext().startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse("https://github.com/DrMaxNix/futharkboard/blob/main/LICENSE")));
+ }
+ });
+
+
+ // RETURN GENERATED VIEW //
+ return view;
+ }
+
+ @Override
+ public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState){
+
+ }
+
+ @Override
+ public void onStart(){
+ super.onStart();
+
+
+ // RELOAD ENABLED-STATE OF IME //
+ futhark_board_is_enabled_reload();
+ }
+
+
+
+ /*
+ Reload all view related to the futhark_board_is_enabled-state
+ */
+ private void futhark_board_is_enabled_reload(){
+ boolean futhark_board_is_enabled = futhark_board_is_enabled();
+
+ //get related views
+ View status_not_enabled = view.findViewById(R.id.status_not_enabled);
+ View status_enabled = view.findViewById(R.id.status_enabled);
+
+ //change status banner
+ if(!futhark_board_is_enabled){
+ //not enabled, show info on how to enable
+ status_not_enabled.setVisibility(View.VISIBLE);
+ status_enabled.setVisibility(View.GONE);
+
+ } else {
+ //enabled, show that it's enabled
+ status_not_enabled.setVisibility(View.GONE);
+ status_enabled.setVisibility(View.VISIBLE);
+ }
+ }
+
+ /*
+ Check if the futharkboard is enabled in settings
+ */
+ private boolean futhark_board_is_enabled(){
+ // GET IMM //
+ InputMethodManager imm = (InputMethodManager) this.getContext().getSystemService(Context.INPUT_METHOD_SERVICE);
+
+
+ // SEARCH //
+ for(InputMethodInfo imi : imm.getEnabledInputMethodList()){
+ if(imi.getServiceName().equals("de.drmaxnix.futharkboard.IME")){
+ //found!
+ return true;
+ }
+ }
+
+
+ // NOTHING FOUND //
+ return false;
+ }
+}
\ No newline at end of file
diff --git a/android-app/app/src/main/res/drawable/empty.xml b/android-app/app/src/main/res/drawable/empty.xml
new file mode 100644
index 0000000..41ea0e9
--- /dev/null
+++ b/android-app/app/src/main/res/drawable/empty.xml
@@ -0,0 +1,8 @@
+
+
+
diff --git a/android-app/app/src/main/res/drawable/key_del.xml b/android-app/app/src/main/res/drawable/key_del.xml
new file mode 100644
index 0000000..17ece2d
--- /dev/null
+++ b/android-app/app/src/main/res/drawable/key_del.xml
@@ -0,0 +1,20 @@
+
+
+
+
diff --git a/android-app/app/src/main/res/drawable/key_enter.xml b/android-app/app/src/main/res/drawable/key_enter.xml
new file mode 100644
index 0000000..33a9e4f
--- /dev/null
+++ b/android-app/app/src/main/res/drawable/key_enter.xml
@@ -0,0 +1,13 @@
+
+
+
diff --git a/android-app/app/src/main/res/drawable/key_space.xml b/android-app/app/src/main/res/drawable/key_space.xml
new file mode 100644
index 0000000..2697bcb
--- /dev/null
+++ b/android-app/app/src/main/res/drawable/key_space.xml
@@ -0,0 +1,13 @@
+
+
+
diff --git a/android-app/app/src/main/res/layout/activity_main.xml b/android-app/app/src/main/res/layout/activity_main.xml
new file mode 100644
index 0000000..e026f92
--- /dev/null
+++ b/android-app/app/src/main/res/layout/activity_main.xml
@@ -0,0 +1,71 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/layout/fragment_start.xml b/android-app/app/src/main/res/layout/fragment_start.xml
new file mode 100644
index 0000000..d44dd90
--- /dev/null
+++ b/android-app/app/src/main/res/layout/fragment_start.xml
@@ -0,0 +1,148 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/layout/key_preview.xml b/android-app/app/src/main/res/layout/key_preview.xml
new file mode 100644
index 0000000..8520806
--- /dev/null
+++ b/android-app/app/src/main/res/layout/key_preview.xml
@@ -0,0 +1,10 @@
+
+
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/layout/keyboard_view.xml b/android-app/app/src/main/res/layout/keyboard_view.xml
new file mode 100644
index 0000000..451a74a
--- /dev/null
+++ b/android-app/app/src/main/res/layout/keyboard_view.xml
@@ -0,0 +1,10 @@
+
+
+
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/mipmap-anydpi-v26/icon.xml b/android-app/app/src/main/res/mipmap-anydpi-v26/icon.xml
new file mode 100644
index 0000000..3a8ac85
--- /dev/null
+++ b/android-app/app/src/main/res/mipmap-anydpi-v26/icon.xml
@@ -0,0 +1,5 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/mipmap-anydpi-v26/icon_round.xml b/android-app/app/src/main/res/mipmap-anydpi-v26/icon_round.xml
new file mode 100644
index 0000000..3a8ac85
--- /dev/null
+++ b/android-app/app/src/main/res/mipmap-anydpi-v26/icon_round.xml
@@ -0,0 +1,5 @@
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/mipmap-hdpi/icon.png b/android-app/app/src/main/res/mipmap-hdpi/icon.png
new file mode 100644
index 0000000..9dcd77b
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-hdpi/icon.png differ
diff --git a/android-app/app/src/main/res/mipmap-hdpi/icon_foreground.png b/android-app/app/src/main/res/mipmap-hdpi/icon_foreground.png
new file mode 100644
index 0000000..f0ab954
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-hdpi/icon_foreground.png differ
diff --git a/android-app/app/src/main/res/mipmap-hdpi/icon_round.png b/android-app/app/src/main/res/mipmap-hdpi/icon_round.png
new file mode 100644
index 0000000..03fd5a5
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-hdpi/icon_round.png differ
diff --git a/android-app/app/src/main/res/mipmap-mdpi/icon.png b/android-app/app/src/main/res/mipmap-mdpi/icon.png
new file mode 100644
index 0000000..cabf0d9
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-mdpi/icon.png differ
diff --git a/android-app/app/src/main/res/mipmap-mdpi/icon_foreground.png b/android-app/app/src/main/res/mipmap-mdpi/icon_foreground.png
new file mode 100644
index 0000000..d473ea2
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-mdpi/icon_foreground.png differ
diff --git a/android-app/app/src/main/res/mipmap-mdpi/icon_round.png b/android-app/app/src/main/res/mipmap-mdpi/icon_round.png
new file mode 100644
index 0000000..29800d3
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-mdpi/icon_round.png differ
diff --git a/android-app/app/src/main/res/mipmap-xhdpi/icon.png b/android-app/app/src/main/res/mipmap-xhdpi/icon.png
new file mode 100644
index 0000000..37fdf7a
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-xhdpi/icon.png differ
diff --git a/android-app/app/src/main/res/mipmap-xhdpi/icon_foreground.png b/android-app/app/src/main/res/mipmap-xhdpi/icon_foreground.png
new file mode 100644
index 0000000..6ce1e70
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-xhdpi/icon_foreground.png differ
diff --git a/android-app/app/src/main/res/mipmap-xhdpi/icon_round.png b/android-app/app/src/main/res/mipmap-xhdpi/icon_round.png
new file mode 100644
index 0000000..6dc65e8
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-xhdpi/icon_round.png differ
diff --git a/android-app/app/src/main/res/mipmap-xxhdpi/icon.png b/android-app/app/src/main/res/mipmap-xxhdpi/icon.png
new file mode 100644
index 0000000..13cb17d
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-xxhdpi/icon.png differ
diff --git a/android-app/app/src/main/res/mipmap-xxhdpi/icon_foreground.png b/android-app/app/src/main/res/mipmap-xxhdpi/icon_foreground.png
new file mode 100644
index 0000000..91c48cd
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-xxhdpi/icon_foreground.png differ
diff --git a/android-app/app/src/main/res/mipmap-xxhdpi/icon_round.png b/android-app/app/src/main/res/mipmap-xxhdpi/icon_round.png
new file mode 100644
index 0000000..bbbd8b7
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-xxhdpi/icon_round.png differ
diff --git a/android-app/app/src/main/res/mipmap-xxxhdpi/icon.png b/android-app/app/src/main/res/mipmap-xxxhdpi/icon.png
new file mode 100644
index 0000000..f8acbce
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-xxxhdpi/icon.png differ
diff --git a/android-app/app/src/main/res/mipmap-xxxhdpi/icon_foreground.png b/android-app/app/src/main/res/mipmap-xxxhdpi/icon_foreground.png
new file mode 100644
index 0000000..4f08b6c
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-xxxhdpi/icon_foreground.png differ
diff --git a/android-app/app/src/main/res/mipmap-xxxhdpi/icon_round.png b/android-app/app/src/main/res/mipmap-xxxhdpi/icon_round.png
new file mode 100644
index 0000000..800ddcb
Binary files /dev/null and b/android-app/app/src/main/res/mipmap-xxxhdpi/icon_round.png differ
diff --git a/android-app/app/src/main/res/values-night/themes.xml b/android-app/app/src/main/res/values-night/themes.xml
new file mode 100644
index 0000000..0bbbe2d
--- /dev/null
+++ b/android-app/app/src/main/res/values-night/themes.xml
@@ -0,0 +1,39 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/values-w820dp/dimens.xml b/android-app/app/src/main/res/values-w820dp/dimens.xml
new file mode 100644
index 0000000..ce6f7e2
--- /dev/null
+++ b/android-app/app/src/main/res/values-w820dp/dimens.xml
@@ -0,0 +1,6 @@
+
+
+ 64dp
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/values/colors.xml b/android-app/app/src/main/res/values/colors.xml
new file mode 100644
index 0000000..bf8c829
--- /dev/null
+++ b/android-app/app/src/main/res/values/colors.xml
@@ -0,0 +1,10 @@
+
+
+ #98d7b9
+ #6daa8e
+ #246148
+ #FF000000
+ #DDFFFFFF
+ #FF444444
+ #FFFFFFFF
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/values/dimens.xml b/android-app/app/src/main/res/values/dimens.xml
new file mode 100644
index 0000000..3686d2f
--- /dev/null
+++ b/android-app/app/src/main/res/values/dimens.xml
@@ -0,0 +1,8 @@
+
+
+ 16dp
+ 16dp
+ 16dp
+ 16dp
+ 8dp
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/values/strings.xml b/android-app/app/src/main/res/values/strings.xml
new file mode 100644
index 0000000..6fb33a0
--- /dev/null
+++ b/android-app/app/src/main/res/values/strings.xml
@@ -0,0 +1,22 @@
+
+ FutharkBoard
+ Start
+
+ Made for all the awesome people who have wished for an App like this for ages.\n~ Kim <33
+ FutharkBoard Website
+
+ Status
+ FutharkBoard is not enabled in your System Settings
+ Open Input Settings
+ FutharkBoard is enabled
+
+
+ Bug Report
+ If you have stumbled upon a mistake or have some suggestions to make FutharkBoard better, feel free to contact me via futharkboard@drmaxnix.de.
+ Write a Mail
+
+ FOSS
+ This App is Free and OpenSource Software. Anyone can get this App for free and review the sourcecode as they like. The app is released under the MIT license by DrMaxNix.
+ View on GitHub
+ View License
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/values/themes.xml b/android-app/app/src/main/res/values/themes.xml
new file mode 100644
index 0000000..40da00b
--- /dev/null
+++ b/android-app/app/src/main/res/values/themes.xml
@@ -0,0 +1,45 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/android-app/app/src/main/res/xml/futhark.xml b/android-app/app/src/main/res/xml/futhark.xml
new file mode 100644
index 0000000..7e59437
--- /dev/null
+++ b/android-app/app/src/main/res/xml/futhark.xml
@@ -0,0 +1,50 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/android-app/app/src/main/res/xml/method.xml b/android-app/app/src/main/res/xml/method.xml
new file mode 100644
index 0000000..48ecba7
--- /dev/null
+++ b/android-app/app/src/main/res/xml/method.xml
@@ -0,0 +1,8 @@
+
+
+
+
+
diff --git a/android-app/build.gradle b/android-app/build.gradle
new file mode 100644
index 0000000..9eb1ea6
--- /dev/null
+++ b/android-app/build.gradle
@@ -0,0 +1,17 @@
+// Top-level build file where you can add configuration options common to all sub-projects/modules.
+buildscript {
+ repositories {
+ google()
+ mavenCentral()
+ }
+ dependencies {
+ classpath "com.android.tools.build:gradle:7.0.1"
+
+ // NOTE: Do not place your application dependencies here; they belong
+ // in the individual module build.gradle files
+ }
+}
+
+task clean(type: Delete) {
+ delete rootProject.buildDir
+}
\ No newline at end of file
diff --git a/android-app/gradle.properties b/android-app/gradle.properties
new file mode 100644
index 0000000..52f5917
--- /dev/null
+++ b/android-app/gradle.properties
@@ -0,0 +1,19 @@
+# Project-wide Gradle settings.
+# IDE (e.g. Android Studio) users:
+# Gradle settings configured through the IDE *will override*
+# any settings specified in this file.
+# For more details on how to configure your build environment visit
+# http://www.gradle.org/docs/current/userguide/build_environment.html
+# Specifies the JVM arguments used for the daemon process.
+# The setting is particularly useful for tweaking memory settings.
+org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8
+# When configured, Gradle will run in incubating parallel mode.
+# This option should only be used with decoupled projects. More details, visit
+# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
+# org.gradle.parallel=true
+# AndroidX package structure to make it clearer which packages are bundled with the
+# Android operating system, and which are packaged with your app"s APK
+# https://developer.android.com/topic/libraries/support-library/androidx-rn
+android.useAndroidX=true
+# Automatically convert third-party libraries to use AndroidX
+android.enableJetifier=true
\ No newline at end of file
diff --git a/android-app/gradle/wrapper/gradle-wrapper.jar b/android-app/gradle/wrapper/gradle-wrapper.jar
new file mode 100644
index 0000000..e708b1c
Binary files /dev/null and b/android-app/gradle/wrapper/gradle-wrapper.jar differ
diff --git a/android-app/gradle/wrapper/gradle-wrapper.properties b/android-app/gradle/wrapper/gradle-wrapper.properties
new file mode 100644
index 0000000..1bddec8
--- /dev/null
+++ b/android-app/gradle/wrapper/gradle-wrapper.properties
@@ -0,0 +1,6 @@
+#Sat Oct 30 15:20:34 CEST 2021
+distributionBase=GRADLE_USER_HOME
+distributionUrl=https\://services.gradle.org/distributions/gradle-7.0.2-bin.zip
+distributionPath=wrapper/dists
+zipStorePath=wrapper/dists
+zipStoreBase=GRADLE_USER_HOME
diff --git a/android-app/gradlew b/android-app/gradlew
new file mode 100644
index 0000000..4f906e0
--- /dev/null
+++ b/android-app/gradlew
@@ -0,0 +1,185 @@
+#!/usr/bin/env sh
+
+#
+# Copyright 2015 the original author or authors.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# https://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+##############################################################################
+##
+## Gradle start up script for UN*X
+##
+##############################################################################
+
+# Attempt to set APP_HOME
+# Resolve links: $0 may be a link
+PRG="$0"
+# Need this for relative symlinks.
+while [ -h "$PRG" ] ; do
+ ls=`ls -ld "$PRG"`
+ link=`expr "$ls" : '.*-> \(.*\)$'`
+ if expr "$link" : '/.*' > /dev/null; then
+ PRG="$link"
+ else
+ PRG=`dirname "$PRG"`"/$link"
+ fi
+done
+SAVED="`pwd`"
+cd "`dirname \"$PRG\"`/" >/dev/null
+APP_HOME="`pwd -P`"
+cd "$SAVED" >/dev/null
+
+APP_NAME="Gradle"
+APP_BASE_NAME=`basename "$0"`
+
+# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
+DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
+
+# Use the maximum available, or set MAX_FD != -1 to use that value.
+MAX_FD="maximum"
+
+warn () {
+ echo "$*"
+}
+
+die () {
+ echo
+ echo "$*"
+ echo
+ exit 1
+}
+
+# OS specific support (must be 'true' or 'false').
+cygwin=false
+msys=false
+darwin=false
+nonstop=false
+case "`uname`" in
+ CYGWIN* )
+ cygwin=true
+ ;;
+ Darwin* )
+ darwin=true
+ ;;
+ MINGW* )
+ msys=true
+ ;;
+ NONSTOP* )
+ nonstop=true
+ ;;
+esac
+
+CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
+
+
+# Determine the Java command to use to start the JVM.
+if [ -n "$JAVA_HOME" ] ; then
+ if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
+ # IBM's JDK on AIX uses strange locations for the executables
+ JAVACMD="$JAVA_HOME/jre/sh/java"
+ else
+ JAVACMD="$JAVA_HOME/bin/java"
+ fi
+ if [ ! -x "$JAVACMD" ] ; then
+ die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+ fi
+else
+ JAVACMD="java"
+ which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+fi
+
+# Increase the maximum file descriptors if we can.
+if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
+ MAX_FD_LIMIT=`ulimit -H -n`
+ if [ $? -eq 0 ] ; then
+ if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
+ MAX_FD="$MAX_FD_LIMIT"
+ fi
+ ulimit -n $MAX_FD
+ if [ $? -ne 0 ] ; then
+ warn "Could not set maximum file descriptor limit: $MAX_FD"
+ fi
+ else
+ warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
+ fi
+fi
+
+# For Darwin, add options to specify how the application appears in the dock
+if $darwin; then
+ GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
+fi
+
+# For Cygwin or MSYS, switch paths to Windows format before running java
+if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then
+ APP_HOME=`cygpath --path --mixed "$APP_HOME"`
+ CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
+
+ JAVACMD=`cygpath --unix "$JAVACMD"`
+
+ # We build the pattern for arguments to be converted via cygpath
+ ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
+ SEP=""
+ for dir in $ROOTDIRSRAW ; do
+ ROOTDIRS="$ROOTDIRS$SEP$dir"
+ SEP="|"
+ done
+ OURCYGPATTERN="(^($ROOTDIRS))"
+ # Add a user-defined pattern to the cygpath arguments
+ if [ "$GRADLE_CYGPATTERN" != "" ] ; then
+ OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
+ fi
+ # Now convert the arguments - kludge to limit ourselves to /bin/sh
+ i=0
+ for arg in "$@" ; do
+ CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
+ CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
+
+ if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
+ eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
+ else
+ eval `echo args$i`="\"$arg\""
+ fi
+ i=`expr $i + 1`
+ done
+ case $i in
+ 0) set -- ;;
+ 1) set -- "$args0" ;;
+ 2) set -- "$args0" "$args1" ;;
+ 3) set -- "$args0" "$args1" "$args2" ;;
+ 4) set -- "$args0" "$args1" "$args2" "$args3" ;;
+ 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
+ 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
+ 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
+ 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
+ 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
+ esac
+fi
+
+# Escape application args
+save () {
+ for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
+ echo " "
+}
+APP_ARGS=`save "$@"`
+
+# Collect all arguments for the java command, following the shell quoting and substitution rules
+eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
+
+exec "$JAVACMD" "$@"
diff --git a/android-app/gradlew.bat b/android-app/gradlew.bat
new file mode 100644
index 0000000..ac1b06f
--- /dev/null
+++ b/android-app/gradlew.bat
@@ -0,0 +1,89 @@
+@rem
+@rem Copyright 2015 the original author or authors.
+@rem
+@rem Licensed under the Apache License, Version 2.0 (the "License");
+@rem you may not use this file except in compliance with the License.
+@rem You may obtain a copy of the License at
+@rem
+@rem https://www.apache.org/licenses/LICENSE-2.0
+@rem
+@rem Unless required by applicable law or agreed to in writing, software
+@rem distributed under the License is distributed on an "AS IS" BASIS,
+@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+@rem See the License for the specific language governing permissions and
+@rem limitations under the License.
+@rem
+
+@if "%DEBUG%" == "" @echo off
+@rem ##########################################################################
+@rem
+@rem Gradle startup script for Windows
+@rem
+@rem ##########################################################################
+
+@rem Set local scope for the variables with windows NT shell
+if "%OS%"=="Windows_NT" setlocal
+
+set DIRNAME=%~dp0
+if "%DIRNAME%" == "" set DIRNAME=.
+set APP_BASE_NAME=%~n0
+set APP_HOME=%DIRNAME%
+
+@rem Resolve any "." and ".." in APP_HOME to make it shorter.
+for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
+
+@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
+set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
+
+@rem Find java.exe
+if defined JAVA_HOME goto findJavaFromJavaHome
+
+set JAVA_EXE=java.exe
+%JAVA_EXE% -version >NUL 2>&1
+if "%ERRORLEVEL%" == "0" goto execute
+
+echo.
+echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
+echo.
+echo Please set the JAVA_HOME variable in your environment to match the
+echo location of your Java installation.
+
+goto fail
+
+:findJavaFromJavaHome
+set JAVA_HOME=%JAVA_HOME:"=%
+set JAVA_EXE=%JAVA_HOME%/bin/java.exe
+
+if exist "%JAVA_EXE%" goto execute
+
+echo.
+echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
+echo.
+echo Please set the JAVA_HOME variable in your environment to match the
+echo location of your Java installation.
+
+goto fail
+
+:execute
+@rem Setup the command line
+
+set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
+
+
+@rem Execute Gradle
+"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
+
+:end
+@rem End local scope for the variables with windows NT shell
+if "%ERRORLEVEL%"=="0" goto mainEnd
+
+:fail
+rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
+rem the _cmd.exe /c_ return code!
+if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
+exit /b 1
+
+:mainEnd
+if "%OS%"=="Windows_NT" endlocal
+
+:omega
diff --git a/android-app/settings.gradle b/android-app/settings.gradle
new file mode 100644
index 0000000..87140f8
--- /dev/null
+++ b/android-app/settings.gradle
@@ -0,0 +1,10 @@
+dependencyResolutionManagement {
+ repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
+ repositories {
+ google()
+ mavenCentral()
+ jcenter() // Warning: this repository is going to shut down soon
+ }
+}
+rootProject.name = "Futharkboard"
+include ':app'