wip
This commit is contained in:
@@ -125,6 +125,95 @@ class ResidenceApi(private val client: HttpClient = ApiClient.httpClient) {
|
|||||||
ApiResult.Error(e.message ?: "Unknown error occurred")
|
ApiResult.Error(e.message ?: "Unknown error occurred")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Share Code Management
|
||||||
|
suspend fun generateShareCode(token: String, residenceId: Int): ApiResult<ResidenceShareCode> {
|
||||||
|
return try {
|
||||||
|
val response = client.post("$baseUrl/residences/$residenceId/generate-share-code/") {
|
||||||
|
header("Authorization", "Token $token")
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.status.isSuccess()) {
|
||||||
|
ApiResult.Success(response.body())
|
||||||
|
} else {
|
||||||
|
val errorMessage = ErrorParser.parseError(response)
|
||||||
|
ApiResult.Error(errorMessage, response.status.value)
|
||||||
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
ApiResult.Error(e.message ?: "Unknown error occurred")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun getShareCode(token: String, residenceId: Int): ApiResult<ResidenceShareCode> {
|
||||||
|
return try {
|
||||||
|
val response = client.get("$baseUrl/residences/$residenceId/share-code/") {
|
||||||
|
header("Authorization", "Token $token")
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.status.isSuccess()) {
|
||||||
|
ApiResult.Success(response.body())
|
||||||
|
} else {
|
||||||
|
val errorMessage = ErrorParser.parseError(response)
|
||||||
|
ApiResult.Error(errorMessage, response.status.value)
|
||||||
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
ApiResult.Error(e.message ?: "Unknown error occurred")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun joinWithCode(token: String, code: String): ApiResult<JoinResidenceResponse> {
|
||||||
|
return try {
|
||||||
|
val response = client.post("$baseUrl/residences/join-with-code/") {
|
||||||
|
header("Authorization", "Token $token")
|
||||||
|
contentType(ContentType.Application.Json)
|
||||||
|
setBody(JoinResidenceRequest(code))
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.status.isSuccess()) {
|
||||||
|
ApiResult.Success(response.body())
|
||||||
|
} else {
|
||||||
|
val errorMessage = ErrorParser.parseError(response)
|
||||||
|
ApiResult.Error(errorMessage, response.status.value)
|
||||||
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
ApiResult.Error(e.message ?: "Unknown error occurred")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// User Management
|
||||||
|
suspend fun getResidenceUsers(token: String, residenceId: Int): ApiResult<ResidenceUsersResponse> {
|
||||||
|
return try {
|
||||||
|
val response = client.get("$baseUrl/residences/$residenceId/users/") {
|
||||||
|
header("Authorization", "Token $token")
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.status.isSuccess()) {
|
||||||
|
ApiResult.Success(response.body())
|
||||||
|
} else {
|
||||||
|
val errorMessage = ErrorParser.parseError(response)
|
||||||
|
ApiResult.Error(errorMessage, response.status.value)
|
||||||
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
ApiResult.Error(e.message ?: "Unknown error occurred")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun removeUser(token: String, residenceId: Int, userId: Int): ApiResult<RemoveUserResponse> {
|
||||||
|
return try {
|
||||||
|
val response = client.delete("$baseUrl/residences/$residenceId/users/$userId/") {
|
||||||
|
header("Authorization", "Token $token")
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.status.isSuccess()) {
|
||||||
|
ApiResult.Success(response.body())
|
||||||
|
} else {
|
||||||
|
val errorMessage = ErrorParser.parseError(response)
|
||||||
|
ApiResult.Error(errorMessage, response.status.value)
|
||||||
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
ApiResult.Error(e.message ?: "Unknown error occurred")
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@kotlinx.serialization.Serializable
|
@kotlinx.serialization.Serializable
|
||||||
|
|||||||
@@ -0,0 +1,134 @@
|
|||||||
|
package com.mycrib.android.ui.components
|
||||||
|
|
||||||
|
import androidx.compose.foundation.layout.*
|
||||||
|
import androidx.compose.material.icons.Icons
|
||||||
|
import androidx.compose.material.icons.filled.Close
|
||||||
|
import androidx.compose.material3.*
|
||||||
|
import androidx.compose.runtime.*
|
||||||
|
import androidx.compose.ui.Alignment
|
||||||
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.text.input.TextFieldValue
|
||||||
|
import androidx.compose.ui.unit.dp
|
||||||
|
import com.mycrib.shared.network.ApiResult
|
||||||
|
import com.mycrib.shared.network.ResidenceApi
|
||||||
|
import com.mycrib.storage.TokenStorage
|
||||||
|
import kotlinx.coroutines.launch
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun JoinResidenceDialog(
|
||||||
|
onDismiss: () -> Unit,
|
||||||
|
onJoined: () -> Unit = {}
|
||||||
|
) {
|
||||||
|
var shareCode by remember { mutableStateOf(TextFieldValue("")) }
|
||||||
|
var isJoining by remember { mutableStateOf(false) }
|
||||||
|
var error by remember { mutableStateOf<String?>(null) }
|
||||||
|
|
||||||
|
val residenceApi = remember { ResidenceApi() }
|
||||||
|
val scope = rememberCoroutineScope()
|
||||||
|
|
||||||
|
AlertDialog(
|
||||||
|
onDismissRequest = onDismiss,
|
||||||
|
title = {
|
||||||
|
Row(
|
||||||
|
modifier = Modifier.fillMaxWidth(),
|
||||||
|
horizontalArrangement = Arrangement.SpaceBetween,
|
||||||
|
verticalAlignment = Alignment.CenterVertically
|
||||||
|
) {
|
||||||
|
Text("Join Residence")
|
||||||
|
IconButton(onClick = onDismiss) {
|
||||||
|
Icon(Icons.Default.Close, "Close")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
text = {
|
||||||
|
Column(modifier = Modifier.fillMaxWidth()) {
|
||||||
|
Text(
|
||||||
|
text = "Enter the 6-character share code to join a residence",
|
||||||
|
style = MaterialTheme.typography.bodyMedium,
|
||||||
|
color = MaterialTheme.colorScheme.onSurfaceVariant,
|
||||||
|
modifier = Modifier.padding(bottom = 16.dp)
|
||||||
|
)
|
||||||
|
|
||||||
|
OutlinedTextField(
|
||||||
|
value = shareCode,
|
||||||
|
onValueChange = {
|
||||||
|
if (it.text.length <= 6) {
|
||||||
|
shareCode = it.copy(text = it.text.uppercase())
|
||||||
|
error = null
|
||||||
|
}
|
||||||
|
},
|
||||||
|
label = { Text("Share Code") },
|
||||||
|
placeholder = { Text("ABC123") },
|
||||||
|
singleLine = true,
|
||||||
|
enabled = !isJoining,
|
||||||
|
isError = error != null,
|
||||||
|
supportingText = {
|
||||||
|
if (error != null) {
|
||||||
|
Text(
|
||||||
|
text = error ?: "",
|
||||||
|
color = MaterialTheme.colorScheme.error
|
||||||
|
)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
modifier = Modifier.fillMaxWidth()
|
||||||
|
)
|
||||||
|
|
||||||
|
if (isJoining) {
|
||||||
|
Box(
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxWidth()
|
||||||
|
.padding(top = 16.dp),
|
||||||
|
contentAlignment = Alignment.Center
|
||||||
|
) {
|
||||||
|
CircularProgressIndicator()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
confirmButton = {
|
||||||
|
Button(
|
||||||
|
onClick = {
|
||||||
|
if (shareCode.text.length == 6) {
|
||||||
|
scope.launch {
|
||||||
|
isJoining = true
|
||||||
|
error = null
|
||||||
|
val token = TokenStorage.getToken()
|
||||||
|
if (token != null) {
|
||||||
|
when (val result = residenceApi.joinWithCode(token, shareCode.text)) {
|
||||||
|
is ApiResult.Success -> {
|
||||||
|
isJoining = false
|
||||||
|
onJoined()
|
||||||
|
onDismiss()
|
||||||
|
}
|
||||||
|
is ApiResult.Error -> {
|
||||||
|
error = result.message
|
||||||
|
isJoining = false
|
||||||
|
}
|
||||||
|
else -> {
|
||||||
|
isJoining = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
error = "Not authenticated"
|
||||||
|
isJoining = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
error = "Share code must be 6 characters"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
enabled = !isJoining && shareCode.text.length == 6
|
||||||
|
) {
|
||||||
|
Text("Join")
|
||||||
|
}
|
||||||
|
},
|
||||||
|
dismissButton = {
|
||||||
|
TextButton(
|
||||||
|
onClick = onDismiss,
|
||||||
|
enabled = !isJoining
|
||||||
|
) {
|
||||||
|
Text("Cancel")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -14,6 +14,7 @@ import androidx.compose.ui.Modifier
|
|||||||
import androidx.compose.ui.text.font.FontWeight
|
import androidx.compose.ui.text.font.FontWeight
|
||||||
import androidx.compose.ui.unit.dp
|
import androidx.compose.ui.unit.dp
|
||||||
import androidx.lifecycle.viewmodel.compose.viewModel
|
import androidx.lifecycle.viewmodel.compose.viewModel
|
||||||
|
import com.mycrib.android.ui.components.JoinResidenceDialog
|
||||||
import com.mycrib.android.ui.components.common.StatItem
|
import com.mycrib.android.ui.components.common.StatItem
|
||||||
import com.mycrib.android.ui.components.residence.TaskStatChip
|
import com.mycrib.android.ui.components.residence.TaskStatChip
|
||||||
import com.mycrib.android.viewmodel.ResidenceViewModel
|
import com.mycrib.android.viewmodel.ResidenceViewModel
|
||||||
@@ -29,11 +30,24 @@ fun ResidencesScreen(
|
|||||||
viewModel: ResidenceViewModel = viewModel { ResidenceViewModel() }
|
viewModel: ResidenceViewModel = viewModel { ResidenceViewModel() }
|
||||||
) {
|
) {
|
||||||
val myResidencesState by viewModel.myResidencesState.collectAsState()
|
val myResidencesState by viewModel.myResidencesState.collectAsState()
|
||||||
|
var showJoinDialog by remember { mutableStateOf(false) }
|
||||||
|
|
||||||
LaunchedEffect(Unit) {
|
LaunchedEffect(Unit) {
|
||||||
viewModel.loadMyResidences()
|
viewModel.loadMyResidences()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (showJoinDialog) {
|
||||||
|
JoinResidenceDialog(
|
||||||
|
onDismiss = {
|
||||||
|
showJoinDialog = false
|
||||||
|
},
|
||||||
|
onJoined = {
|
||||||
|
// Reload residences after joining
|
||||||
|
viewModel.loadMyResidences()
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
Scaffold(
|
Scaffold(
|
||||||
topBar = {
|
topBar = {
|
||||||
TopAppBar(
|
TopAppBar(
|
||||||
@@ -44,6 +58,9 @@ fun ResidencesScreen(
|
|||||||
)
|
)
|
||||||
},
|
},
|
||||||
actions = {
|
actions = {
|
||||||
|
IconButton(onClick = { showJoinDialog = true }) {
|
||||||
|
Icon(Icons.Default.GroupAdd, contentDescription = "Join with Code")
|
||||||
|
}
|
||||||
IconButton(onClick = onNavigateToProfile) {
|
IconButton(onClick = onNavigateToProfile) {
|
||||||
Icon(Icons.Default.AccountCircle, contentDescription = "Profile")
|
Icon(Icons.Default.AccountCircle, contentDescription = "Profile")
|
||||||
}
|
}
|
||||||
@@ -174,6 +191,26 @@ fun ResidencesScreen(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Spacer(modifier = Modifier.height(8.dp))
|
||||||
|
OutlinedButton(
|
||||||
|
onClick = { showJoinDialog = true },
|
||||||
|
modifier = Modifier
|
||||||
|
.fillMaxWidth(0.7f)
|
||||||
|
.height(56.dp),
|
||||||
|
shape = RoundedCornerShape(12.dp)
|
||||||
|
) {
|
||||||
|
Row(
|
||||||
|
horizontalArrangement = Arrangement.spacedBy(8.dp),
|
||||||
|
verticalAlignment = Alignment.CenterVertically
|
||||||
|
) {
|
||||||
|
Icon(Icons.Default.GroupAdd, contentDescription = null)
|
||||||
|
Text(
|
||||||
|
"Join with Code",
|
||||||
|
style = MaterialTheme.typography.titleMedium,
|
||||||
|
fontWeight = FontWeight.SemiBold
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
106
iosApp/iosApp/Residence/JoinResidenceView.swift
Normal file
106
iosApp/iosApp/Residence/JoinResidenceView.swift
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
import SwiftUI
|
||||||
|
import ComposeApp
|
||||||
|
|
||||||
|
struct JoinResidenceView: View {
|
||||||
|
@Environment(\.dismiss) private var dismiss
|
||||||
|
let onJoined: () -> Void
|
||||||
|
|
||||||
|
@State private var shareCode: String = ""
|
||||||
|
@State private var isJoining = false
|
||||||
|
@State private var errorMessage: String?
|
||||||
|
|
||||||
|
private let residenceApi = ResidenceApi(client: ApiClient_iosKt.createHttpClient())
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
NavigationView {
|
||||||
|
Form {
|
||||||
|
Section {
|
||||||
|
TextField("Share Code", text: $shareCode)
|
||||||
|
.textInputAutocapitalization(.characters)
|
||||||
|
.autocorrectionDisabled()
|
||||||
|
.onChange(of: shareCode) { newValue in
|
||||||
|
// Limit to 6 characters and uppercase
|
||||||
|
if newValue.count > 6 {
|
||||||
|
shareCode = String(newValue.prefix(6))
|
||||||
|
}
|
||||||
|
shareCode = shareCode.uppercased()
|
||||||
|
errorMessage = nil
|
||||||
|
}
|
||||||
|
.disabled(isJoining)
|
||||||
|
} header: {
|
||||||
|
Text("Enter Share Code")
|
||||||
|
} footer: {
|
||||||
|
Text("Enter the 6-character code shared with you to join a residence")
|
||||||
|
.foregroundColor(.secondary)
|
||||||
|
}
|
||||||
|
|
||||||
|
if let error = errorMessage {
|
||||||
|
Section {
|
||||||
|
Text(error)
|
||||||
|
.foregroundColor(.red)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Section {
|
||||||
|
Button(action: joinResidence) {
|
||||||
|
HStack {
|
||||||
|
Spacer()
|
||||||
|
if isJoining {
|
||||||
|
ProgressView()
|
||||||
|
.progressViewStyle(CircularProgressViewStyle())
|
||||||
|
} else {
|
||||||
|
Text("Join Residence")
|
||||||
|
.fontWeight(.semibold)
|
||||||
|
}
|
||||||
|
Spacer()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.disabled(shareCode.count != 6 || isJoining)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.navigationTitle("Join Residence")
|
||||||
|
.navigationBarTitleDisplayMode(.inline)
|
||||||
|
.toolbar {
|
||||||
|
ToolbarItem(placement: .navigationBarLeading) {
|
||||||
|
Button("Cancel") {
|
||||||
|
dismiss()
|
||||||
|
}
|
||||||
|
.disabled(isJoining)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func joinResidence() {
|
||||||
|
guard shareCode.count == 6 else {
|
||||||
|
errorMessage = "Share code must be 6 characters"
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
guard let token = TokenStorage.shared.getToken() else {
|
||||||
|
errorMessage = "Not authenticated"
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
isJoining = true
|
||||||
|
errorMessage = nil
|
||||||
|
|
||||||
|
residenceApi.joinWithCode(token: token, code: shareCode) { result, error in
|
||||||
|
if result is ApiResultSuccess<JoinResidenceResponse> {
|
||||||
|
self.isJoining = false
|
||||||
|
self.onJoined()
|
||||||
|
self.dismiss()
|
||||||
|
} else if let errorResult = result as? ApiResultError {
|
||||||
|
self.errorMessage = errorResult.message
|
||||||
|
self.isJoining = false
|
||||||
|
} else if let error = error {
|
||||||
|
self.errorMessage = error.localizedDescription
|
||||||
|
self.isJoining = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#Preview {
|
||||||
|
JoinResidenceView(onJoined: {})
|
||||||
|
}
|
||||||
@@ -4,6 +4,7 @@ import ComposeApp
|
|||||||
struct ResidencesListView: View {
|
struct ResidencesListView: View {
|
||||||
@StateObject private var viewModel = ResidenceViewModel()
|
@StateObject private var viewModel = ResidenceViewModel()
|
||||||
@State private var showingAddResidence = false
|
@State private var showingAddResidence = false
|
||||||
|
@State private var showingJoinResidence = false
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
ZStack {
|
ZStack {
|
||||||
@@ -54,7 +55,13 @@ struct ResidencesListView: View {
|
|||||||
.navigationTitle("My Properties")
|
.navigationTitle("My Properties")
|
||||||
.navigationBarTitleDisplayMode(.large)
|
.navigationBarTitleDisplayMode(.large)
|
||||||
.toolbar {
|
.toolbar {
|
||||||
ToolbarItem(placement: .navigationBarTrailing) {
|
ToolbarItemGroup(placement: .navigationBarTrailing) {
|
||||||
|
Button(action: {
|
||||||
|
showingJoinResidence = true
|
||||||
|
}) {
|
||||||
|
Image(systemName: "person.badge.plus")
|
||||||
|
}
|
||||||
|
|
||||||
Button(action: {
|
Button(action: {
|
||||||
showingAddResidence = true
|
showingAddResidence = true
|
||||||
}) {
|
}) {
|
||||||
@@ -65,6 +72,11 @@ struct ResidencesListView: View {
|
|||||||
.sheet(isPresented: $showingAddResidence) {
|
.sheet(isPresented: $showingAddResidence) {
|
||||||
AddResidenceView(isPresented: $showingAddResidence)
|
AddResidenceView(isPresented: $showingAddResidence)
|
||||||
}
|
}
|
||||||
|
.sheet(isPresented: $showingJoinResidence) {
|
||||||
|
JoinResidenceView(onJoined: {
|
||||||
|
viewModel.loadMyResidences()
|
||||||
|
})
|
||||||
|
}
|
||||||
.onAppear {
|
.onAppear {
|
||||||
viewModel.loadMyResidences()
|
viewModel.loadMyResidences()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -58,6 +58,8 @@ struct ResidenceCard: View {
|
|||||||
id: 1,
|
id: 1,
|
||||||
owner: 1,
|
owner: 1,
|
||||||
ownerUsername: "testuser",
|
ownerUsername: "testuser",
|
||||||
|
isPrimaryOwner: false,
|
||||||
|
userCount: 1,
|
||||||
name: "My Home",
|
name: "My Home",
|
||||||
propertyType: "House",
|
propertyType: "House",
|
||||||
streetAddress: "123 Main St",
|
streetAddress: "123 Main St",
|
||||||
|
|||||||
Reference in New Issue
Block a user