Skip to content
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
19 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
70 changes: 70 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
# Android Studio
<<<<<<< HEAD
.idea/
*.iml
.gradle/
build/
captures/
local.properties
.externalNativeBuild
.cxx

# Kotlin and Java
*.class
*.kotlin_module
*.log
*.ctxt
*.jar
*.war
*.nar
*.ear
*.zip
*.tar.gz
*.rar

# Virtual Machine
.DS_Store
.mtj.tmp/
hs_err_pid*

# Dependency directories
/node_modules/
kotlin-js-store/

# Other
.env
__pycache__/
/out/
=======
*.iml
.gradle/
build/
.idea/
local.properties

# Gradle files
*.gradle
**/build/

# Compiled class files
*.class

# Log files
*.log

# Virtual machine crash logs
hs_err_pid*

# Mac system files
.DS_Store

# Sensitive or local configuration
*.env

# External libraries
app/libs/

# Test results
test-results/
reports/
>>>>>>> pr-4-Ralfmal-kotlinTodoApp
26 changes: 26 additions & 0 deletions app/build.gradle
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
plugins {
id 'com.android.application'
id 'kotlin-android'
id 'kotlin-kapt'
}

dependencies {
// Kotlin standard library
implementation "org.jetbrains.kotlin:kotlin-stdlib:1.8.0"

// AndroidX
implementation 'androidx.core:core-ktx:1.10.1'
implementation 'androidx.appcompat:appcompat:1.6.1'
implementation 'androidx.lifecycle:lifecycle-runtime-ktx:2.6.1'

// Room Database
implementation 'androidx.room:room-runtime:2.5.2'
implementation 'androidx.room:room-ktx:2.5.2'
kapt 'androidx.room:room-compiler:2.5.2'

// Test dependencies
testImplementation 'junit:junit:4.13.2'
testImplementation 'org.robolectric:robolectric:4.10.3'
testImplementation 'androidx.test:core:1.5.0'
testImplementation 'androidx.test.ext:junit:1.1.5'
}
18 changes: 18 additions & 0 deletions app/src/main/java/com/example/todoapp/data/TodoItem.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
package com.example.todoapp.data

data class TodoItem(
val id: Long = 0, // Unique identifier, defaults to 0 for new items
val title: String,
val description: String = "", // Optional description with default empty string
val isCompleted: Boolean = false, // Default state is not completed
val createdAt: Long = System.currentTimeMillis() // Timestamp of creation
) {
/**
* Validates the todo item's core attributes
*
* @return Boolean indicating if the todo item is valid
*/
fun validate(): Boolean {
return title.trim().isNotBlank() && title.length <= 100 // Added length constraint
}
}
71 changes: 71 additions & 0 deletions app/src/main/java/com/example/todoapp/ui/AddTodoActivity.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
package com.example.todoapp.ui

import android.os.Bundle
import android.widget.Button
import android.widget.Toast
import androidx.appcompat.app.AppCompatActivity
import androidx.lifecycle.lifecycleScope
import com.google.android.material.textfield.TextInputEditText
import com.example.todoapp.R
import com.example.todoapp.data.TodoItem
import com.example.todoapp.data.TodoRepository
import kotlinx.coroutines.launch

class AddTodoActivity : AppCompatActivity() {
private lateinit var titleEditText: TextInputEditText
private lateinit var descriptionEditText: TextInputEditText
private lateinit var saveButton: Button
private lateinit var todoRepository: TodoRepository

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_add_todo)

// Initialize views
titleEditText = findViewById(R.id.titleEditText)
descriptionEditText = findViewById(R.id.descriptionEditText)
saveButton = findViewById(R.id.saveButton)

// Initialize repository
todoRepository = TodoRepository(application)

// Set up save button click listener
saveButton.setOnClickListener {
saveTodoItem()
}
}

private fun saveTodoItem() {
val title = titleEditText.text.toString().trim()
val description = descriptionEditText.text.toString().trim()

// Enhanced validation
if (title.isEmpty()) {
titleEditText.error = "Title cannot be empty"
return
}

if (title.length > 100) {
titleEditText.error = "Title must be 100 characters or less"
return
}

// Create TodoItem
val todoItem = TodoItem(
title = title,
description = description,
isCompleted = false
)

// Save asynchronously
lifecycleScope.launch {
try {
todoRepository.insert(todoItem)
Toast.makeText(this@AddTodoActivity, "Todo saved successfully!", Toast.LENGTH_SHORT).show()
finish() // Close the activity after saving
} catch (e: Exception) {
Toast.makeText(this@AddTodoActivity, "Failed to save todo: ${e.localizedMessage}", Toast.LENGTH_SHORT).show()
}
}
}
}
71 changes: 71 additions & 0 deletions app/src/main/java/com/example/todoapp/ui/CreateTodoItemActivity.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
package com.example.todoapp.ui

