๐Ÿ”— Redirect Bluesky links to your preferred client
3
fork

Configure Feed

Select the types of activity you want to include in your feed.

Initial commit

Zachary Wander 3eed71b7

+1343
+15
.gitignore
··· 1 + *.iml 2 + .gradle 3 + /local.properties 4 + /.idea/caches 5 + /.idea/libraries 6 + /.idea/modules.xml 7 + /.idea/workspace.xml 8 + /.idea/navEditor.xml 9 + /.idea/assetWizardSettings.xml 10 + .DS_Store 11 + /build 12 + /captures 13 + .externalNativeBuild 14 + .cxx 15 + local.properties
+3
.idea/.gitignore
··· 1 + # Default ignored files 2 + /shelf/ 3 + /workspace.xml
+1
app/.gitignore
··· 1 + /build
+73
app/build.gradle.kts
··· 1 + @Suppress("DSL_SCOPE_VIOLATION") // TODO: Remove once KTIJ-19369 is fixed 2 + plugins { 3 + alias(libs.plugins.androidApplication) 4 + alias(libs.plugins.kotlinAndroid) 5 + } 6 + 7 + android { 8 + namespace = "dev.zwander.mastodonredirect" 9 + compileSdk = 33 10 + 11 + defaultConfig { 12 + applicationId = "dev.zwander.mastodonredirect" 13 + minSdk = 24 14 + targetSdk = 33 15 + versionCode = 1 16 + versionName = "1.0" 17 + 18 + testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" 19 + vectorDrawables { 20 + useSupportLibrary = true 21 + } 22 + } 23 + 24 + buildTypes { 25 + release { 26 + isMinifyEnabled = false 27 + proguardFiles( 28 + getDefaultProguardFile("proguard-android-optimize.txt"), 29 + "proguard-rules.pro" 30 + ) 31 + } 32 + } 33 + compileOptions { 34 + sourceCompatibility = JavaVersion.VERSION_1_8 35 + targetCompatibility = JavaVersion.VERSION_1_8 36 + } 37 + kotlinOptions { 38 + jvmTarget = "1.8" 39 + } 40 + buildFeatures { 41 + compose = true 42 + } 43 + composeOptions { 44 + kotlinCompilerExtensionVersion = "1.4.7" 45 + } 46 + packaging { 47 + resources { 48 + excludes += "/META-INF/{AL2.0,LGPL2.1}" 49 + } 50 + } 51 + } 52 + 53 + dependencies { 54 + 55 + implementation(libs.core.ktx) 56 + implementation(libs.lifecycle.runtime.ktx) 57 + implementation(libs.activity.compose) 58 + implementation(platform(libs.compose.bom)) 59 + implementation(libs.ui) 60 + implementation(libs.ui.graphics) 61 + implementation(libs.ui.tooling.preview) 62 + implementation(libs.material3) 63 + implementation(libs.materialComponents) 64 + implementation(libs.preferences) 65 + 66 + testImplementation(libs.junit) 67 + androidTestImplementation(libs.androidx.test.ext.junit) 68 + androidTestImplementation(libs.espresso.core) 69 + androidTestImplementation(platform(libs.compose.bom)) 70 + androidTestImplementation(libs.ui.test.junit4) 71 + debugImplementation(libs.ui.tooling) 72 + debugImplementation(libs.ui.test.manifest) 73 + }
+21
app/proguard-rules.pro
··· 1 + # Add project specific ProGuard rules here. 2 + # You can control the set of applied configuration files using the 3 + # proguardFiles setting in build.gradle. 4 + # 5 + # For more details, see 6 + # http://developer.android.com/guide/developing/tools/proguard.html 7 + 8 + # If your project uses WebView with JS, uncomment the following 9 + # and specify the fully qualified class name to the JavaScript interface 10 + # class: 11 + #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 12 + # public *; 13 + #} 14 + 15 + # Uncomment this to preserve the line number information for 16 + # debugging stack traces. 17 + #-keepattributes SourceFile,LineNumberTable 18 + 19 + # If you keep the line number information, uncomment this to 20 + # hide the original source file name. 21 + #-renamesourcefileattribute SourceFile
+24
app/src/androidTest/java/dev/zwander/mastodonredirect/ExampleInstrumentedTest.kt
··· 1 + package dev.zwander.mastodonredirect 2 + 3 + import androidx.test.platform.app.InstrumentationRegistry 4 + import androidx.test.ext.junit.runners.AndroidJUnit4 5 + 6 + import org.junit.Test 7 + import org.junit.runner.RunWith 8 + 9 + import org.junit.Assert.* 10 + 11 + /** 12 + * Instrumented test, which will execute on an Android device. 13 + * 14 + * See [testing documentation](http://d.android.com/tools/testing). 15 + */ 16 + @RunWith(AndroidJUnit4::class) 17 + class ExampleInstrumentedTest { 18 + @Test 19 + fun useAppContext() { 20 + // Context of the app under test. 21 + val appContext = InstrumentationRegistry.getInstrumentation().targetContext 22 + assertEquals("dev.zwander.mastodonredirect", appContext.packageName) 23 + } 24 + }
+153
app/src/main/AndroidManifest.xml
··· 1 + <?xml version="1.0" encoding="utf-8"?> 2 + <manifest xmlns:android="http://schemas.android.com/apk/res/android" 3 + xmlns:tools="http://schemas.android.com/tools"> 4 + 5 + <application 6 + android:allowBackup="true" 7 + android:dataExtractionRules="@xml/data_extraction_rules" 8 + android:fullBackupContent="@xml/backup_rules" 9 + android:icon="@mipmap/ic_launcher" 10 + android:label="@string/app_name" 11 + android:roundIcon="@mipmap/ic_launcher_round" 12 + android:supportsRtl="true" 13 + android:theme="@style/Theme.MastodonRedirect" 14 + tools:targetApi="31"> 15 + <activity 16 + android:name=".MainActivity" 17 + android:exported="true"> 18 + <intent-filter> 19 + <action android:name="android.intent.action.MAIN" /> 20 + 21 + <category android:name="android.intent.category.LAUNCHER" /> 22 + </intent-filter> 23 + </activity> 24 + 25 + <activity android:name=".RedirectActivity" 26 + android:exported="true" 27 + android:theme="@style/AppTheme.Transparent" 28 + android:excludeFromRecents="true"> 29 + <intent-filter> 30 + <action android:name="android.intent.action.VIEW" /> 31 + <category android:name="android.intent.category.DEFAULT" /> 32 + <category android:name="android.intent.category.BROWSABLE" /> 33 + 34 + <data android:scheme="https" /> 35 + <data android:scheme="http" /> 36 + 37 + <!-- https://instances.social/list/advanced#lang=&allowed=&prohibited=&min-users=&max-users= --> 38 + <data android:host="mastodon.social" /> 39 + <data android:host="mastodon.online" /> 40 + <data android:host="androiddev.social" /> 41 + <data android:host="pawoo.net" /> 42 + <data android:host="daystorm.netz.org" /> 43 + <data android:host="mstdn.jp" /> 44 + <data android:host="baraag.net" /> 45 + <data android:host="mastodon.cloud" /> 46 + <data android:host="mstdn.social" /> 47 + <data android:host="mastodon.world" /> 48 + <data android:host="counter.social" /> 49 + <data android:host="mas.to" /> 50 + <data android:host="pravda.me" /> 51 + <data android:host="gc2.jp" /> 52 + <data android:host="pixelfed.social" /> 53 + <data android:host="techhub.social" /> 54 + <data android:host="universeodon.com" /> 55 + <data android:host="mastodon.sdf.org" /> 56 + <data android:host="mastodonapp.uk" /> 57 + <data android:host="masto.ai" /> 58 + <data android:host="mastodon.uno" /> 59 + <data android:host="c.im" /> 60 + <data android:host="fosstodon.org" /> 61 + <data android:host="infosec.exchange" /> 62 + <data android:host="mastodon.top" /> 63 + <data android:host="brighteon.social" /> 64 + <data android:host="mstdn.party" /> 65 + <data android:host="hachyderm.io" /> 66 + <data android:host="kids.0px.io" /> 67 + <data android:host="troet.cafe" /> 68 + <data android:host="m.cmx.im" /> 69 + <data android:host="mastodon.nl" /> 70 + <data android:host="ohai.social" /> 71 + <data android:host="mamot.fr" /> 72 + <data android:host="sfba.social" /> 73 + <data android:host="kolektiva.social" /> 74 + <data android:host="mindly.social" /> 75 + <data android:host="mstdn.ca" /> 76 + <data android:host="mastodon.art" /> 77 + <data android:host="fedibird.com" /> 78 + <data android:host="social.vivaldi.net" /> 79 + <data android:host="loforo.com" /> 80 + <data android:host="tkz.one" /> 81 + <data android:host="mast.lat" /> 82 + <data android:host="mastorock.com" /> 83 + <data android:host="toot.community" /> 84 + <data android:host="poa.st" /> 85 + <data android:host="alive.bar" /> 86 + <data android:host="mastodon.scot" /> 87 + <data android:host="mastodon.xyz" /> 88 + <data android:host="mastodon.gamedev.place" /> 89 + <data android:host="qoto.org" /> 90 + <data android:host="mastodon.au" /> 91 + <data android:host="ioc.exchange" /> 92 + <data android:host="pixelfed.de" /> 93 + <data android:host="piaille.fr" /> 94 + <data android:host="aus.social" /> 95 + <data android:host="ravenation.club" /> 96 + <data android:host="masto.nu" /> 97 + <data android:host="wxw.moe" /> 98 + <data android:host="twingyeo.kr" /> 99 + <data android:host="mastouille.fr" /> 100 + <data android:host="spinster.xyz" /> 101 + <data android:host="newsie.social" /> 102 + <data android:host="planet.moe" /> 103 + <data android:host="social.tchncs.de" /> 104 + <data android:host="mastodon.bida.im" /> 105 + <data android:host="mastodon.ie" /> 106 + <data android:host="mastodon.com.tr" /> 107 + <data android:host="det.social" /> 108 + <data android:host="nerdculture.de" /> 109 + <data android:host="nrw.social" /> 110 + <data android:host="m.rthome.me" /> 111 + <data android:host="o3o.ca" /> 112 + <data android:host="adbl.link" /> 113 + <data android:host="mastodontech.de" /> 114 + <data android:host="mastodon.nu" /> 115 + <data android:host="thu.closed.social" /> 116 + <data android:host="theblower.au" /> 117 + <data android:host="mathstodon.xyz" /> 118 + <data android:host="glasgow.social" /> 119 + <data android:host="mao.mastodonhub.com" /> 120 + <data android:host="sueden.social" /> 121 + <data android:host="ieji.de" /> 122 + <data android:host="toot.wales" /> 123 + <data android:host="noc.social" /> 124 + <data android:host="mastodon-japan.net" /> 125 + <data android:host="me.dm" /> 126 + <data android:host="norden.social" /> 127 + <data android:host="eldritch.cafe" /> 128 + <data android:host="kinky.business" /> 129 + <data android:host="tech.lgbt" /> 130 + <data android:host="octodon.social" /> 131 + <data android:host="meow.social" /> 132 + <data android:host="mastodon.green" /> 133 + <data android:host="social.cologne" /> 134 + <data android:host="bookwyrm.social" /> 135 + <data android:host="awscommunity.social" /> 136 + <data android:host="mastodon.iriseden.eu" /> 137 + <data android:host="indieweb.social" /> 138 + <data android:host="social.linux.pizza" /> 139 + <data android:host="mstdn.plus" /> 140 + <data android:host="muenchen.social" /> 141 + <data android:host="framapiaf.org" /> 142 + <data android:host="toot.io" /> 143 + <data android:host="sself.co" /> 144 + <data android:host="med-mastodon.com" /> 145 + <data android:host="ruhr.social" /> 146 + <data android:host="sigmoid.social" /> 147 + <data android:host="social.anoxinon.de" /> 148 + <data android:host="mstdn.tokyocameraclub.com" /> 149 + </intent-filter> 150 + </activity> 151 + </application> 152 + 153 + </manifest>
+64
app/src/main/java/dev/zwander/mastodonredirect/LaunchStrategy.kt
··· 1 + package dev.zwander.mastodonredirect 2 + 3 + import android.content.ComponentName 4 + import android.content.Context 5 + import android.content.Intent 6 + import android.net.Uri 7 + import android.util.Log 8 + import androidx.annotation.StringRes 9 + 10 + val launchStrategies by lazy { 11 + mapOf( 12 + Megalodon.key to Megalodon, 13 + Fedilab.key to Fedilab, 14 + ) 15 + } 16 + 17 + sealed class LaunchStrategy( 18 + val key: String, 19 + @StringRes val labelRes: Int, 20 + ) { 21 + abstract fun Context.createIntents(url: String?): List<Intent> 22 + } 23 + 24 + object Megalodon : LaunchStrategy("MEGALODON", R.string.megalodon) { 25 + override fun Context.createIntents(url: String?): List<Intent> { 26 + return listOf( 27 + Intent(Intent.ACTION_SEND).apply { 28 + putExtra(Intent.EXTRA_TEXT, url) 29 + `package` = "org.joinmastodon.android.sk" 30 + component = ComponentName( 31 + "org.joinmastodon.android.sk", 32 + "org.joinmastodon.android.ExternalShareActivity" 33 + ) 34 + }, 35 + ) 36 + } 37 + } 38 + 39 + object Fedilab : LaunchStrategy("FEDILAB", R.string.fedilab) { 40 + override fun Context.createIntents(url: String?): List<Intent> { 41 + Log.e("MastodonRedirect", "url: $url") 42 + 43 + val baseIntent = Intent(Intent.ACTION_VIEW) 44 + baseIntent.data = url?.let { Uri.parse(url) } 45 + 46 + return listOf( 47 + Intent(baseIntent).apply { 48 + `package` = "app.fedilab.android" 49 + component = ComponentName( 50 + "app.fedilab.android", 51 + "app.fedilab.android.activities.MainActivity" 52 + ) 53 + 54 + }, 55 + Intent(baseIntent).apply { 56 + `package` = "fr.gouv.etalab.mastodon" 57 + component = ComponentName( 58 + "fr.gouv.etalab.mastodon", 59 + "app.fedilab.android.activities.MainActivity" 60 + ) 61 + } 62 + ) 63 + } 64 + }
+91
app/src/main/java/dev/zwander/mastodonredirect/MainActivity.kt
··· 1 + package dev.zwander.mastodonredirect 2 + 3 + import android.os.Bundle 4 + import androidx.activity.ComponentActivity 5 + import androidx.activity.compose.setContent 6 + import androidx.compose.animation.animateColorAsState 7 + import androidx.compose.foundation.layout.Arrangement 8 + import androidx.compose.foundation.layout.PaddingValues 9 + import androidx.compose.foundation.layout.Row 10 + import androidx.compose.foundation.layout.fillMaxSize 11 + import androidx.compose.foundation.layout.fillMaxWidth 12 + import androidx.compose.foundation.layout.heightIn 13 + import androidx.compose.foundation.layout.padding 14 + import androidx.compose.foundation.lazy.LazyColumn 15 + import androidx.compose.foundation.lazy.items 16 + import androidx.compose.material3.CardDefaults 17 + import androidx.compose.material3.ElevatedCard 18 + import androidx.compose.material3.ExperimentalMaterial3Api 19 + import androidx.compose.material3.MaterialTheme 20 + import androidx.compose.material3.Surface 21 + import androidx.compose.material3.Text 22 + import androidx.compose.runtime.getValue 23 + import androidx.compose.runtime.setValue 24 + import androidx.compose.ui.Alignment 25 + import androidx.compose.ui.Modifier 26 + import androidx.compose.ui.platform.LocalContext 27 + import androidx.compose.ui.res.stringResource 28 + import androidx.compose.ui.text.style.TextAlign 29 + import androidx.compose.ui.unit.dp 30 + import dev.zwander.mastodonredirect.ui.theme.MastodonRedirectTheme 31 + 32 + class MainActivity : ComponentActivity() { 33 + @OptIn(ExperimentalMaterial3Api::class) 34 + override fun onCreate(savedInstanceState: Bundle?) { 35 + super.onCreate(savedInstanceState) 36 + setContent { 37 + MastodonRedirectTheme { 38 + // A surface container using the 'background' color from the theme 39 + Surface( 40 + modifier = Modifier.fillMaxSize(), 41 + color = MaterialTheme.colorScheme.background 42 + ) { 43 + val context = LocalContext.current 44 + var selectedStrategy by context.rememberPreferenceState( 45 + key = Prefs.SELECTED_APP, 46 + value = { prefs.selectedApp }, 47 + onChanged = { prefs.selectedApp = it }, 48 + ) 49 + 50 + LazyColumn( 51 + modifier = Modifier.fillMaxSize(), 52 + contentPadding = PaddingValues(8.dp), 53 + verticalArrangement = Arrangement.spacedBy(8.dp), 54 + ) { 55 + items(launchStrategies.entries.toList(), { it.key }) { (key, strategy) -> 56 + val color by animateColorAsState( 57 + targetValue = if (selectedStrategy == strategy) MaterialTheme.colorScheme.primary else MaterialTheme.colorScheme.secondary, 58 + label = "CardColor-$key", 59 + ) 60 + 61 + ElevatedCard( 62 + onClick = { selectedStrategy = strategy }, 63 + colors = CardDefaults.elevatedCardColors( 64 + containerColor = color, 65 + ), 66 + ) { 67 + Row( 68 + modifier = Modifier.fillMaxWidth() 69 + .heightIn(min = 56.dp) 70 + .padding(8.dp), 71 + verticalAlignment = Alignment.CenterVertically, 72 + ) { 73 + Text( 74 + text = stringResource(id = strategy.labelRes), 75 + modifier = Modifier.fillMaxWidth(), 76 + textAlign = TextAlign.Center, 77 + color = if (selectedStrategy == strategy) { 78 + MaterialTheme.colorScheme.onPrimary 79 + } else { 80 + MaterialTheme.colorScheme.onSecondary 81 + }, 82 + ) 83 + } 84 + } 85 + } 86 + } 87 + } 88 + } 89 + } 90 + } 91 + }
+54
app/src/main/java/dev/zwander/mastodonredirect/PrefUtils.kt
··· 1 + package dev.zwander.mastodonredirect 2 + 3 + import android.content.Context 4 + import android.content.SharedPreferences 5 + import androidx.compose.runtime.Composable 6 + import androidx.compose.runtime.DisposableEffect 7 + import androidx.compose.runtime.LaunchedEffect 8 + import androidx.compose.runtime.MutableState 9 + import androidx.compose.runtime.mutableStateOf 10 + import androidx.compose.runtime.remember 11 + 12 + @Composable 13 + fun <T> Context.rememberPreferenceState( 14 + key: String, 15 + value: () -> T, 16 + onChanged: (T) -> Unit 17 + ): MutableState<T> { 18 + val state = remember(key) { 19 + mutableStateOf(value()) 20 + } 21 + 22 + LaunchedEffect(key1 = state.value) { 23 + onChanged(state.value) 24 + } 25 + 26 + DisposableEffect(key1 = key) { 27 + val listener = SharedPreferences.OnSharedPreferenceChangeListener { _, k -> 28 + if (key == k) { 29 + state.value = value() 30 + } 31 + } 32 + 33 + prefs.preferences.registerOnSharedPreferenceChangeListener(listener) 34 + 35 + onDispose { 36 + prefs.preferences.unregisterOnSharedPreferenceChangeListener(listener) 37 + } 38 + } 39 + 40 + return state 41 + } 42 + 43 + @Composable 44 + fun Context.rememberBooleanPreferenceState( 45 + key: String, 46 + enabled: () -> Boolean, 47 + onEnabledChanged: (Boolean) -> Unit 48 + ): MutableState<Boolean> { 49 + return rememberPreferenceState( 50 + key = key, 51 + value = enabled, 52 + onChanged = onEnabledChanged 53 + ) 54 + }
+34
app/src/main/java/dev/zwander/mastodonredirect/Prefs.kt
··· 1 + package dev.zwander.mastodonredirect 2 + 3 + import android.annotation.SuppressLint 4 + import android.content.Context 5 + import android.content.ContextWrapper 6 + import android.content.SharedPreferences 7 + import androidx.core.content.edit 8 + import androidx.preference.PreferenceManager 9 + 10 + val Context.prefs: Prefs 11 + get() = Prefs.getInstance(this) 12 + 13 + class Prefs private constructor(context: Context) : ContextWrapper(context) { 14 + companion object { 15 + @SuppressLint("StaticFieldLeak") 16 + private var instance: Prefs? = null 17 + 18 + fun getInstance(context: Context): Prefs { 19 + return instance ?: Prefs(context.applicationContext ?: context).apply { 20 + instance = this 21 + } 22 + } 23 + 24 + const val SELECTED_APP = "selected_app" 25 + } 26 + 27 + val preferences: SharedPreferences = PreferenceManager.getDefaultSharedPreferences(this) 28 + 29 + var selectedApp: LaunchStrategy 30 + get() = launchStrategies[preferences.getString(SELECTED_APP, Megalodon.key)] ?: Megalodon 31 + set(value) { 32 + preferences.edit { putString(SELECTED_APP, value.key) } 33 + } 34 + }
+33
app/src/main/java/dev/zwander/mastodonredirect/RedirectActivity.kt
··· 1 + package dev.zwander.mastodonredirect 2 + 3 + import android.os.Bundle 4 + import android.util.Log 5 + import androidx.activity.ComponentActivity 6 + import kotlinx.coroutines.CoroutineScope 7 + import kotlinx.coroutines.MainScope 8 + import kotlinx.coroutines.cancel 9 + 10 + class RedirectActivity : ComponentActivity(), CoroutineScope by MainScope() { 11 + override fun onCreate(savedInstanceState: Bundle?) { 12 + super.onCreate(savedInstanceState) 13 + 14 + val url = intent?.data?.toString() 15 + 16 + prefs.selectedApp.run { 17 + createIntents(url).forEach { 18 + try { 19 + startActivity(it) 20 + } catch (e: Exception) { 21 + Log.e("MastodonRedirect", "Error launching.", e) 22 + } 23 + } 24 + } 25 + 26 + finish() 27 + } 28 + 29 + override fun onDestroy() { 30 + super.onDestroy() 31 + cancel() 32 + } 33 + }
+11
app/src/main/java/dev/zwander/mastodonredirect/ui/theme/Color.kt
··· 1 + package dev.zwander.mastodonredirect.ui.theme 2 + 3 + import androidx.compose.ui.graphics.Color 4 + 5 + val Purple80 = Color(0xFFD0BCFF) 6 + val PurpleGrey80 = Color(0xFFCCC2DC) 7 + val Pink80 = Color(0xFFEFB8C8) 8 + 9 + val Purple40 = Color(0xFF6650a4) 10 + val PurpleGrey40 = Color(0xFF625b71) 11 + val Pink40 = Color(0xFF7D5260)
+70
app/src/main/java/dev/zwander/mastodonredirect/ui/theme/Theme.kt
··· 1 + package dev.zwander.mastodonredirect.ui.theme 2 + 3 + import android.app.Activity 4 + import android.os.Build 5 + import androidx.compose.foundation.isSystemInDarkTheme 6 + import androidx.compose.material3.MaterialTheme 7 + import androidx.compose.material3.darkColorScheme 8 + import androidx.compose.material3.dynamicDarkColorScheme 9 + import androidx.compose.material3.dynamicLightColorScheme 10 + import androidx.compose.material3.lightColorScheme 11 + import androidx.compose.runtime.Composable 12 + import androidx.compose.runtime.SideEffect 13 + import androidx.compose.ui.graphics.toArgb 14 + import androidx.compose.ui.platform.LocalContext 15 + import androidx.compose.ui.platform.LocalView 16 + import androidx.core.view.WindowCompat 17 + 18 + private val DarkColorScheme = darkColorScheme( 19 + primary = Purple80, 20 + secondary = PurpleGrey80, 21 + tertiary = Pink80 22 + ) 23 + 24 + private val LightColorScheme = lightColorScheme( 25 + primary = Purple40, 26 + secondary = PurpleGrey40, 27 + tertiary = Pink40 28 + 29 + /* Other default colors to override 30 + background = Color(0xFFFFFBFE), 31 + surface = Color(0xFFFFFBFE), 32 + onPrimary = Color.White, 33 + onSecondary = Color.White, 34 + onTertiary = Color.White, 35 + onBackground = Color(0xFF1C1B1F), 36 + onSurface = Color(0xFF1C1B1F), 37 + */ 38 + ) 39 + 40 + @Composable 41 + fun MastodonRedirectTheme( 42 + darkTheme: Boolean = isSystemInDarkTheme(), 43 + // Dynamic color is available on Android 12+ 44 + dynamicColor: Boolean = true, 45 + content: @Composable () -> Unit 46 + ) { 47 + val colorScheme = when { 48 + dynamicColor && Build.VERSION.SDK_INT >= Build.VERSION_CODES.S -> { 49 + val context = LocalContext.current 50 + if (darkTheme) dynamicDarkColorScheme(context) else dynamicLightColorScheme(context) 51 + } 52 + 53 + darkTheme -> DarkColorScheme 54 + else -> LightColorScheme 55 + } 56 + val view = LocalView.current 57 + if (!view.isInEditMode) { 58 + SideEffect { 59 + val window = (view.context as Activity).window 60 + window.statusBarColor = colorScheme.primary.toArgb() 61 + WindowCompat.getInsetsController(window, view).isAppearanceLightStatusBars = darkTheme 62 + } 63 + } 64 + 65 + MaterialTheme( 66 + colorScheme = colorScheme, 67 + typography = Typography, 68 + content = content 69 + ) 70 + }
+34
app/src/main/java/dev/zwander/mastodonredirect/ui/theme/Type.kt
··· 1 + package dev.zwander.mastodonredirect.ui.theme 2 + 3 + import androidx.compose.material3.Typography 4 + import androidx.compose.ui.text.TextStyle 5 + import androidx.compose.ui.text.font.FontFamily 6 + import androidx.compose.ui.text.font.FontWeight 7 + import androidx.compose.ui.unit.sp 8 + 9 + // Set of Material typography styles to start with 10 + val Typography = Typography( 11 + bodyLarge = TextStyle( 12 + fontFamily = FontFamily.Default, 13 + fontWeight = FontWeight.Normal, 14 + fontSize = 16.sp, 15 + lineHeight = 24.sp, 16 + letterSpacing = 0.5.sp 17 + ) 18 + /* Other default text styles to override 19 + titleLarge = TextStyle( 20 + fontFamily = FontFamily.Default, 21 + fontWeight = FontWeight.Normal, 22 + fontSize = 22.sp, 23 + lineHeight = 28.sp, 24 + letterSpacing = 0.sp 25 + ), 26 + labelSmall = TextStyle( 27 + fontFamily = FontFamily.Default, 28 + fontWeight = FontWeight.Medium, 29 + fontSize = 11.sp, 30 + lineHeight = 16.sp, 31 + letterSpacing = 0.5.sp 32 + ) 33 + */ 34 + )
+170
app/src/main/res/drawable/ic_launcher_background.xml
··· 1 + <?xml version="1.0" encoding="utf-8"?> 2 + <vector xmlns:android="http://schemas.android.com/apk/res/android" 3 + android:width="108dp" 4 + android:height="108dp" 5 + android:viewportWidth="108" 6 + android:viewportHeight="108"> 7 + <path 8 + android:fillColor="#3DDC84" 9 + android:pathData="M0,0h108v108h-108z" /> 10 + <path 11 + android:fillColor="#00000000" 12 + android:pathData="M9,0L9,108" 13 + android:strokeWidth="0.8" 14 + android:strokeColor="#33FFFFFF" /> 15 + <path 16 + android:fillColor="#00000000" 17 + android:pathData="M19,0L19,108" 18 + android:strokeWidth="0.8" 19 + android:strokeColor="#33FFFFFF" /> 20 + <path 21 + android:fillColor="#00000000" 22 + android:pathData="M29,0L29,108" 23 + android:strokeWidth="0.8" 24 + android:strokeColor="#33FFFFFF" /> 25 + <path 26 + android:fillColor="#00000000" 27 + android:pathData="M39,0L39,108" 28 + android:strokeWidth="0.8" 29 + android:strokeColor="#33FFFFFF" /> 30 + <path 31 + android:fillColor="#00000000" 32 + android:pathData="M49,0L49,108" 33 + android:strokeWidth="0.8" 34 + android:strokeColor="#33FFFFFF" /> 35 + <path 36 + android:fillColor="#00000000" 37 + android:pathData="M59,0L59,108" 38 + android:strokeWidth="0.8" 39 + android:strokeColor="#33FFFFFF" /> 40 + <path 41 + android:fillColor="#00000000" 42 + android:pathData="M69,0L69,108" 43 + android:strokeWidth="0.8" 44 + android:strokeColor="#33FFFFFF" /> 45 + <path 46 + android:fillColor="#00000000" 47 + android:pathData="M79,0L79,108" 48 + android:strokeWidth="0.8" 49 + android:strokeColor="#33FFFFFF" /> 50 + <path 51 + android:fillColor="#00000000" 52 + android:pathData="M89,0L89,108" 53 + android:strokeWidth="0.8" 54 + android:strokeColor="#33FFFFFF" /> 55 + <path 56 + android:fillColor="#00000000" 57 + android:pathData="M99,0L99,108" 58 + android:strokeWidth="0.8" 59 + android:strokeColor="#33FFFFFF" /> 60 + <path 61 + android:fillColor="#00000000" 62 + android:pathData="M0,9L108,9" 63 + android:strokeWidth="0.8" 64 + android:strokeColor="#33FFFFFF" /> 65 + <path 66 + android:fillColor="#00000000" 67 + android:pathData="M0,19L108,19" 68 + android:strokeWidth="0.8" 69 + android:strokeColor="#33FFFFFF" /> 70 + <path 71 + android:fillColor="#00000000" 72 + android:pathData="M0,29L108,29" 73 + android:strokeWidth="0.8" 74 + android:strokeColor="#33FFFFFF" /> 75 + <path 76 + android:fillColor="#00000000" 77 + android:pathData="M0,39L108,39" 78 + android:strokeWidth="0.8" 79 + android:strokeColor="#33FFFFFF" /> 80 + <path 81 + android:fillColor="#00000000" 82 + android:pathData="M0,49L108,49" 83 + android:strokeWidth="0.8" 84 + android:strokeColor="#33FFFFFF" /> 85 + <path 86 + android:fillColor="#00000000" 87 + android:pathData="M0,59L108,59" 88 + android:strokeWidth="0.8" 89 + android:strokeColor="#33FFFFFF" /> 90 + <path 91 + android:fillColor="#00000000" 92 + android:pathData="M0,69L108,69" 93 + android:strokeWidth="0.8" 94 + android:strokeColor="#33FFFFFF" /> 95 + <path 96 + android:fillColor="#00000000" 97 + android:pathData="M0,79L108,79" 98 + android:strokeWidth="0.8" 99 + android:strokeColor="#33FFFFFF" /> 100 + <path 101 + android:fillColor="#00000000" 102 + android:pathData="M0,89L108,89" 103 + android:strokeWidth="0.8" 104 + android:strokeColor="#33FFFFFF" /> 105 + <path 106 + android:fillColor="#00000000" 107 + android:pathData="M0,99L108,99" 108 + android:strokeWidth="0.8" 109 + android:strokeColor="#33FFFFFF" /> 110 + <path 111 + android:fillColor="#00000000" 112 + android:pathData="M19,29L89,29" 113 + android:strokeWidth="0.8" 114 + android:strokeColor="#33FFFFFF" /> 115 + <path 116 + android:fillColor="#00000000" 117 + android:pathData="M19,39L89,39" 118 + android:strokeWidth="0.8" 119 + android:strokeColor="#33FFFFFF" /> 120 + <path 121 + android:fillColor="#00000000" 122 + android:pathData="M19,49L89,49" 123 + android:strokeWidth="0.8" 124 + android:strokeColor="#33FFFFFF" /> 125 + <path 126 + android:fillColor="#00000000" 127 + android:pathData="M19,59L89,59" 128 + android:strokeWidth="0.8" 129 + android:strokeColor="#33FFFFFF" /> 130 + <path 131 + android:fillColor="#00000000" 132 + android:pathData="M19,69L89,69" 133 + android:strokeWidth="0.8" 134 + android:strokeColor="#33FFFFFF" /> 135 + <path 136 + android:fillColor="#00000000" 137 + android:pathData="M19,79L89,79" 138 + android:strokeWidth="0.8" 139 + android:strokeColor="#33FFFFFF" /> 140 + <path 141 + android:fillColor="#00000000" 142 + android:pathData="M29,19L29,89" 143 + android:strokeWidth="0.8" 144 + android:strokeColor="#33FFFFFF" /> 145 + <path 146 + android:fillColor="#00000000" 147 + android:pathData="M39,19L39,89" 148 + android:strokeWidth="0.8" 149 + android:strokeColor="#33FFFFFF" /> 150 + <path 151 + android:fillColor="#00000000" 152 + android:pathData="M49,19L49,89" 153 + android:strokeWidth="0.8" 154 + android:strokeColor="#33FFFFFF" /> 155 + <path 156 + android:fillColor="#00000000" 157 + android:pathData="M59,19L59,89" 158 + android:strokeWidth="0.8" 159 + android:strokeColor="#33FFFFFF" /> 160 + <path 161 + android:fillColor="#00000000" 162 + android:pathData="M69,19L69,89" 163 + android:strokeWidth="0.8" 164 + android:strokeColor="#33FFFFFF" /> 165 + <path 166 + android:fillColor="#00000000" 167 + android:pathData="M79,19L79,89" 168 + android:strokeWidth="0.8" 169 + android:strokeColor="#33FFFFFF" /> 170 + </vector>
+30
app/src/main/res/drawable/ic_launcher_foreground.xml
··· 1 + <vector xmlns:android="http://schemas.android.com/apk/res/android" 2 + xmlns:aapt="http://schemas.android.com/aapt" 3 + android:width="108dp" 4 + android:height="108dp" 5 + android:viewportWidth="108" 6 + android:viewportHeight="108"> 7 + <path android:pathData="M31,63.928c0,0 6.4,-11 12.1,-13.1c7.2,-2.6 26,-1.4 26,-1.4l38.1,38.1L107,108.928l-32,-1L31,63.928z"> 8 + <aapt:attr name="android:fillColor"> 9 + <gradient 10 + android:endX="85.84757" 11 + android:endY="92.4963" 12 + android:startX="42.9492" 13 + android:startY="49.59793" 14 + android:type="linear"> 15 + <item 16 + android:color="#44000000" 17 + android:offset="0.0" /> 18 + <item 19 + android:color="#00000000" 20 + android:offset="1.0" /> 21 + </gradient> 22 + </aapt:attr> 23 + </path> 24 + <path 25 + android:fillColor="#FFFFFF" 26 + android:fillType="nonZero" 27 + android:pathData="M65.3,45.828l3.8,-6.6c0.2,-0.4 0.1,-0.9 -0.3,-1.1c-0.4,-0.2 -0.9,-0.1 -1.1,0.3l-3.9,6.7c-6.3,-2.8 -13.4,-2.8 -19.7,0l-3.9,-6.7c-0.2,-0.4 -0.7,-0.5 -1.1,-0.3C38.8,38.328 38.7,38.828 38.9,39.228l3.8,6.6C36.2,49.428 31.7,56.028 31,63.928h46C76.3,56.028 71.8,49.428 65.3,45.828zM43.4,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2c-0.3,-0.7 -0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C45.3,56.528 44.5,57.328 43.4,57.328L43.4,57.328zM64.6,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2s-0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C66.5,56.528 65.6,57.328 64.6,57.328L64.6,57.328z" 28 + android:strokeWidth="1" 29 + android:strokeColor="#00000000" /> 30 + </vector>
+6
app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml
··· 1 + <?xml version="1.0" encoding="utf-8"?> 2 + <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> 3 + <background android:drawable="@drawable/ic_launcher_background" /> 4 + <foreground android:drawable="@drawable/ic_launcher_foreground" /> 5 + <monochrome android:drawable="@drawable/ic_launcher_foreground" /> 6 + </adaptive-icon>
+6
app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml
··· 1 + <?xml version="1.0" encoding="utf-8"?> 2 + <adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> 3 + <background android:drawable="@drawable/ic_launcher_background" /> 4 + <foreground android:drawable="@drawable/ic_launcher_foreground" /> 5 + <monochrome android:drawable="@drawable/ic_launcher_foreground" /> 6 + </adaptive-icon>
app/src/main/res/mipmap-hdpi/ic_launcher.webp

This is a binary file and will not be displayed.

app/src/main/res/mipmap-hdpi/ic_launcher_round.webp

This is a binary file and will not be displayed.

app/src/main/res/mipmap-mdpi/ic_launcher.webp

This is a binary file and will not be displayed.

app/src/main/res/mipmap-mdpi/ic_launcher_round.webp

This is a binary file and will not be displayed.

app/src/main/res/mipmap-xhdpi/ic_launcher.webp

This is a binary file and will not be displayed.

app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp

This is a binary file and will not be displayed.

app/src/main/res/mipmap-xxhdpi/ic_launcher.webp

This is a binary file and will not be displayed.

app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp

This is a binary file and will not be displayed.

app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp

This is a binary file and will not be displayed.

app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp

This is a binary file and will not be displayed.

+10
app/src/main/res/values/colors.xml
··· 1 + <?xml version="1.0" encoding="utf-8"?> 2 + <resources> 3 + <color name="purple_200">#FFBB86FC</color> 4 + <color name="purple_500">#FF6200EE</color> 5 + <color name="purple_700">#FF3700B3</color> 6 + <color name="teal_200">#FF03DAC5</color> 7 + <color name="teal_700">#FF018786</color> 8 + <color name="black">#FF000000</color> 9 + <color name="white">#FFFFFFFF</color> 10 + </resources>
+7
app/src/main/res/values/strings.xml
··· 1 + <resources> 2 + <string name="app_name">Mastodon Redirect</string> 3 + 4 + <string name="selected_app">Selected App</string> 5 + <string name="megalodon">Megalodon</string> 6 + <string name="fedilab">Fedilab</string> 7 + </resources>
+19
app/src/main/res/values/themes.xml
··· 1 + <?xml version="1.0" encoding="utf-8"?> 2 + <resources> 3 + <style name="Theme.MastodonRedirect" parent="Theme.Material3.DayNight"> 4 + <item name="android:windowActionBar">false</item> 5 + <item name="windowNoTitle">true</item> 6 + </style> 7 + 8 + <style name="AppTheme.Transparent" parent="Theme.MastodonRedirect"> 9 + <item name="android:windowIsTranslucent">true</item> 10 + <item name="android:windowBackground">@android:color/transparent</item> 11 + <item name="android:windowContentOverlay">@null</item> 12 + <item name="android:windowNoTitle">true</item> 13 + <item name="android:windowIsFloating">true</item> 14 + <item name="android:backgroundDimEnabled">false</item> 15 + <item name="android:windowEnterAnimation">@null</item> 16 + <item name="android:windowExitAnimation">@null</item> 17 + <item name="android:windowNoDisplay">true</item> 18 + </style> 19 + </resources>
+13
app/src/main/res/xml/backup_rules.xml
··· 1 + <?xml version="1.0" encoding="utf-8"?><!-- 2 + Sample backup rules file; uncomment and customize as necessary. 3 + See https://developer.android.com/guide/topics/data/autobackup 4 + for details. 5 + Note: This file is ignored for devices older that API 31 6 + See https://developer.android.com/about/versions/12/backup-restore 7 + --> 8 + <full-backup-content> 9 + <!-- 10 + <include domain="sharedpref" path="."/> 11 + <exclude domain="sharedpref" path="device.xml"/> 12 + --> 13 + </full-backup-content>
+19
app/src/main/res/xml/data_extraction_rules.xml
··· 1 + <?xml version="1.0" encoding="utf-8"?><!-- 2 + Sample data extraction rules file; uncomment and customize as necessary. 3 + See https://developer.android.com/about/versions/12/backup-restore#xml-changes 4 + for details. 5 + --> 6 + <data-extraction-rules> 7 + <cloud-backup> 8 + <!-- TODO: Use <include> and <exclude> to control what is backed up. 9 + <include .../> 10 + <exclude .../> 11 + --> 12 + </cloud-backup> 13 + <!-- 14 + <device-transfer> 15 + <include .../> 16 + <exclude .../> 17 + </device-transfer> 18 + --> 19 + </data-extraction-rules>
+17
app/src/test/java/dev/zwander/mastodonredirect/ExampleUnitTest.kt
··· 1 + package dev.zwander.mastodonredirect 2 + 3 + import org.junit.Test 4 + 5 + import org.junit.Assert.* 6 + 7 + /** 8 + * Example local unit test, which will execute on the development machine (host). 9 + * 10 + * See [testing documentation](http://d.android.com/tools/testing). 11 + */ 12 + class ExampleUnitTest { 13 + @Test 14 + fun addition_isCorrect() { 15 + assertEquals(4, 2 + 2) 16 + } 17 + }
+7
build.gradle.kts
··· 1 + // Top-level build file where you can add configuration options common to all sub-projects/modules. 2 + @Suppress("DSL_SCOPE_VIOLATION") // TODO: Remove once KTIJ-19369 is fixed 3 + plugins { 4 + alias(libs.plugins.androidApplication) apply false 5 + alias(libs.plugins.kotlinAndroid) apply false 6 + } 7 + true // Needed to make the Suppress annotation work for the plugins block
+23
gradle.properties
··· 1 + # Project-wide Gradle settings. 2 + # IDE (e.g. Android Studio) users: 3 + # Gradle settings configured through the IDE *will override* 4 + # any settings specified in this file. 5 + # For more details on how to configure your build environment visit 6 + # http://www.gradle.org/docs/current/userguide/build_environment.html 7 + # Specifies the JVM arguments used for the daemon process. 8 + # The setting is particularly useful for tweaking memory settings. 9 + org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8 10 + # When configured, Gradle will run in incubating parallel mode. 11 + # This option should only be used with decoupled projects. More details, visit 12 + # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects 13 + # org.gradle.parallel=true 14 + # AndroidX package structure to make it clearer which packages are bundled with the 15 + # Android operating system, and which are packaged with your app's APK 16 + # https://developer.android.com/topic/libraries/support-library/androidx-rn 17 + android.useAndroidX=true 18 + # Kotlin code style for this project: "official" or "obsolete": 19 + kotlin.code.style=official 20 + # Enables namespacing of each library's R class so that its R class includes only the 21 + # resources declared in the library itself and none from the library's dependencies, 22 + # thereby reducing the size of the R class for that library 23 + android.nonTransitiveRClass=true
+37
gradle/libs.versions.toml
··· 1 + [versions] 2 + agp = "8.2.0-alpha11" 3 + kotlin = "1.8.21" 4 + core-ktx = "1.10.1" 5 + junit = "4.13.2" 6 + androidx-test-ext-junit = "1.1.5" 7 + espresso-core = "3.5.1" 8 + lifecycle-runtime-ktx = "2.6.1" 9 + activity-compose = "1.7.2" 10 + compose-bom = "2023.06.01" 11 + material-components = "1.9.0" 12 + preferences = "1.2.0" 13 + 14 + [libraries] 15 + core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "core-ktx" } 16 + junit = { group = "junit", name = "junit", version.ref = "junit" } 17 + androidx-test-ext-junit = { group = "androidx.test.ext", name = "junit", version.ref = "androidx-test-ext-junit" } 18 + espresso-core = { group = "androidx.test.espresso", name = "espresso-core", version.ref = "espresso-core" } 19 + lifecycle-runtime-ktx = { group = "androidx.lifecycle", name = "lifecycle-runtime-ktx", version.ref = "lifecycle-runtime-ktx" } 20 + activity-compose = { group = "androidx.activity", name = "activity-compose", version.ref = "activity-compose" } 21 + compose-bom = { group = "androidx.compose", name = "compose-bom", version.ref = "compose-bom" } 22 + ui = { group = "androidx.compose.ui", name = "ui" } 23 + ui-graphics = { group = "androidx.compose.ui", name = "ui-graphics" } 24 + ui-tooling = { group = "androidx.compose.ui", name = "ui-tooling" } 25 + ui-tooling-preview = { group = "androidx.compose.ui", name = "ui-tooling-preview" } 26 + ui-test-manifest = { group = "androidx.compose.ui", name = "ui-test-manifest" } 27 + ui-test-junit4 = { group = "androidx.compose.ui", name = "ui-test-junit4" } 28 + material3 = { group = "androidx.compose.material3", name = "material3" } 29 + materialComponents = { group = "com.google.android.material", name = "material", version.ref = "material-components" } 30 + preferences = { group = "androidx.preference", name = "preference", version.ref = "preferences" } 31 + 32 + [plugins] 33 + androidApplication = { id = "com.android.application", version.ref = "agp" } 34 + kotlinAndroid = { id = "org.jetbrains.kotlin.android", version.ref = "kotlin" } 35 + 36 + [bundles] 37 +
gradle/wrapper/gradle-wrapper.jar

This is a binary file and will not be displayed.

+6
gradle/wrapper/gradle-wrapper.properties
··· 1 + #Sun Jul 09 21:08:01 EDT 2023 2 + distributionBase=GRADLE_USER_HOME 3 + distributionPath=wrapper/dists 4 + distributionUrl=https\://services.gradle.org/distributions/gradle-8.1-bin.zip 5 + zipStoreBase=GRADLE_USER_HOME 6 + zipStorePath=wrapper/dists
+185
gradlew
··· 1 + #!/usr/bin/env sh 2 + 3 + # 4 + # Copyright 2015 the original author or authors. 5 + # 6 + # Licensed under the Apache License, Version 2.0 (the "License"); 7 + # you may not use this file except in compliance with the License. 8 + # You may obtain a copy of the License at 9 + # 10 + # https://www.apache.org/licenses/LICENSE-2.0 11 + # 12 + # Unless required by applicable law or agreed to in writing, software 13 + # distributed under the License is distributed on an "AS IS" BASIS, 14 + # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 15 + # See the License for the specific language governing permissions and 16 + # limitations under the License. 17 + # 18 + 19 + ############################################################################## 20 + ## 21 + ## Gradle start up script for UN*X 22 + ## 23 + ############################################################################## 24 + 25 + # Attempt to set APP_HOME 26 + # Resolve links: $0 may be a link 27 + PRG="$0" 28 + # Need this for relative symlinks. 29 + while [ -h "$PRG" ] ; do 30 + ls=`ls -ld "$PRG"` 31 + link=`expr "$ls" : '.*-> \(.*\)$'` 32 + if expr "$link" : '/.*' > /dev/null; then 33 + PRG="$link" 34 + else 35 + PRG=`dirname "$PRG"`"/$link" 36 + fi 37 + done 38 + SAVED="`pwd`" 39 + cd "`dirname \"$PRG\"`/" >/dev/null 40 + APP_HOME="`pwd -P`" 41 + cd "$SAVED" >/dev/null 42 + 43 + APP_NAME="Gradle" 44 + APP_BASE_NAME=`basename "$0"` 45 + 46 + # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 47 + DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' 48 + 49 + # Use the maximum available, or set MAX_FD != -1 to use that value. 50 + MAX_FD="maximum" 51 + 52 + warn () { 53 + echo "$*" 54 + } 55 + 56 + die () { 57 + echo 58 + echo "$*" 59 + echo 60 + exit 1 61 + } 62 + 63 + # OS specific support (must be 'true' or 'false'). 64 + cygwin=false 65 + msys=false 66 + darwin=false 67 + nonstop=false 68 + case "`uname`" in 69 + CYGWIN* ) 70 + cygwin=true 71 + ;; 72 + Darwin* ) 73 + darwin=true 74 + ;; 75 + MINGW* ) 76 + msys=true 77 + ;; 78 + NONSTOP* ) 79 + nonstop=true 80 + ;; 81 + esac 82 + 83 + CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 84 + 85 + 86 + # Determine the Java command to use to start the JVM. 87 + if [ -n "$JAVA_HOME" ] ; then 88 + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 89 + # IBM's JDK on AIX uses strange locations for the executables 90 + JAVACMD="$JAVA_HOME/jre/sh/java" 91 + else 92 + JAVACMD="$JAVA_HOME/bin/java" 93 + fi 94 + if [ ! -x "$JAVACMD" ] ; then 95 + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 96 + 97 + Please set the JAVA_HOME variable in your environment to match the 98 + location of your Java installation." 99 + fi 100 + else 101 + JAVACMD="java" 102 + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 103 + 104 + Please set the JAVA_HOME variable in your environment to match the 105 + location of your Java installation." 106 + fi 107 + 108 + # Increase the maximum file descriptors if we can. 109 + if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then 110 + MAX_FD_LIMIT=`ulimit -H -n` 111 + if [ $? -eq 0 ] ; then 112 + if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 113 + MAX_FD="$MAX_FD_LIMIT" 114 + fi 115 + ulimit -n $MAX_FD 116 + if [ $? -ne 0 ] ; then 117 + warn "Could not set maximum file descriptor limit: $MAX_FD" 118 + fi 119 + else 120 + warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 121 + fi 122 + fi 123 + 124 + # For Darwin, add options to specify how the application appears in the dock 125 + if $darwin; then 126 + GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 127 + fi 128 + 129 + # For Cygwin or MSYS, switch paths to Windows format before running java 130 + if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then 131 + APP_HOME=`cygpath --path --mixed "$APP_HOME"` 132 + CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 133 + 134 + JAVACMD=`cygpath --unix "$JAVACMD"` 135 + 136 + # We build the pattern for arguments to be converted via cygpath 137 + ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 138 + SEP="" 139 + for dir in $ROOTDIRSRAW ; do 140 + ROOTDIRS="$ROOTDIRS$SEP$dir" 141 + SEP="|" 142 + done 143 + OURCYGPATTERN="(^($ROOTDIRS))" 144 + # Add a user-defined pattern to the cygpath arguments 145 + if [ "$GRADLE_CYGPATTERN" != "" ] ; then 146 + OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 147 + fi 148 + # Now convert the arguments - kludge to limit ourselves to /bin/sh 149 + i=0 150 + for arg in "$@" ; do 151 + CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 152 + CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 153 + 154 + if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 155 + eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 156 + else 157 + eval `echo args$i`="\"$arg\"" 158 + fi 159 + i=`expr $i + 1` 160 + done 161 + case $i in 162 + 0) set -- ;; 163 + 1) set -- "$args0" ;; 164 + 2) set -- "$args0" "$args1" ;; 165 + 3) set -- "$args0" "$args1" "$args2" ;; 166 + 4) set -- "$args0" "$args1" "$args2" "$args3" ;; 167 + 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 168 + 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 169 + 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 170 + 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 171 + 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 172 + esac 173 + fi 174 + 175 + # Escape application args 176 + save () { 177 + for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done 178 + echo " " 179 + } 180 + APP_ARGS=`save "$@"` 181 + 182 + # Collect all arguments for the java command, following the shell quoting and substitution rules 183 + eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" 184 + 185 + exec "$JAVACMD" "$@"
+89
gradlew.bat
··· 1 + @rem 2 + @rem Copyright 2015 the original author or authors. 3 + @rem 4 + @rem Licensed under the Apache License, Version 2.0 (the "License"); 5 + @rem you may not use this file except in compliance with the License. 6 + @rem You may obtain a copy of the License at 7 + @rem 8 + @rem https://www.apache.org/licenses/LICENSE-2.0 9 + @rem 10 + @rem Unless required by applicable law or agreed to in writing, software 11 + @rem distributed under the License is distributed on an "AS IS" BASIS, 12 + @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 + @rem See the License for the specific language governing permissions and 14 + @rem limitations under the License. 15 + @rem 16 + 17 + @if "%DEBUG%" == "" @echo off 18 + @rem ########################################################################## 19 + @rem 20 + @rem Gradle startup script for Windows 21 + @rem 22 + @rem ########################################################################## 23 + 24 + @rem Set local scope for the variables with windows NT shell 25 + if "%OS%"=="Windows_NT" setlocal 26 + 27 + set DIRNAME=%~dp0 28 + if "%DIRNAME%" == "" set DIRNAME=. 29 + set APP_BASE_NAME=%~n0 30 + set APP_HOME=%DIRNAME% 31 + 32 + @rem Resolve any "." and ".." in APP_HOME to make it shorter. 33 + for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi 34 + 35 + @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 36 + set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" 37 + 38 + @rem Find java.exe 39 + if defined JAVA_HOME goto findJavaFromJavaHome 40 + 41 + set JAVA_EXE=java.exe 42 + %JAVA_EXE% -version >NUL 2>&1 43 + if "%ERRORLEVEL%" == "0" goto execute 44 + 45 + echo. 46 + echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 47 + echo. 48 + echo Please set the JAVA_HOME variable in your environment to match the 49 + echo location of your Java installation. 50 + 51 + goto fail 52 + 53 + :findJavaFromJavaHome 54 + set JAVA_HOME=%JAVA_HOME:"=% 55 + set JAVA_EXE=%JAVA_HOME%/bin/java.exe 56 + 57 + if exist "%JAVA_EXE%" goto execute 58 + 59 + echo. 60 + echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 61 + echo. 62 + echo Please set the JAVA_HOME variable in your environment to match the 63 + echo location of your Java installation. 64 + 65 + goto fail 66 + 67 + :execute 68 + @rem Setup the command line 69 + 70 + set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 71 + 72 + 73 + @rem Execute Gradle 74 + "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* 75 + 76 + :end 77 + @rem End local scope for the variables with windows NT shell 78 + if "%ERRORLEVEL%"=="0" goto mainEnd 79 + 80 + :fail 81 + rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 82 + rem the _cmd.exe /c_ return code! 83 + if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 84 + exit /b 1 85 + 86 + :mainEnd 87 + if "%OS%"=="Windows_NT" endlocal 88 + 89 + :omega
+18
settings.gradle.kts
··· 1 + pluginManagement { 2 + repositories { 3 + google() 4 + mavenCentral() 5 + gradlePluginPortal() 6 + } 7 + } 8 + dependencyResolutionManagement { 9 + repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS) 10 + repositories { 11 + google() 12 + mavenCentral() 13 + } 14 + } 15 + 16 + rootProject.name = "Mastodon Redirect" 17 + include(":app") 18 +