@ -0,0 +1,57 @@
|
||||
apply plugin: 'com.android.application' |
||||
apply plugin: 'kotlin-android' |
||||
apply plugin: 'kotlin-android-extensions' |
||||
|
||||
android { |
||||
compileSdkVersion 30 |
||||
buildToolsVersion "30.0.2" |
||||
|
||||
defaultConfig { |
||||
applicationId "org.mosad.teapod" |
||||
minSdkVersion 23 |
||||
targetSdkVersion 30 |
||||
versionCode 1 |
||||
versionName "1.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 |
||||
} |
||||
kotlinOptions { |
||||
jvmTarget = '1.8' |
||||
} |
||||
} |
||||
|
||||
dependencies { |
||||
implementation fileTree(dir: "libs", include: ["*.jar"]) |
||||
implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version" |
||||
implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.9' |
||||
implementation 'androidx.core:core-ktx:1.3.2' |
||||
implementation 'androidx.appcompat:appcompat:1.2.0' |
||||
implementation 'com.google.android.material:material:1.2.1' |
||||
implementation 'androidx.constraintlayout:constraintlayout:2.0.1' |
||||
implementation 'androidx.navigation:navigation-fragment:2.3.0' |
||||
implementation 'androidx.navigation:navigation-ui:2.3.0' |
||||
implementation 'androidx.lifecycle:lifecycle-extensions:2.2.0' |
||||
implementation 'androidx.navigation:navigation-fragment-ktx:2.3.0' |
||||
implementation 'androidx.navigation:navigation-ui-ktx:2.3.0' |
||||
|
||||
implementation 'org.jsoup:jsoup:1.13.1' |
||||
implementation 'com.github.bumptech.glide:glide:4.11.0' |
||||
implementation 'androidx.legacy:legacy-support-v4:1.0.0' |
||||
implementation 'androidx.recyclerview:recyclerview:1.1.0' |
||||
|
||||
testImplementation 'junit:junit:4.12' |
||||
androidTestImplementation 'androidx.test.ext:junit:1.1.2' |
||||
androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0' |
||||
|
||||
} |
@ -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 |
@ -0,0 +1,24 @@
|
||||
package org.mosad.teapod |
||||
|
||||
import androidx.test.platform.app.InstrumentationRegistry |
||||
import androidx.test.ext.junit.runners.AndroidJUnit4 |
||||
|
||||
import org.junit.Test |
||||
import org.junit.runner.RunWith |
||||
|
||||
import org.junit.Assert.* |
||||
|
||||
/** |
||||
* Instrumented test, which will execute on an Android device. |
||||
* |
||||
* See [testing documentation](http://d.android.com/tools/testing). |
||||
*/ |
||||
@RunWith(AndroidJUnit4::class) |
||||
class ExampleInstrumentedTest { |
||||
@Test |
||||
fun useAppContext() { |
||||
// Context of the app under test. |
||||
val appContext = InstrumentationRegistry.getInstrumentation().targetContext |
||||
assertEquals("org.mosad.teapod", appContext.packageName) |
||||
} |
||||
} |
@ -0,0 +1,24 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android" |
||||
package="org.mosad.teapod"> |
||||
<uses-permission android:name="android.permission.INTERNET"/> |
||||
|
||||
<application |
||||
android:allowBackup="true" |
||||
android:icon="@mipmap/ic_launcher" |
||||
android:label="@string/app_name" |
||||
android:roundIcon="@mipmap/ic_launcher_round" |
||||
android:supportsRtl="true" |
||||
android:theme="@style/AppTheme"> |
||||
<activity |
||||
android:name=".MainActivity" |
||||
android:label="@string/app_name"> |
||||
<intent-filter> |
||||
<action android:name="android.intent.action.MAIN" /> |
||||
|
||||
<category android:name="android.intent.category.LAUNCHER" /> |
||||
</intent-filter> |
||||
</activity> |
||||
</application> |
||||
|
||||
</manifest> |
@ -0,0 +1,35 @@
|
||||
package org.mosad.teapod |
||||
|
||||
import android.os.Bundle |
||||
import com.google.android.material.bottomnavigation.BottomNavigationView |
||||
import androidx.appcompat.app.AppCompatActivity |
||||
import androidx.navigation.findNavController |
||||
import androidx.navigation.ui.AppBarConfiguration |
||||
import androidx.navigation.ui.setupActionBarWithNavController |
||||
import androidx.navigation.ui.setupWithNavController |
||||
import kotlinx.coroutines.GlobalScope |
||||
import kotlinx.coroutines.launch |
||||
import org.mosad.teapod.parser.AoDParser |
||||
|
||||
class MainActivity : AppCompatActivity() { |
||||
|
||||
override fun onCreate(savedInstanceState: Bundle?) { |
||||
super.onCreate(savedInstanceState) |
||||
setContentView(R.layout.activity_main) |
||||
val navView: BottomNavigationView = findViewById(R.id.nav_view) |
||||
|
||||
val navController = findNavController(R.id.nav_host_fragment) |
||||
// Passing each menu ID as a set of Ids because each |
||||
// menu should be considered as top level destinations. |
||||
val appBarConfiguration = AppBarConfiguration(setOf( |
||||
R.id.navigation_home, R.id.navigation_library, R.id.navigation_search)) |
||||
setupActionBarWithNavController(navController, appBarConfiguration) |
||||
navView.setupWithNavController(navController) |
||||
|
||||
load() |
||||
} |
||||
|
||||
private fun load() { |
||||
// TODO |
||||
} |
||||
} |
@ -0,0 +1,57 @@
|
||||
package org.mosad.teapod.dummy |
||||
|
||||
import java.util.ArrayList |
||||
import java.util.HashMap |
||||
|
||||
/** |
||||
* Helper class for providing sample content for user interfaces created by |
||||
* Android template wizards. |
||||
* |
||||
* TODO: Replace all uses of this class before publishing your app. |
||||
*/ |
||||
object DummyContent { |
||||
|
||||
/** |
||||
* An array of sample (dummy) items. |
||||
*/ |
||||
val ITEMS: MutableList<DummyItem> = ArrayList() |
||||
|
||||
/** |
||||
* A map of sample (dummy) items, by ID. |
||||
*/ |
||||
val ITEM_MAP: MutableMap<String, DummyItem> = HashMap() |
||||
|
||||
private val COUNT = 25 |
||||
|
||||
init { |
||||
// Add some sample items. |
||||
for (i in 1..COUNT) { |
||||
addItem(createDummyItem(i)) |
||||
} |
||||
} |
||||
|
||||
private fun addItem(item: DummyItem) { |
||||
ITEMS.add(item) |
||||
ITEM_MAP.put(item.id, item) |
||||
} |
||||
|
||||
private fun createDummyItem(position: Int): DummyItem { |
||||
return DummyItem(position.toString(), "Item " + position, makeDetails(position)) |
||||
} |
||||
|
||||
private fun makeDetails(position: Int): String { |
||||
val builder = StringBuilder() |
||||
builder.append("Details about Item: ").append(position) |
||||
for (i in 0..position - 1) { |
||||
builder.append("\nMore details information here.") |
||||
} |
||||
return builder.toString() |
||||
} |
||||
|
||||
/** |
||||
* A dummy item representing a piece of content. |
||||
*/ |
||||
data class DummyItem(val id: String, val content: String, val details: String) { |
||||
override fun toString(): String = content |
||||
} |
||||
} |
@ -0,0 +1,92 @@
|
||||
package org.mosad.teapod.parser |
||||
|
||||
import kotlinx.coroutines.Dispatchers |
||||
import kotlinx.coroutines.runBlocking |
||||
import kotlinx.coroutines.withContext |
||||
import org.jsoup.Connection |
||||
import org.jsoup.Jsoup |
||||
import org.mosad.teapod.util.GUIMedia |
||||
|
||||
class AoDParser { |
||||
|
||||
private val baseURL = "https://www.anime-on-demand.de" |
||||
private val loginPath = "/users/sign_in" |
||||
|
||||
private val login = "" |
||||
private val pwd = "" |
||||
|
||||
private fun login(): MutableMap<String, String> = runBlocking { |
||||
|
||||
|
||||
val userAgent = "Mozilla/5.0 (X11; Linux x86_64; rv:80.0) Gecko/20100101 Firefox/80.0" |
||||
|
||||
withContext(Dispatchers.Default) { |
||||
val con = Jsoup.connect(baseURL) |
||||
|
||||
|
||||
// get the authenticity token |
||||
val resAuth = con.url(baseURL + loginPath)//Jsoup.connect(baseURL + loginPath) |
||||
.header("User-Agent", userAgent) |
||||
.execute() |
||||
|
||||
val authenticityToken = resAuth.parse().select("meta[name=csrf-token]").attr("content") |
||||
println("Authenticity token is: $authenticityToken") |
||||
|
||||
val cookies = resAuth.cookies() |
||||
println("cookies: $cookies") |
||||
|
||||
val data = mapOf( |
||||
Pair("user[login]", login), |
||||
Pair("user[password]", pwd), |
||||
Pair("user[remember_me]", "1"), |
||||
Pair("commit", "Einloggen"), |
||||
Pair("authenticity_token", authenticityToken) |
||||
) |
||||
|
||||
val resLogin = Jsoup.connect(baseURL + loginPath) |
||||
.method(Connection.Method.POST) |
||||
.data(data) |
||||
.postDataCharset("UTF-8") |
||||
.cookies(cookies) |
||||
.execute() |
||||
|
||||
//println(resLogin.body()) |
||||
|
||||
val loginSuccess = resLogin.body().contains("Hallo, du bist jetzt angemeldet.") |
||||
println("Status: ${resLogin.statusCode()} (${resLogin.statusMessage()}), login successful: $loginSuccess") |
||||
|
||||
return@withContext resLogin.cookies() |
||||
} |
||||
} |
||||
|
||||
// https://www.anime-on-demand.de/animes |
||||
fun listAnime(): ArrayList<GUIMedia> = runBlocking { |
||||
withContext(Dispatchers.Default) { |
||||
val cookies = login() |
||||
|
||||
|
||||
val res = Jsoup.connect("$baseURL/animes") |
||||
.cookies(cookies) |
||||
.get() |
||||
|
||||
//println(res) |
||||
|
||||
val anime = arrayListOf<GUIMedia>() |
||||
res.select("div.animebox").forEach { |
||||
val media = GUIMedia( |
||||
it.select("h3.animebox-title").text(), |
||||
it.select("p.animebox-image").select("img").attr("src"), |
||||
it.select("p.animebox-link").select("a").attr("href"), |
||||
it.select("p.animebox-shorttext").text() |
||||
) |
||||
|
||||
anime.add(media) |
||||
} |
||||
|
||||
println("got ${anime.size} anime") |
||||
|
||||
return@withContext anime |
||||
} |
||||
} |
||||
|
||||
} |
@ -0,0 +1,22 @@
|
||||
package org.mosad.teapod.preferences |
||||
|
||||
object Preferences { |
||||
|
||||
var login = "" |
||||
internal set |
||||
var password = "" |
||||
internal set |
||||
|
||||
|
||||
fun saveCredentials(login: String, password: String) { |
||||
this.login = login |
||||
this.password = password |
||||
|
||||
// TODO save |
||||
} |
||||
|
||||
fun load() { |
||||
// TODO |
||||
|
||||
} |
||||
} |
@ -0,0 +1,22 @@
|
||||
package org.mosad.teapod.ui.account |
||||
|
||||
import android.os.Bundle |
||||
import android.view.LayoutInflater |
||||
import android.view.View |
||||
import android.view.ViewGroup |
||||
import androidx.fragment.app.Fragment |
||||
import kotlinx.android.synthetic.main.fragment_account.* |
||||
import org.mosad.teapod.R |
||||
|
||||
class AccountFragment : Fragment() { |
||||
|
||||
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { |
||||
return inflater.inflate(R.layout.fragment_account, container, false) |
||||
} |
||||
|
||||
override fun onViewCreated(view: View, savedInstanceState: Bundle?) { |
||||
super.onViewCreated(view, savedInstanceState) |
||||
|
||||
text_account.text = "This is the Account Fragment" |
||||
} |
||||
} |
@ -0,0 +1,28 @@
|
||||
package org.mosad.teapod.ui.components |
||||
|
||||
import android.content.Context |
||||
import android.graphics.Bitmap |
||||
import android.widget.LinearLayout |
||||
import com.bumptech.glide.Glide |
||||
import kotlinx.android.synthetic.main.linear_media.view.* |
||||
import org.mosad.teapod.R |
||||
|
||||
|
||||
class MediaLinearLayout(context: Context?) : LinearLayout(context) { |
||||
|
||||
init { |
||||
inflate(context, R.layout.linear_media, this) |
||||
} |
||||
|
||||
fun setTitle(title: String): MediaLinearLayout = apply { |
||||
text_title.text = title |
||||
} |
||||
|
||||
fun setPoster(url: String): MediaLinearLayout = apply { |
||||
Glide.with(context) |
||||
.load(url) |
||||
.into(image_poster) |
||||
} |
||||
|
||||
|
||||
} |
@ -0,0 +1,32 @@
|
||||
package org.mosad.teapod.ui.home |
||||
|
||||
import android.os.Bundle |
||||
import android.view.LayoutInflater |
||||
import android.view.View |
||||
import android.view.ViewGroup |
||||
import android.widget.TextView |
||||
import androidx.fragment.app.Fragment |
||||
import androidx.lifecycle.Observer |
||||
import androidx.lifecycle.ViewModelProviders |
||||
import org.mosad.teapod.R |
||||
|
||||
class HomeFragment : Fragment() { |
||||
|
||||
private lateinit var homeViewModel: HomeViewModel |
||||
|
||||
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { |
||||
homeViewModel = ViewModelProviders.of(this).get(HomeViewModel::class.java) |
||||
val root = inflater.inflate(R.layout.fragment_home, container, false) |
||||
val textView: TextView = root.findViewById(R.id.text_home) |
||||
homeViewModel.text.observe(viewLifecycleOwner, Observer { |
||||
textView.text = it |
||||
}) |
||||
return root |
||||
} |
||||
|
||||
override fun onViewCreated(view: View, savedInstanceState: Bundle?) { |
||||
super.onViewCreated(view, savedInstanceState) |
||||
|
||||
println("created!!!!") |
||||
} |
||||
} |
@ -0,0 +1,13 @@
|
||||
package org.mosad.teapod.ui.home |
||||
|
||||
import androidx.lifecycle.LiveData |
||||
import androidx.lifecycle.MutableLiveData |
||||
import androidx.lifecycle.ViewModel |
||||
|
||||
class HomeViewModel : ViewModel() { |
||||
|
||||
private val _text = MutableLiveData<String>().apply { |
||||
value = "This is the home Fragment" |
||||
} |
||||
val text: LiveData<String> = _text |
||||
} |
@ -0,0 +1,57 @@
|
||||
package org.mosad.teapod.ui.library |
||||
|
||||
import android.os.Bundle |
||||
import android.view.LayoutInflater |
||||
import android.view.View |
||||
import android.view.ViewGroup |
||||
import android.widget.ArrayAdapter |
||||
import androidx.fragment.app.Fragment |
||||
import kotlinx.android.synthetic.main.fragment_library.* |
||||
import kotlinx.coroutines.Dispatchers |
||||
import kotlinx.coroutines.GlobalScope |
||||
import kotlinx.coroutines.launch |
||||
import kotlinx.coroutines.withContext |
||||
import org.mosad.teapod.R |
||||
import org.mosad.teapod.parser.AoDParser |
||||
import org.mosad.teapod.ui.components.MediaLinearLayout |
||||
import org.mosad.teapod.util.CustomAdapter |
||||
import org.mosad.teapod.util.GUIMedia |
||||
|
||||
class LibraryFragment : Fragment() { |
||||
|
||||
private var mediaList = arrayListOf<GUIMedia>() |
||||
private lateinit var adapter : CustomAdapter |
||||
|
||||
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? { |
||||
val root = inflater.inflate(R.layout.fragment_library, container, false) |
||||
return root |
||||
} |
||||
|
||||
override fun onViewCreated(view: View, savedInstanceState: Bundle?) { |
||||
super.onViewCreated(view, savedInstanceState) |
||||
|
||||
|
||||
adapter = CustomAdapter(requireContext(),layoutInflater, mediaList)//ArrayAdapter(requireContext(), R.layout.linear_media, R.id.text_title, mediaList) |
||||
|
||||
list_library.adapter = adapter |
||||
|
||||
text_dashboard.text = "Loading Animes ..." |
||||
|
||||
loadAnimeList() |
||||
} |
||||
|
||||
private fun loadAnimeList() = GlobalScope.launch { |
||||
|
||||
val parser = AoDParser() |
||||
mediaList = parser.listAnime() |
||||
|
||||
text_dashboard.text = "got ${mediaList.size} animes" |
||||
|
||||
withContext(Dispatchers.Main) { |
||||
adapter.notifyDataSetChanged() |
||||
println("notifiyed") |
||||
} |
||||
|
||||
} |
||||
|
||||
} |
@ -0,0 +1,13 @@
|
||||
package org.mosad.teapod.ui.library |
||||
|
||||
import androidx.lifecycle.LiveData |
||||
import androidx.lifecycle.MutableLiveData |
||||
import androidx.lifecycle.ViewModel |
||||
|
||||
class LibraryViewModel : ViewModel() { |
||||
|
||||
private val _text = MutableLiveData<String>().apply { |
||||
value = "This is the library Fragment" |
||||
} |
||||
val text: LiveData<String> = _text |
||||
} |
@ -0,0 +1,31 @@
|
||||
package org.mosad.teapod.ui.search |
||||
|
||||
import android.os.Bundle |
||||
import android.view.LayoutInflater |
||||
import android.view.View |
||||
import android.view.ViewGroup |
||||
import android.widget.TextView |
||||
import androidx.fragment.app.Fragment |
||||
import androidx.lifecycle.Observer |
||||
import androidx.lifecycle.ViewModelProviders |
||||
import org.mosad.teapod.R |
||||
|
||||
class SearchFragment : Fragment() { |
||||
|
||||
private lateinit var notificationsViewModel: SearchViewModel |
||||
|
||||
override fun onCreateView( |
||||
inflater: LayoutInflater, |
||||
container: ViewGroup?, |
||||
savedInstanceState: Bundle? |
||||
): View? { |
||||
notificationsViewModel = |
||||
ViewModelProviders.of(this).get(SearchViewModel::class.java) |
||||
val root = inflater.inflate(R.layout.fragment_search, container, false) |
||||
val textView: TextView = root.findViewById(R.id.text_notifications) |
||||
notificationsViewModel.text.observe(viewLifecycleOwner, Observer { |
||||
textView.text = it |
||||
}) |
||||
return root |
||||
} |
||||
} |
@ -0,0 +1,13 @@
|
||||
package org.mosad.teapod.ui.search |
||||
|
||||
import androidx.lifecycle.LiveData |
||||
import androidx.lifecycle.MutableLiveData |
||||
import androidx.lifecycle.ViewModel |
||||
|
||||
class SearchViewModel : ViewModel() { |
||||
|
||||
private val _text = MutableLiveData<String>().apply { |
||||
value = "This is the search Fragment" |
||||
} |
||||
val text: LiveData<String> = _text |
||||
} |
@ -0,0 +1,33 @@
|
||||
package org.mosad.teapod.util |
||||
|
||||
import android.content.Context |
||||
import android.view.LayoutInflater |
||||
import android.view.View |
||||
import android.view.ViewGroup |
||||
import android.widget.ArrayAdapter |
||||
import android.widget.ImageView |
||||
import android.widget.TextView |
||||
import com.bumptech.glide.Glide |
||||
import org.mosad.teapod.R |
||||
|
||||
class CustomAdapter(context: Context, private val layoutInflater: LayoutInflater, private val media: ArrayList<GUIMedia>) : ArrayAdapter<GUIMedia>(context, R.layout.linear_media) { |
||||
|
||||
override fun getView(position: Int, convertView: View?, parent: ViewGroup): View { |
||||
println("view") |
||||
|
||||
return if (convertView == null) { |
||||
val guiMedia = layoutInflater.inflate(R.layout.linear_media, null) |
||||
|
||||
val textTitle = guiMedia.findViewById<TextView>(R.id.text_title) |
||||
val imagePoster = guiMedia.findViewById<ImageView>(R.id.image_poster) |
||||
|
||||
textTitle.text = media[position].title |
||||
Glide.with(context).load(media[position].imageLink).into(imagePoster) |
||||
|
||||
guiMedia |
||||
} else { |
||||
convertView |
||||
} |
||||
} |
||||
|
||||
} |
@ -0,0 +1,3 @@
|
||||
package org.mosad.teapod.util |
||||
|
||||
data class GUIMedia(val title: String, val imageLink: String, val shortDesc : String, val link: String) |
@ -0,0 +1,30 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:aapt="http://schemas.android.com/aapt" |
||||
android:width="108dp" |
||||
android:height="108dp" |
||||
android:viewportWidth="108" |
||||
android:viewportHeight="108"> |
||||
<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"> |
||||
<aapt:attr name="android:fillColor"> |
||||
<gradient |
||||
android:endX="85.84757" |
||||
android:endY="92.4963" |
||||
android:startX="42.9492" |
||||
android:startY="49.59793" |
||||
android:type="linear"> |
||||
<item |
||||
android:color="#44000000" |
||||
android:offset="0.0" /> |
||||
<item |
||||
android:color="#00000000" |
||||
android:offset="1.0" /> |
||||
</gradient> |
||||
</aapt:attr> |
||||
</path> |
||||
<path |
||||
android:fillColor="#FFFFFF" |
||||
android:fillType="nonZero" |
||||
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" |
||||
android:strokeWidth="1" |
||||
android:strokeColor="#00000000" /> |
||||
</vector> |
@ -0,0 +1,10 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android" |
||||
android:width="24dp" |
||||
android:height="24dp" |
||||
android:viewportWidth="24" |
||||
android:viewportHeight="24" |
||||
android:tint="?attr/colorControlNormal"> |
||||
<path |
||||
android:fillColor="@android:color/white" |
||||
android:pathData="M3,5v14c0,1.1 0.89,2 2,2h14c1.1,0 2,-0.9 2,-2L21,5c0,-1.1 -0.9,-2 -2,-2L5,3c-1.11,0 -2,0.9 -2,2zM15,9c0,1.66 -1.34,3 -3,3s-3,-1.34 -3,-3 1.34,-3 3,-3 3,1.34 3,3zM6,17c0,-2 4,-3.1 6,-3.1s6,1.1 6,3.1v1L6,18v-1z"/> |
||||
</vector> |
@ -0,0 +1,10 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android" |
||||
android:width="24dp" |
||||
android:height="24dp" |
||||
android:viewportWidth="24" |
||||
android:viewportHeight="24" |
||||
android:tint="?attr/colorControlNormal"> |
||||
<path |
||||
android:fillColor="@android:color/white" |
||||
android:pathData="M15.5,14h-0.79l-0.28,-0.27C15.41,12.59 16,11.11 16,9.5 16,5.91 13.09,3 9.5,3S3,5.91 3,9.5 5.91,16 9.5,16c1.61,0 3.09,-0.59 4.23,-1.57l0.27,0.28v0.79l5,4.99L20.49,19l-4.99,-5zM9.5,14C7.01,14 5,11.99 5,9.5S7.01,5 9.5,5 14,7.01 14,9.5 11.99,14 9.5,14z"/> |
||||
</vector> |
@ -0,0 +1,10 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android" |
||||
android:width="24dp" |
||||
android:height="24dp" |
||||
android:viewportWidth="24" |
||||
android:viewportHeight="24" |
||||
android:tint="?attr/colorControlNormal"> |
||||
<path |
||||
android:fillColor="@android:color/white" |
||||
android:pathData="M4,6L2,6v14c0,1.1 0.9,2 2,2h14v-2L4,20L4,6zM20,2L8,2c-1.1,0 -2,0.9 -2,2v12c0,1.1 0.9,2 2,2h12c1.1,0 2,-0.9 2,-2L22,4c0,-1.1 -0.9,-2 -2,-2zM12,14.5v-9l6,4.5 -6,4.5z"/> |
||||
</vector> |
@ -0,0 +1,9 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android" |
||||
android:width="24dp" |
||||
android:height="24dp" |
||||
android:viewportWidth="24.0" |
||||
android:viewportHeight="24.0"> |
||||
<path |
||||
android:fillColor="#FF000000" |
||||
android:pathData="M10,20v-6h4v6h5v-8h3L12,3 2,12h3v8z" /> |
||||
</vector> |
@ -0,0 +1,170 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android" |
||||
android:width="108dp" |
||||
android:height="108dp" |
||||
android:viewportWidth="108" |
||||
android:viewportHeight="108"> |
||||
<path |
||||
android:fillColor="#3DDC84" |
||||
android:pathData="M0,0h108v108h-108z" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M9,0L9,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M19,0L19,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M29,0L29,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M39,0L39,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M49,0L49,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M59,0L59,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M69,0L69,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M79,0L79,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M89,0L89,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M99,0L99,108" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,9L108,9" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,19L108,19" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,29L108,29" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,39L108,39" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,49L108,49" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,59L108,59" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,69L108,69" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,79L108,79" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,89L108,89" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M0,99L108,99" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M19,29L89,29" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M19,39L89,39" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M19,49L89,49" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M19,59L89,59" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M19,69L89,69" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M19,79L89,79" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M29,19L29,89" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M39,19L39,89" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M49,19L49,89" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M59,19L59,89" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M69,19L69,89" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
<path |
||||
android:fillColor="#00000000" |
||||
android:pathData="M79,19L79,89" |
||||
android:strokeWidth="0.8" |
||||
android:strokeColor="#33FFFFFF" /> |
||||
</vector> |
@ -0,0 +1,33 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:app="http://schemas.android.com/apk/res-auto" |
||||
android:id="@+id/container" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
android:paddingTop="?attr/actionBarSize"> |
||||
|
||||
<com.google.android.material.bottomnavigation.BottomNavigationView |
||||
android:id="@+id/nav_view" |
||||
android:layout_width="0dp" |
||||
android:layout_height="wrap_content" |
||||
android:layout_marginStart="0dp" |
||||
android:layout_marginEnd="0dp" |
||||
android:background="?android:attr/windowBackground" |
||||
app:layout_constraintBottom_toBottomOf="parent" |
||||
app:layout_constraintLeft_toLeftOf="parent" |
||||
app:layout_constraintRight_toRightOf="parent" |
||||
app:menu="@menu/bottom_nav_menu" /> |
||||
|
||||
<fragment |
||||
android:id="@+id/nav_host_fragment" |
||||
android:name="androidx.navigation.fragment.NavHostFragment" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
app:defaultNavHost="true" |
||||
app:layout_constraintBottom_toTopOf="@id/nav_view" |
||||
app:layout_constraintLeft_toLeftOf="parent" |
||||
app:layout_constraintRight_toRightOf="parent" |
||||
app:layout_constraintTop_toTopOf="parent" |
||||
app:navGraph="@navigation/mobile_navigation" /> |
||||
|
||||
</androidx.constraintlayout.widget.ConstraintLayout> |
@ -0,0 +1,22 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:app="http://schemas.android.com/apk/res-auto" |
||||
xmlns:tools="http://schemas.android.com/tools" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
tools:context=".ui.account.AccountFragment"> |
||||
|
||||
<TextView |
||||
android:id="@+id/text_account" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:layout_marginStart="8dp" |
||||
android:layout_marginTop="8dp" |
||||
android:layout_marginEnd="8dp" |
||||
android:textAlignment="center" |
||||
android:textSize="20sp" |
||||
app:layout_constraintBottom_toBottomOf="parent" |
||||
app:layout_constraintEnd_toEndOf="parent" |
||||
app:layout_constraintStart_toStartOf="parent" |
||||
app:layout_constraintTop_toTopOf="parent" /> |
||||
</androidx.constraintlayout.widget.ConstraintLayout> |
@ -0,0 +1,22 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:app="http://schemas.android.com/apk/res-auto" |
||||
xmlns:tools="http://schemas.android.com/tools" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
tools:context=".ui.home.HomeFragment"> |
||||
|
||||
<TextView |
||||
android:id="@+id/text_home" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:layout_marginStart="8dp" |
||||
android:layout_marginTop="8dp" |
||||
android:layout_marginEnd="8dp" |
||||
android:textAlignment="center" |
||||
android:textSize="20sp" |
||||
app:layout_constraintBottom_toBottomOf="parent" |
||||
app:layout_constraintEnd_toEndOf="parent" |
||||
app:layout_constraintStart_toStartOf="parent" |
||||
app:layout_constraintTop_toTopOf="parent" /> |
||||
</androidx.constraintlayout.widget.ConstraintLayout> |
@ -0,0 +1,42 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:app="http://schemas.android.com/apk/res-auto" |
||||
xmlns:tools="http://schemas.android.com/tools" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
tools:context=".ui.library.LibraryFragment"> |
||||
|
||||
<ListView |
||||
android:id="@+id/list_library" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
app:layout_constraintBottom_toBottomOf="parent" |
||||
app:layout_constraintEnd_toEndOf="parent" |
||||
app:layout_constraintStart_toStartOf="parent" |
||||
app:layout_constraintTop_toTopOf="parent" /> |
||||
|
||||
<ScrollView |
||||
android:id="@+id/scroll_library" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
android:visibility="gone"> |
||||
|
||||
<LinearLayout |
||||
android:id="@+id/linear_library" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:orientation="vertical"> |
||||
|
||||
<TextView |
||||
android:id="@+id/text_dashboard" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:layout_marginStart="8dp" |
||||
android:layout_marginTop="8dp" |
||||
android:layout_marginEnd="8dp" |
||||
android:textAlignment="center" |
||||
android:textSize="20sp" /> |
||||
</LinearLayout> |
||||
</ScrollView> |
||||
|
||||
</androidx.constraintlayout.widget.ConstraintLayout> |
@ -0,0 +1,22 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:app="http://schemas.android.com/apk/res-auto" |
||||
xmlns:tools="http://schemas.android.com/tools" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
tools:context=".ui.search.SearchFragment"> |
||||
|
||||
<TextView |
||||
android:id="@+id/text_notifications" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:layout_marginStart="8dp" |
||||
android:layout_marginTop="8dp" |
||||
android:layout_marginEnd="8dp" |
||||
android:textAlignment="center" |
||||
android:textSize="20sp" |
||||
app:layout_constraintBottom_toBottomOf="parent" |
||||
app:layout_constraintEnd_toEndOf="parent" |
||||
app:layout_constraintStart_toStartOf="parent" |
||||
app:layout_constraintTop_toTopOf="parent" /> |
||||
</androidx.constraintlayout.widget.ConstraintLayout> |
@ -0,0 +1,28 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:app="http://schemas.android.com/apk/res-auto" |
||||
xmlns:tools="http://schemas.android.com/tools" |
||||
android:id="@+id/linear_media" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:orientation="vertical" |
||||
android:paddingStart="7dp" |
||||
android:paddingTop="2dp" |
||||
android:paddingEnd="7dp" |
||||
android:paddingBottom="3dp"> |
||||
|
||||
<TextView |
||||
android:id="@+id/text_title" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:text="TextView" |
||||
android:textAlignment="center" |
||||
android:textSize="18sp" |
||||
android:textStyle="bold" /> |
||||
|
||||
<ImageView |
||||
android:id="@+id/image_poster" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
tools:srcCompat="@drawable/ic_launcher_background" /> |
||||
</LinearLayout> |
@ -0,0 +1,23 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<menu xmlns:android="http://schemas.android.com/apk/res/android"> |
||||
|
||||
<item |
||||
android:id="@+id/navigation_home" |
||||
android:icon="@drawable/ic_home_black_24dp" |
||||
android:title="@string/title_home" /> |
||||
|
||||
<item |
||||
android:id="@+id/navigation_library" |
||||
android:icon="@drawable/ic_baseline_video_library_24" |
||||
android:title="@string/title_library" /> |
||||
|
||||
<item |
||||
android:id="@+id/navigation_search" |
||||
android:icon="@drawable/ic_baseline_search_24" |
||||
android:title="@string/title_search" /> |
||||
<item |
||||
android:id="@+id/navigation_account" |
||||
android:icon="@drawable/ic_baseline_account_box_24" |
||||
android:title="@string/title_account" /> |
||||
|
||||
</menu> |
@ -0,0 +1,5 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> |
||||
<background android:drawable="@drawable/ic_launcher_background" /> |
||||
<foreground android:drawable="@drawable/ic_launcher_foreground" /> |
||||
</adaptive-icon> |
@ -0,0 +1,5 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android"> |
||||
<background android:drawable="@drawable/ic_launcher_background" /> |
||||
<foreground android:drawable="@drawable/ic_launcher_foreground" /> |
||||
</adaptive-icon> |
After Width: | Height: | Size: 3.5 KiB |
After Width: | Height: | Size: 5.2 KiB |
After Width: | Height: | Size: 2.6 KiB |
After Width: | Height: | Size: 3.3 KiB |
After Width: | Height: | Size: 4.8 KiB |
After Width: | Height: | Size: 7.3 KiB |
After Width: | Height: | Size: 7.7 KiB |
After Width: | Height: | Size: 12 KiB |
After Width: | Height: | Size: 10 KiB |
After Width: | Height: | Size: 16 KiB |
@ -0,0 +1,32 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<navigation xmlns:android="http://schemas.android.com/apk/res/android" |
||||
xmlns:app="http://schemas.android.com/apk/res-auto" |
||||
xmlns:tools="http://schemas.android.com/tools" |
||||
android:id="@+id/mobile_navigation" |
||||
app:startDestination="@+id/navigation_home"> |
||||
|
||||
<fragment |
||||
android:id="@+id/navigation_home" |
||||
android:name="org.mosad.teapod.ui.home.HomeFragment" |
||||
android:label="@string/title_home" |
||||
tools:layout="@layout/fragment_home" /> |
||||
|
||||
<fragment |
||||
android:id="@+id/navigation_library" |
||||
android:name="org.mosad.teapod.ui.library.LibraryFragment" |
||||
android:label="@string/title_library" |
||||
tools:layout="@layout/fragment_library" /> |
||||
|
||||
<fragment |
||||
android:id="@+id/navigation_search" |
||||
android:name="org.mosad.teapod.ui.search.SearchFragment" |
||||
android:label="@string/title_search" |
||||
tools:layout="@layout/fragment_search" /> |
||||
|
||||
<fragment |
||||
android:id="@+id/navigation_account" |
||||
android:name="org.mosad.teapod.ui.account.AccountFragment" |
||||
android:label="@string/title_account" |
||||
tools:layout="@layout/fragment_account" /> |
||||
|
||||
</navigation> |
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?> |
||||
<resources> |
||||
<color name="colorPrimary">#6200EE</color> |
||||
<color name="colorPrimaryDark">#3700B3</color> |
||||
<color name="colorAccent">#03DAC5</color> |
||||
</resources> |
@ -0,0 +1,6 @@
|
||||
<resources> |
||||
<!-- Default screen margins, per the Android Design guidelines. --> |
||||
<dimen name="activity_horizontal_margin">16dp</dimen> |
||||
<dimen name="activity_vertical_margin">16dp</dimen> |
||||
<dimen name="text_margin">16dp</dimen> |
||||
</resources> |
@ -0,0 +1,7 @@
|
||||
<resources> |
||||
<string name="app_name">Teapod</string> |
||||
<string name="title_home">Home</string> |
||||
<string name="title_library">Library</string> |
||||
<string name="title_search">Search</string> |
||||
<string name="title_account">Account</string> |
||||
</resources> |
@ -0,0 +1,10 @@
|
||||
<resources> |
||||
<!-- Base application theme. --> |
||||
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar"> |
||||
<!-- Customize your theme here. --> |
||||
<item name="colorPrimary">@color/colorPrimary</item> |
||||
<item name="colorPrimaryDark">@color/colorPrimaryDark</item> |
||||
<item name="colorAccent">@color/colorAccent</item> |
||||
</style> |
||||
|
||||
</resources> |
@ -0,0 +1,17 @@
|
||||
package org.mosad.teapod |
||||
|
||||
import org.junit.Test |
||||
|
||||
import org.junit.Assert.* |
||||
|
||||
/** |
||||
* Example local unit test, which will execute on the development machine (host). |
||||
* |
||||
* See [testing documentation](http://d.android.com/tools/testing). |
||||
*/ |
||||
class ExampleUnitTest { |
||||
@Test |
||||
fun addition_isCorrect() { |
||||
assertEquals(4, 2 + 2) |
||||
} |
||||
} |
@ -0,0 +1,26 @@
|
||||
// Top-level build file where you can add configuration options common to all sub-projects/modules. |
||||
buildscript { |
||||
ext.kotlin_version = "1.4.10" |
||||
repositories { |
||||
google() |
||||
jcenter() |
||||
} |
||||
dependencies { |
||||
classpath 'com.android.tools.build:gradle:4.0.2' |
||||
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" |
||||
|
||||
// NOTE: Do not place your application dependencies here; they belong |
||||
// in the individual module build.gradle files |
||||
} |
||||
} |
||||
|
||||
allprojects { |
||||
repositories { |
||||
google() |
||||
jcenter() |
||||
} |
||||
} |
||||
|
||||
task clean(type: Delete) { |
||||
delete rootProject.buildDir |
||||
} |
@ -0,0 +1,21 @@
|
||||
# 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 |
||||
# 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 |
||||
# Kotlin code style for this project: "official" or "obsolete": |
||||
kotlin.code.style=official |
@ -0,0 +1,6 @@
|
||||
#Thu Oct 08 16:06:13 CEST 2020 |
||||
distributionBase=GRADLE_USER_HOME |
||||
distributionPath=wrapper/dists |
||||
zipStoreBase=GRADLE_USER_HOME |
||||
zipStorePath=wrapper/dists |
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-6.1.1-all.zip |
@ -0,0 +1,172 @@
|
||||
#!/usr/bin/env sh |
||||
|
||||
############################################################################## |
||||
## |
||||
## 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="" |
||||
|
||||
# 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, switch paths to Windows format before running java |
||||
if $cygwin ; 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=$((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" |
||||
|
||||
# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong |
||||
if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then |
||||
cd "$(dirname "$0")" |
||||