import android.content.Intent
import android.os.Bundle
import android.widget.Button
import android.widget.EditText
import androidx.appcompat.app.AppCompatActivity
import androidx.lifecycle.lifecycleScope
import com.example.todoapp.R
import com.example.todoapp.data.TodoItem
import com.example.todoapp.data.TodoRepository
import kotlinx.coroutines.launch

class CreateTodoItemActivity : AppCompatActivity() {

private lateinit var titleEditText: EditText
private lateinit var descriptionEditText: EditText
private lateinit var saveButton: Button
private lateinit var todoRepository: TodoRepository

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_create_todo_item)

// Initialize views
titleEditText = findViewById(R.id.edit_text_todo_title)
descriptionEditText = findViewById(R.id.edit_text_todo_description)
saveButton = findViewById(R.id.button_save_todo)

// Initialize repository (assumes dependency injection or manual initialization)
todoRepository = TodoRepository(application)

saveButton.setOnClickListener {
saveTodoItem()
}
}

private fun saveTodoItem() {
val title = titleEditText.text.toString().trim()
val description = descriptionEditText.text.toString().trim()

// Basic validation
if (title.isEmpty()) {
titleEditText.error = "Title cannot be empty"
return
}

// Create todo item
val todoItem = TodoItem(
title = title,
description = description,
isCompleted = false
)

// Save todo item using coroutines
lifecycleScope.launch {
todoRepository.insert(todoItem)
// Navigate back to main activity
navigateToMainActivity()
}
}

private fun navigateToMainActivity() {
val intent = Intent(this, MainActivity::class.java)
// Clear the back stack so user can't navigate back to create todo screen
intent.flags = Intent.FLAG_ACTIVITY_CLEAR_TOP or Intent.FLAG_ACTIVITY_NEW_TASK
startActivity(intent)
// Finish current activity
finish()
}
}
42 changes: 42 additions & 0 deletions app/src/main/res/layout/activity_add_todo.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:padding="16dp">

<com.google.android.material.textfield.TextInputLayout
android:id="@+id/titleInputLayout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="Todo Title">

<com.google.android.material.textfield.TextInputEditText
android:id="@+id/titleEditText"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:inputType="text" />
</com.google.android.material.textfield.TextInputLayout>

<com.google.android.material.textfield.TextInputLayout
android:id="@+id/descriptionInputLayout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="Todo Description"
android:layout_marginTop="16dp">

<com.google.android.material.textfield.TextInputEditText
android:id="@+id/descriptionEditText"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:inputType="textMultiLine"
android:minLines="3" />
</com.google.android.material.textfield.TextInputLayout>

<Button
android:id="@+id/saveButton"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="16dp"
android:text="Save Todo" />
</LinearLayout>
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
package com.example.todoapp

import android.content.Intent
import androidx.test.core.app.ActivityScenario
import androidx.test.core.app.ApplicationProvider
import androidx.test.ext.junit.runners.AndroidJUnit4
import com.example.todoapp.ui.CreateTodoItemActivity
import com.example.todoapp.ui.MainActivity
import org.junit.Assert.assertEquals
import org.junit.Test
import org.junit.runner.RunWith
import org.robolectric.Robolectric
import org.robolectric.Shadows.shadowOf

@RunWith(AndroidJUnit4::class)
class CreateTodoItemActivityTest {

@Test
fun `test navigation to main activity after saving todo item`() {
// Create an activity scenario
val scenario = ActivityScenario.launch(CreateTodoItemActivity::class.java)

scenario.onActivity { activity ->
// Mock saving a todo item by directly calling private method
val method = activity.javaClass.getDeclaredMethod("navigateToMainActivity")
method.isAccessible = true
method.invoke(activity)

// Get the next started intent
val shadowActivity = Robolectric.shadowOf(activity)
val startedIntent = shadowActivity.nextStartedActivity

// Verify intent navigation
assertEquals(MainActivity::class.java.name,
startedIntent.component?.className)

// Verify intent flags
assertEquals(
Intent.FLAG_ACTIVITY_CLEAR_TOP or Intent.FLAG_ACTIVITY_NEW_TASK,
startedIntent.flags
)
}
}
}
Loading