Compare commits
1 Commits
home_page
...
update_age
| Author | SHA1 | Date | |
|---|---|---|---|
| d8df67bae5 |
@@ -81,7 +81,9 @@ data class CreateGroupChatRequestBody(
|
||||
|
||||
data class JoinGroupChatRequestBody(
|
||||
@SerializedName("trtcId")
|
||||
val trtcId: String,
|
||||
val trtcId: String? = null,
|
||||
@SerializedName("roomId")
|
||||
val roomId: Int? = null,
|
||||
)
|
||||
|
||||
data class LoginUserRequestBody(
|
||||
@@ -271,6 +273,132 @@ data class RemoveAccountRequestBody(
|
||||
val password: String,
|
||||
)
|
||||
|
||||
// API 错误响应(用于加入房间等接口的错误处理)
|
||||
data class ApiErrorResponse(
|
||||
@SerializedName("err")
|
||||
val error: String,
|
||||
@SerializedName("success")
|
||||
val success: Boolean
|
||||
)
|
||||
|
||||
// 群聊中的用户信息
|
||||
data class GroupChatUser(
|
||||
@SerializedName("ID")
|
||||
val id: Int,
|
||||
@SerializedName("CreatedAt")
|
||||
val createdAt: String,
|
||||
@SerializedName("UpdatedAt")
|
||||
val updatedAt: String,
|
||||
@SerializedName("DeletedAt")
|
||||
val deletedAt: String?,
|
||||
@SerializedName("userSessionId")
|
||||
val userSessionId: String,
|
||||
@SerializedName("sessions")
|
||||
val sessions: Any?, // 根据实际需要可以定义具体类型
|
||||
@SerializedName("prompts")
|
||||
val prompts: Any?, // 根据实际需要可以定义具体类型
|
||||
@SerializedName("isAgent")
|
||||
val isAgent: Boolean
|
||||
)
|
||||
|
||||
// 智能体角色信息
|
||||
data class GroupChatPrompt(
|
||||
@SerializedName("ID")
|
||||
val id: Int,
|
||||
@SerializedName("CreatedAt")
|
||||
val createdAt: String,
|
||||
@SerializedName("UpdatedAt")
|
||||
val updatedAt: String,
|
||||
@SerializedName("DeletedAt")
|
||||
val deletedAt: String?,
|
||||
@SerializedName("Title")
|
||||
val title: String,
|
||||
@SerializedName("Desc")
|
||||
val desc: String,
|
||||
@SerializedName("Value")
|
||||
val value: String,
|
||||
@SerializedName("Enable")
|
||||
val enable: Boolean,
|
||||
@SerializedName("UserSessions")
|
||||
val userSessions: Any?, // 根据实际需要可以定义具体类型
|
||||
@SerializedName("Avatar")
|
||||
val avatar: String,
|
||||
@SerializedName("AuthorId")
|
||||
val authorId: Int?,
|
||||
@SerializedName("Author")
|
||||
val author: Any?, // 根据实际需要可以定义具体类型
|
||||
@SerializedName("TokenCount")
|
||||
val tokenCount: Int,
|
||||
@SerializedName("OpenId")
|
||||
val openId: String,
|
||||
@SerializedName("Public")
|
||||
val public: Boolean,
|
||||
@SerializedName("BreakMode")
|
||||
val breakMode: Boolean,
|
||||
@SerializedName("DocNamespace")
|
||||
val docNamespace: String,
|
||||
@SerializedName("UseRag")
|
||||
val useRag: Boolean,
|
||||
@SerializedName("RagThreshold")
|
||||
val ragThreshold: Double,
|
||||
@SerializedName("WorkflowId")
|
||||
val workflowId: Int?,
|
||||
@SerializedName("Workflow")
|
||||
val workflow: Any?, // 根据实际需要可以定义具体类型
|
||||
@SerializedName("WorkflowInputs")
|
||||
val workflowInputs: Any?, // 根据实际需要可以定义具体类型
|
||||
@SerializedName("Source")
|
||||
val source: String,
|
||||
@SerializedName("categories")
|
||||
val categories: Any? // 根据实际需要可以定义具体类型
|
||||
)
|
||||
|
||||
// 群聊详细信息响应
|
||||
data class GroupChatResponse(
|
||||
@SerializedName("ID")
|
||||
val id: Int,
|
||||
@SerializedName("CreatedAt")
|
||||
val createdAt: String,
|
||||
@SerializedName("UpdatedAt")
|
||||
val updatedAt: String,
|
||||
@SerializedName("DeletedAt")
|
||||
val deletedAt: String?,
|
||||
@SerializedName("name")
|
||||
val name: String,
|
||||
@SerializedName("description")
|
||||
val description: String,
|
||||
@SerializedName("creatorId")
|
||||
val creatorId: Int,
|
||||
@SerializedName("creator")
|
||||
val creator: Any?, // 根据实际需要可以定义具体类型
|
||||
@SerializedName("trtcRoomId")
|
||||
val trtcRoomId: String,
|
||||
@SerializedName("trtcType")
|
||||
val trtcType: String,
|
||||
@SerializedName("cover")
|
||||
val cover: String,
|
||||
@SerializedName("avatar")
|
||||
val avatar: String,
|
||||
@SerializedName("recommendBanner")
|
||||
val recommendBanner: String,
|
||||
@SerializedName("isRecommended")
|
||||
val isRecommended: Boolean,
|
||||
@SerializedName("allowInHot")
|
||||
val allowInHot: Boolean,
|
||||
@SerializedName("users")
|
||||
val users: List<GroupChatUser>,
|
||||
@SerializedName("prompts")
|
||||
val prompts: List<GroupChatPrompt>,
|
||||
@SerializedName("source")
|
||||
val source: String
|
||||
)
|
||||
class CategoryTemplateTranslation(
|
||||
@SerializedName("name")
|
||||
val name: String,
|
||||
@SerializedName("description")
|
||||
val description: String,
|
||||
)
|
||||
|
||||
data class CategoryTemplate(
|
||||
@SerializedName("id")
|
||||
val id: Int,
|
||||
@@ -295,19 +423,19 @@ data class CategoryTemplate(
|
||||
@SerializedName("createdAt")
|
||||
val createdAt: String,
|
||||
@SerializedName("updatedAt")
|
||||
val updatedAt: String
|
||||
)
|
||||
|
||||
data class CategoryListResponse(
|
||||
@SerializedName("page")
|
||||
val page: Int,
|
||||
@SerializedName("pageSize")
|
||||
val pageSize: Int,
|
||||
@SerializedName("total")
|
||||
val total: Int,
|
||||
@SerializedName("list")
|
||||
val list: List<CategoryTemplate>
|
||||
)
|
||||
val updatedAt: String,
|
||||
@SerializedName("translations")
|
||||
val translations: Map<String, CategoryTemplateTranslation>?
|
||||
) {
|
||||
/**
|
||||
* 获取本地化名称,优先使用当前语言的翻译,如果没有则使用默认名称
|
||||
*/
|
||||
fun getLocalizedName(): String {
|
||||
// 这里可以根据需要添加国际化逻辑
|
||||
// 目前直接返回默认名称
|
||||
return name
|
||||
}
|
||||
}
|
||||
|
||||
interface RaveNowAPI {
|
||||
@GET("membership/config")
|
||||
@@ -588,9 +716,26 @@ interface RaveNowAPI {
|
||||
suspend fun getAgent(
|
||||
@Query("page") page: Int = 1,
|
||||
@Query("pageSize") pageSize: Int = 20,
|
||||
@Query("withWorkflow") withWorkflow: Int = 1,
|
||||
@Query("order") order: String? = null,
|
||||
@Query("orderKey") orderKey: String? = null,
|
||||
@Query("createdAt") createdAt: String? = null,
|
||||
@Query("updatedAt") updatedAt: String? = null,
|
||||
@Query("createdStart") createdStart: String? = null,
|
||||
@Query("createdEnd") createdEnd: String? = null,
|
||||
@Query("updatedStart") updatedStart: String? = null,
|
||||
@Query("updatedEnd") updatedEnd: String? = null,
|
||||
@Query("title") title: String? = null,
|
||||
@Query("authorId") authorId: Int? = null,
|
||||
@Query("authorOpenId") authorOpenId: String? = null,
|
||||
@Query("showPrivate") showPrivate: String? = null,
|
||||
@Query("explore") explore: String? = null,
|
||||
@Query("desc") desc: String? = null,
|
||||
@Query("withWorkflow") withWorkflow: String? = null,
|
||||
@Query("hasAvatar") hasAvatar: String? = null,
|
||||
@Query("random") random: String? = null,
|
||||
@Query("categoryName") categoryName: String? = null,
|
||||
@Query("categoryIds") categoryIds: List<Int>? = null,
|
||||
@Query("uncategorized") uncategorized: String? = null,
|
||||
): Response<DataContainer<ListContainer<Agent>>>
|
||||
|
||||
@GET("outside/my/prompts")
|
||||
@@ -619,7 +764,10 @@ interface RaveNowAPI {
|
||||
suspend fun agentMoment(@Body body: AgentMomentRequestBody): Response<DataContainer<String>>
|
||||
|
||||
@GET("outside/rooms/open")
|
||||
suspend fun createGroupChatAi(@Query("trtcGroupId") trtcGroupId: String): Response<DataContainer<Unit>>
|
||||
suspend fun createGroupChatAi(
|
||||
@Query("trtcGroupId") trtcGroupId: String? = null,
|
||||
@Query("roomId") roomId: Int? = null
|
||||
): Response<DataContainer<GroupChatResponse>>
|
||||
|
||||
@POST("outside/rooms/create-single-chat")
|
||||
suspend fun createSingleChat(@Body body: SingleChatRequestBody): Response<DataContainer<Unit>>
|
||||
@@ -634,6 +782,7 @@ interface RaveNowAPI {
|
||||
suspend fun getRooms(@Query("page") page: Int = 1,
|
||||
@Query("pageSize") pageSize: Int = 20,
|
||||
@Query("isRecommended") isRecommended: Int = 1,
|
||||
@Query("random") random: Int? = null,
|
||||
): Response<ListContainer<Room>>
|
||||
|
||||
@GET("outside/rooms/detail")
|
||||
@@ -655,7 +804,7 @@ interface RaveNowAPI {
|
||||
@Query("withParent") withParent: Boolean? = null,
|
||||
@Query("withCount") withCount: Boolean? = null,
|
||||
@Query("hideEmpty") hideEmpty: Boolean? = null
|
||||
): Response<DataContainer<CategoryListResponse>>
|
||||
): Response<ListContainer<CategoryTemplate>>
|
||||
|
||||
@GET("outside/categories/tree")
|
||||
suspend fun getCategoryTree(
|
||||
@@ -668,14 +817,6 @@ interface RaveNowAPI {
|
||||
@Path("id") id: Int
|
||||
): Response<DataContainer<CategoryTemplate>>
|
||||
|
||||
@GET("outside/prompts")
|
||||
suspend fun getPromptsByCategory(
|
||||
@Query("categoryIds") categoryIds: List<Int>? = null,
|
||||
@Query("categoryName") categoryName: String? = null,
|
||||
@Query("uncategorized") uncategorized: String? = null,
|
||||
@Query("page") page: Int? = null,
|
||||
@Query("pageSize") pageSize: Int? = null
|
||||
): Response<ListContainer<Agent>>
|
||||
|
||||
}
|
||||
|
||||
|
||||
@@ -12,6 +12,7 @@ import androidx.compose.foundation.layout.Row
|
||||
import androidx.compose.foundation.layout.Spacer
|
||||
import androidx.compose.foundation.layout.WindowInsets
|
||||
import androidx.compose.foundation.layout.asPaddingValues
|
||||
import androidx.compose.foundation.layout.fillMaxHeight
|
||||
import androidx.compose.foundation.layout.fillMaxSize
|
||||
import androidx.compose.foundation.layout.fillMaxWidth
|
||||
import androidx.compose.foundation.layout.height
|
||||
@@ -36,6 +37,7 @@ import androidx.compose.runtime.mutableStateOf
|
||||
import androidx.compose.runtime.remember
|
||||
import androidx.compose.runtime.rememberCoroutineScope
|
||||
import androidx.compose.runtime.setValue
|
||||
import androidx.compose.runtime.snapshotFlow
|
||||
import androidx.compose.ui.Alignment
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.draw.clip
|
||||
@@ -47,6 +49,10 @@ import androidx.compose.ui.res.stringResource
|
||||
import androidx.compose.ui.text.font.FontWeight
|
||||
import androidx.compose.ui.unit.dp
|
||||
import androidx.compose.ui.unit.sp
|
||||
import androidx.compose.ui.platform.LocalContext
|
||||
import androidx.compose.ui.window.Dialog
|
||||
import androidx.compose.ui.window.DialogProperties
|
||||
import androidx.compose.material.CircularProgressIndicator
|
||||
import androidx.lifecycle.viewmodel.compose.viewModel
|
||||
import androidx.navigation.NavHostController
|
||||
import com.aiosman.ravenow.AppStore
|
||||
@@ -69,13 +75,11 @@ import com.aiosman.ravenow.utils.DebounceUtils
|
||||
import com.aiosman.ravenow.utils.ResourceCleanupManager
|
||||
import kotlinx.coroutines.launch
|
||||
import androidx.compose.foundation.lazy.LazyRow
|
||||
import androidx.compose.foundation.lazy.grid.GridCells
|
||||
import androidx.compose.foundation.lazy.grid.LazyVerticalGrid
|
||||
import androidx.compose.foundation.rememberScrollState
|
||||
import androidx.compose.foundation.verticalScroll
|
||||
import androidx.compose.foundation.lazy.grid.items
|
||||
import androidx.compose.ui.zIndex
|
||||
|
||||
@OptIn( ExperimentalFoundationApi::class)
|
||||
@OptIn(ExperimentalFoundationApi::class)
|
||||
@Composable
|
||||
fun Agent() {
|
||||
val AppColors = LocalAppTheme.current
|
||||
@@ -96,6 +100,18 @@ fun Agent() {
|
||||
viewModel.ensureDataLoaded()
|
||||
}
|
||||
|
||||
// 监听滚动状态,实现自动加载更多
|
||||
LaunchedEffect(scrollState) {
|
||||
snapshotFlow { scrollState.value }
|
||||
.collect { scrollValue ->
|
||||
val maxScroll = scrollState.maxValue
|
||||
if (scrollValue >= maxScroll - 100 && !viewModel.isLoading) {
|
||||
// 滚动到接近底部时加载更多网格数据
|
||||
viewModel.loadMoreGridAgentData()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 防抖状态
|
||||
var lastClickTime by remember { mutableStateOf(0L) }
|
||||
|
||||
@@ -107,27 +123,25 @@ fun Agent() {
|
||||
}
|
||||
}
|
||||
|
||||
Column(
|
||||
Box(
|
||||
modifier = Modifier.fillMaxSize()
|
||||
) {
|
||||
// 固定顶部搜索条
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.fillMaxSize()
|
||||
.verticalScroll(scrollState)
|
||||
.padding(
|
||||
top = statusBarPaddingValues.calculateTopPadding(),
|
||||
bottom = navigationBarPaddings,
|
||||
start = 16.dp,
|
||||
end = 16.dp
|
||||
),
|
||||
horizontalAlignment = Alignment.CenterHorizontally,
|
||||
.fillMaxWidth()
|
||||
.background(AppColors.background)
|
||||
.zIndex(999.0f)
|
||||
.height(44.dp + statusBarPaddingValues.calculateTopPadding())
|
||||
) {
|
||||
Row(
|
||||
modifier = Modifier
|
||||
.wrapContentHeight()
|
||||
.height(44.dp)
|
||||
.fillMaxWidth(),
|
||||
.fillMaxWidth()
|
||||
.fillMaxHeight().padding(top = 32.dp, start = 16.dp, end = 16.dp),
|
||||
horizontalArrangement = Arrangement.Start,
|
||||
verticalAlignment = Alignment.CenterVertically
|
||||
) {
|
||||
androidx.compose.material3.Text(
|
||||
Text(
|
||||
text = "Rave AI",
|
||||
fontSize = 20.sp,
|
||||
fontWeight = FontWeight.W900,
|
||||
@@ -149,58 +163,21 @@ fun Agent() {
|
||||
colorFilter = ColorFilter.tint(AppColors.text)
|
||||
)
|
||||
}
|
||||
Spacer(modifier = Modifier.height(15.dp))
|
||||
// // 搜索框
|
||||
// Row(
|
||||
// modifier = Modifier
|
||||
// .height(36.dp)
|
||||
// .weight(1f)
|
||||
// .clip(shape = RoundedCornerShape(8.dp))
|
||||
// .background(AppColors.inputBackground)
|
||||
// .padding(horizontal = 8.dp, vertical = 0.dp)
|
||||
// .noRippleClickable {
|
||||
// // 搜索框点击事件
|
||||
// },
|
||||
// verticalAlignment = Alignment.CenterVertically
|
||||
// ) {
|
||||
// Icon(
|
||||
// painter = painterResource(id = R.drawable.rider_pro_nav_search),
|
||||
// contentDescription = null,
|
||||
// tint = AppColors.inputHint
|
||||
// )
|
||||
// Box {
|
||||
// Text(
|
||||
// text = stringResource(R.string.search),
|
||||
// modifier = Modifier.padding(start = 8.dp),
|
||||
// color = AppColors.inputHint,
|
||||
// fontSize = 17.sp
|
||||
// )
|
||||
// }
|
||||
// }
|
||||
// Spacer(modifier = Modifier.width(16.dp))
|
||||
// // 创建智能体
|
||||
// Icon(
|
||||
// modifier = Modifier
|
||||
// .size(36.dp)
|
||||
// .noRippleClickable {
|
||||
// if (DebounceUtils.simpleDebounceClick(lastClickTime, 500L) {
|
||||
// // 检查游客模式,如果是游客则跳转登录
|
||||
// if (GuestLoginCheckOut.needLogin(GuestLoginCheckOutScene.CREATE_AGENT)) {
|
||||
// navController.navigate(NavigationRoute.Login.route)
|
||||
// } else {
|
||||
// // 导航到添加智能体页面
|
||||
// navController.navigate(
|
||||
// NavigationRoute.AddAgent.route
|
||||
// )
|
||||
// }
|
||||
// }) {
|
||||
// lastClickTime = System.currentTimeMillis()
|
||||
// }
|
||||
// },
|
||||
// painter = painterResource(id = R.drawable.rider_pro_new_post_add_pic),
|
||||
// contentDescription = null,
|
||||
// tint = AppColors.text
|
||||
// )
|
||||
}
|
||||
|
||||
// 可滚动的内容区域
|
||||
Column(
|
||||
modifier = Modifier
|
||||
.fillMaxSize()
|
||||
.verticalScroll(scrollState)
|
||||
.padding(
|
||||
top = 44.dp + statusBarPaddingValues.calculateTopPadding() + 15.dp,
|
||||
bottom = navigationBarPaddings,
|
||||
start = 16.dp,
|
||||
end = 16.dp
|
||||
),
|
||||
horizontalAlignment = Alignment.CenterHorizontally,
|
||||
) {
|
||||
|
||||
Column(
|
||||
modifier = Modifier
|
||||
@@ -208,30 +185,11 @@ fun Agent() {
|
||||
.padding(vertical = 8.dp)
|
||||
|
||||
) {
|
||||
// 使用 ViewModel 中的选中状态
|
||||
val selectedTabIndex = viewModel.selectedCategoryIndex
|
||||
|
||||
// // 标题
|
||||
// Row(
|
||||
// verticalAlignment = Alignment.CenterVertically,
|
||||
// modifier = Modifier.padding(bottom = 12.dp)
|
||||
// ) {
|
||||
// Image(
|
||||
// painter = painterResource(R.mipmap.rider_pro_agent2),
|
||||
// contentDescription = "agent",
|
||||
// modifier = Modifier.size(28.dp),
|
||||
//
|
||||
// )
|
||||
// Spacer(modifier = Modifier.width(4.dp))
|
||||
// androidx.compose.material3.Text(
|
||||
// text = stringResource(R.string.agent_recommend_agent),
|
||||
// fontSize = 16.sp,
|
||||
// fontWeight = androidx.compose.ui.text.font.FontWeight.W600,
|
||||
// color = AppColors.text
|
||||
// )
|
||||
// }
|
||||
|
||||
var selectedTabIndex by remember { mutableStateOf(0) }
|
||||
|
||||
// 标签页
|
||||
// 动态标签页
|
||||
if (viewModel.categories.isNotEmpty()) {
|
||||
LazyRow(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
@@ -240,105 +198,36 @@ fun Agent() {
|
||||
horizontalArrangement = Arrangement.Start,
|
||||
verticalAlignment = Alignment.CenterVertically
|
||||
) {
|
||||
viewModel.categories.forEachIndexed { index, category ->
|
||||
item {
|
||||
CustomTabItem(
|
||||
text = stringResource(R.string.agent_recommend),
|
||||
isSelected = selectedTabIndex == 0,
|
||||
text = category.getLocalizedName(),
|
||||
isSelected = selectedTabIndex == index,
|
||||
onClick = {
|
||||
selectedTabIndex = 0
|
||||
viewModel.loadAllAgents()
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
item {
|
||||
TabSpacer()
|
||||
}
|
||||
|
||||
// 动态添加分类标签
|
||||
viewModel.categories.take(4).forEachIndexed { index, category ->
|
||||
item {
|
||||
CustomTabItem(
|
||||
text = category.name,
|
||||
isSelected = selectedTabIndex == index + 1,
|
||||
onClick = {
|
||||
selectedTabIndex = index + 1
|
||||
viewModel.loadAgentsByCategory(category.id)
|
||||
viewModel.selectCategory(index)
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
if (index < viewModel.categories.size - 1) {
|
||||
item {
|
||||
TabSpacer()
|
||||
}
|
||||
}
|
||||
|
||||
item {
|
||||
CustomTabItem(
|
||||
text = "scenes",
|
||||
isSelected = selectedTabIndex == 1,
|
||||
onClick = {
|
||||
selectedTabIndex = 1
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 显示当前选中分类的 Agent 数据
|
||||
AgentViewPagerSection(agentItems = viewModel.topAgentItems, viewModel)
|
||||
}
|
||||
|
||||
// 推荐聊天房间
|
||||
ChatRoomsSection(
|
||||
chatRooms = viewModel.chatRooms,
|
||||
navController = LocalNavController.current
|
||||
)
|
||||
}
|
||||
|
||||
item {
|
||||
TabSpacer()
|
||||
}
|
||||
|
||||
item {
|
||||
CustomTabItem(
|
||||
text = "voices",
|
||||
isSelected = selectedTabIndex == 6,
|
||||
onClick = {
|
||||
selectedTabIndex = 6
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
item {
|
||||
TabSpacer()
|
||||
}
|
||||
|
||||
item {
|
||||
CustomTabItem(
|
||||
text = "anime",
|
||||
isSelected = selectedTabIndex == 7,
|
||||
onClick = {
|
||||
selectedTabIndex = 7
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
item {
|
||||
TabSpacer()
|
||||
}
|
||||
|
||||
item {
|
||||
CustomTabItem(
|
||||
text = "assist",
|
||||
isSelected = selectedTabIndex == 8,
|
||||
onClick = {
|
||||
selectedTabIndex = 8
|
||||
}
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
when {
|
||||
selectedTabIndex == 0 -> {
|
||||
AgentViewPagerSection(agentItems = viewModel.agentItems.take(15), viewModel)
|
||||
}
|
||||
selectedTabIndex in 1..viewModel.categories.size -> {
|
||||
AgentViewPagerSection(agentItems = viewModel.agentItems.take(15), viewModel)
|
||||
}
|
||||
else -> {
|
||||
val shuffledAgents = viewModel.agentItems.shuffled().take(15)
|
||||
AgentViewPagerSection(agentItems = shuffledAgents, viewModel)
|
||||
}
|
||||
}
|
||||
}
|
||||
Spacer(modifier = Modifier.height(20.dp))
|
||||
|
||||
Row(
|
||||
modifier = Modifier
|
||||
@@ -363,35 +252,78 @@ fun Agent() {
|
||||
)
|
||||
|
||||
}
|
||||
Spacer(modifier = Modifier.height(16.dp))
|
||||
Column(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
.weight(1f)
|
||||
) {
|
||||
val agentItems = viewModel.agentItems
|
||||
LazyVerticalGrid(
|
||||
columns = GridCells.Fixed(2),
|
||||
modifier = Modifier.fillMaxWidth(),
|
||||
horizontalArrangement = Arrangement.spacedBy(16.dp),
|
||||
verticalArrangement = Arrangement.spacedBy(32.dp)
|
||||
) {
|
||||
items(agentItems) { agentItem ->
|
||||
AgentCardSquare(
|
||||
agentItem = agentItem,
|
||||
Spacer(modifier = Modifier.height(20.dp))
|
||||
|
||||
// Agent两列网格布局
|
||||
AgentGridLayout(
|
||||
agentItems = viewModel.gridAgentItems,
|
||||
viewModel = viewModel,
|
||||
navController = LocalNavController.current
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
fun AgentGridLayout(
|
||||
agentItems: List<AgentItem>,
|
||||
viewModel: AgentViewModel,
|
||||
navController: NavHostController
|
||||
) {
|
||||
Column(
|
||||
modifier = Modifier.fillMaxWidth()
|
||||
) {
|
||||
// 将agentItems按两列分组
|
||||
agentItems.chunked(2).forEachIndexed { rowIndex, rowItems ->
|
||||
Row(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
.padding(
|
||||
top = if (rowIndex == 0) 30.dp else 20.dp, // 第一行添加更多顶部间距
|
||||
bottom = 20.dp
|
||||
),
|
||||
horizontalArrangement = Arrangement.spacedBy(16.dp)
|
||||
) {
|
||||
// 第一列
|
||||
Box(
|
||||
modifier = Modifier.weight(1f)
|
||||
) {
|
||||
AgentCardSquare(
|
||||
agentItem = rowItems[0],
|
||||
viewModel = viewModel,
|
||||
navController = navController
|
||||
)
|
||||
}
|
||||
|
||||
// 第二列(如果存在)
|
||||
if (rowItems.size > 1) {
|
||||
Box(
|
||||
modifier = Modifier.weight(1f)
|
||||
) {
|
||||
AgentCardSquare(
|
||||
agentItem = rowItems[1],
|
||||
viewModel = viewModel,
|
||||
navController = navController
|
||||
)
|
||||
}
|
||||
} else {
|
||||
// 如果只有一列,添加空白占位
|
||||
Spacer(modifier = Modifier.weight(1f))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@SuppressLint("SuspiciousIndentation")
|
||||
@Composable
|
||||
fun AgentCardSquare(agentItem: AgentItem, viewModel: AgentViewModel, navController: NavHostController) {
|
||||
fun AgentCardSquare(
|
||||
agentItem: AgentItem,
|
||||
viewModel: AgentViewModel,
|
||||
navController: NavHostController
|
||||
) {
|
||||
val AppColors = LocalAppTheme.current
|
||||
val cardHeight = 200.dp
|
||||
val cardHeight = 180.dp
|
||||
val avatarSize = cardHeight / 3 // 头像大小为方块高度的三分之一
|
||||
|
||||
// 防抖状态
|
||||
@@ -400,9 +332,8 @@ fun AgentCardSquare(agentItem: AgentItem, viewModel: AgentViewModel, navControll
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
.padding(top = avatarSize / 2)
|
||||
.height(cardHeight)
|
||||
.background(AppColors.nonActive, RoundedCornerShape(12.dp)) // 修改背景颜色
|
||||
.background(AppColors.secondaryBackground, RoundedCornerShape(12.dp)) // 主题背景
|
||||
.clickable {
|
||||
if (DebounceUtils.simpleDebounceClick(lastClickTime, 500L) {
|
||||
viewModel.goToProfile(agentItem.openId, navController)
|
||||
@@ -417,7 +348,7 @@ fun AgentCardSquare(agentItem: AgentItem, viewModel: AgentViewModel, navControll
|
||||
modifier = Modifier
|
||||
.offset(y = -avatarSize / 2)
|
||||
.size(avatarSize)
|
||||
.background(Color.White, RoundedCornerShape(avatarSize / 2))
|
||||
.background(AppColors.background, RoundedCornerShape(avatarSize / 2))
|
||||
.clip(RoundedCornerShape(avatarSize / 2)),
|
||||
contentAlignment = Alignment.Center
|
||||
) {
|
||||
@@ -444,12 +375,12 @@ fun AgentCardSquare(agentItem: AgentItem, viewModel: AgentViewModel, navControll
|
||||
Column(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
.padding(top = avatarSize / 2 + 8.dp, start = 8.dp, end = 8.dp, bottom = 8.dp),
|
||||
.padding(top = avatarSize / 2 + 8.dp, start = 8.dp, end = 8.dp, bottom = 48.dp), // 为底部聊天按钮留出空间
|
||||
horizontalAlignment = Alignment.CenterHorizontally
|
||||
) {
|
||||
androidx.compose.material3.Text(
|
||||
text = agentItem.title,
|
||||
fontSize = 14.sp,
|
||||
fontSize = 16.sp,
|
||||
fontWeight = androidx.compose.ui.text.font.FontWeight.W600,
|
||||
color = AppColors.text,
|
||||
maxLines = 1,
|
||||
@@ -458,29 +389,25 @@ fun AgentCardSquare(agentItem: AgentItem, viewModel: AgentViewModel, navControll
|
||||
|
||||
Spacer(modifier = Modifier.height(8.dp))
|
||||
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.height(85.dp)
|
||||
.fillMaxWidth()
|
||||
) {
|
||||
androidx.compose.material3.Text(
|
||||
text = agentItem.desc,
|
||||
fontSize = 12.sp,
|
||||
fontSize = 14.sp,
|
||||
color = AppColors.secondaryText,
|
||||
maxLines = 5,
|
||||
maxLines = 2,
|
||||
overflow = androidx.compose.ui.text.style.TextOverflow.Ellipsis,
|
||||
modifier = Modifier.weight(1f, fill = false)
|
||||
)
|
||||
}
|
||||
|
||||
Spacer(modifier = Modifier.height(8.dp))
|
||||
|
||||
// 聊天按钮,位于底部居中
|
||||
// 聊天按钮,固定在底部居中,距离底部有一定边距
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.width(60.dp)
|
||||
.align(Alignment.BottomCenter)
|
||||
.padding(bottom = 12.dp) // 距离底部的边距
|
||||
.width(80.dp)
|
||||
.height(32.dp)
|
||||
.background(
|
||||
color = Color(0X147c7480),
|
||||
color = AppColors.inputBackground,
|
||||
shape = RoundedCornerShape(8.dp)
|
||||
)
|
||||
.clickable {
|
||||
@@ -509,11 +436,11 @@ fun AgentCardSquare(agentItem: AgentItem, viewModel: AgentViewModel, navControll
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@OptIn(ExperimentalFoundationApi::class)
|
||||
@Composable
|
||||
fun AgentViewPagerSection(agentItems: List<AgentItem>,viewModel: AgentViewModel) {
|
||||
fun AgentViewPagerSection(agentItems: List<AgentItem>, viewModel: AgentViewModel) {
|
||||
val AppColors = LocalAppTheme.current
|
||||
|
||||
// 每页显示5个agent
|
||||
@@ -586,7 +513,13 @@ fun AgentViewPagerSection(agentItems: List<AgentItem>,viewModel: AgentViewModel)
|
||||
}
|
||||
|
||||
@Composable
|
||||
fun AgentPage(viewModel: AgentViewModel,agentItems: List<AgentItem>, page: Int, modifier: Modifier = Modifier,navController: NavHostController) {
|
||||
fun AgentPage(
|
||||
viewModel: AgentViewModel,
|
||||
agentItems: List<AgentItem>,
|
||||
page: Int,
|
||||
modifier: Modifier = Modifier,
|
||||
navController: NavHostController
|
||||
) {
|
||||
Column(
|
||||
modifier = modifier
|
||||
.fillMaxSize()
|
||||
@@ -594,7 +527,11 @@ fun AgentPage(viewModel: AgentViewModel,agentItems: List<AgentItem>, page: Int,
|
||||
) {
|
||||
// 显示3个agent
|
||||
agentItems.forEachIndexed { index, agentItem ->
|
||||
AgentCard2(agentItem = agentItem, viewModel = viewModel, navController = LocalNavController.current)
|
||||
AgentCard2(
|
||||
agentItem = agentItem,
|
||||
viewModel = viewModel,
|
||||
navController = LocalNavController.current
|
||||
)
|
||||
if (index < agentItems.size - 1) {
|
||||
Spacer(modifier = Modifier.height(8.dp))
|
||||
}
|
||||
@@ -604,7 +541,7 @@ fun AgentPage(viewModel: AgentViewModel,agentItems: List<AgentItem>, page: Int,
|
||||
|
||||
@SuppressLint("SuspiciousIndentation")
|
||||
@Composable
|
||||
fun AgentCard2(viewModel: AgentViewModel,agentItem: AgentItem,navController: NavHostController) {
|
||||
fun AgentCard2(viewModel: AgentViewModel, agentItem: AgentItem, navController: NavHostController) {
|
||||
val AppColors = LocalAppTheme.current
|
||||
|
||||
// 防抖状态
|
||||
@@ -620,7 +557,7 @@ fun AgentCard2(viewModel: AgentViewModel,agentItem: AgentItem,navController: Nav
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.size(48.dp)
|
||||
.background(Color(0xFFF5F5F5), RoundedCornerShape(24.dp))
|
||||
.background(AppColors.secondaryBackground, RoundedCornerShape(24.dp))
|
||||
.clickable {
|
||||
if (DebounceUtils.simpleDebounceClick(lastClickTime, 500L) {
|
||||
viewModel.goToProfile(agentItem.openId, navController)
|
||||
@@ -684,7 +621,7 @@ fun AgentCard2(viewModel: AgentViewModel,agentItem: AgentItem,navController: Nav
|
||||
modifier = Modifier
|
||||
.size(width = 60.dp, height = 32.dp)
|
||||
.background(
|
||||
color = Color(0X147c7480),
|
||||
color = AppColors.inputBackground,
|
||||
shape = RoundedCornerShape(8.dp)
|
||||
)
|
||||
.clickable {
|
||||
@@ -714,3 +651,186 @@ fun AgentCard2(viewModel: AgentViewModel,agentItem: AgentItem,navController: Nav
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
fun ChatRoomsSection(
|
||||
chatRooms: List<ChatRoom>,
|
||||
navController: NavHostController
|
||||
) {
|
||||
val AppColors = LocalAppTheme.current
|
||||
|
||||
Column(
|
||||
modifier = Modifier.fillMaxWidth()
|
||||
) {
|
||||
// 标题
|
||||
Row(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
.padding(bottom = 16.dp),
|
||||
verticalAlignment = Alignment.CenterVertically
|
||||
) {
|
||||
Image(
|
||||
painter = painterResource(R.mipmap.rider_pro_agent2),
|
||||
contentDescription = "chat room",
|
||||
modifier = Modifier.size(28.dp)
|
||||
)
|
||||
Spacer(modifier = Modifier.width(4.dp))
|
||||
androidx.compose.material3.Text(
|
||||
text = "群聊",
|
||||
fontSize = 16.sp,
|
||||
fontWeight = androidx.compose.ui.text.font.FontWeight.W600,
|
||||
color = AppColors.text
|
||||
)
|
||||
}
|
||||
|
||||
// 3行宫格布局
|
||||
Column(
|
||||
modifier = Modifier.fillMaxWidth()
|
||||
) {
|
||||
// 将聊天房间按3个一组分组
|
||||
chatRooms.chunked(3).forEach { rowRooms ->
|
||||
Row(
|
||||
modifier = Modifier
|
||||
.fillMaxWidth()
|
||||
.padding(bottom = 12.dp),
|
||||
horizontalArrangement = Arrangement.spacedBy(12.dp)
|
||||
) {
|
||||
rowRooms.forEach { chatRoom ->
|
||||
ChatRoomCard(
|
||||
chatRoom = chatRoom,
|
||||
navController = navController,
|
||||
modifier = Modifier.weight(1f)
|
||||
)
|
||||
}
|
||||
// 如果这一行不足3个,添加空白占位
|
||||
repeat(3 - rowRooms.size) {
|
||||
Spacer(modifier = Modifier.weight(1f))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
fun ChatRoomCard(
|
||||
chatRoom: ChatRoom,
|
||||
navController: NavHostController,
|
||||
modifier: Modifier = Modifier
|
||||
) {
|
||||
val AppColors = LocalAppTheme.current
|
||||
val cardSize = 100.dp
|
||||
val viewModel: AgentViewModel = viewModel()
|
||||
val context = LocalContext.current
|
||||
|
||||
// 防抖状态
|
||||
var lastClickTime by remember { mutableStateOf(0L) }
|
||||
|
||||
// Loading 对话框
|
||||
if (viewModel.isJoiningRoom) {
|
||||
Dialog(
|
||||
onDismissRequest = { /* 阻止用户关闭对话框 */ },
|
||||
properties = DialogProperties(
|
||||
dismissOnBackPress = false,
|
||||
dismissOnClickOutside = false
|
||||
)
|
||||
) {
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.size(120.dp)
|
||||
.background(
|
||||
color = AppColors.background,
|
||||
shape = RoundedCornerShape(12.dp)
|
||||
),
|
||||
contentAlignment = Alignment.Center
|
||||
) {
|
||||
Column(
|
||||
horizontalAlignment = Alignment.CenterHorizontally,
|
||||
verticalArrangement = Arrangement.Center
|
||||
) {
|
||||
CircularProgressIndicator(
|
||||
modifier = Modifier.size(32.dp),
|
||||
color = AppColors.main
|
||||
)
|
||||
Spacer(modifier = Modifier.height(12.dp))
|
||||
androidx.compose.material3.Text(
|
||||
text = "加入中...",
|
||||
fontSize = 14.sp,
|
||||
color = AppColors.text
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 正方形卡片,文字重叠在底部
|
||||
Box(
|
||||
modifier = modifier
|
||||
.size(cardSize)
|
||||
.background(AppColors.secondaryBackground, RoundedCornerShape(12.dp))
|
||||
.clickable(enabled = !viewModel.isJoiningRoom) {
|
||||
if (!viewModel.isJoiningRoom && DebounceUtils.simpleDebounceClick(lastClickTime, 500L) {
|
||||
// 加入群聊房间
|
||||
viewModel.joinRoom(
|
||||
id = chatRoom.id,
|
||||
name = chatRoom.name,
|
||||
avatar = chatRoom.avatar,
|
||||
context = context,
|
||||
navController = navController,
|
||||
onSuccess = {
|
||||
// 成功加入房间
|
||||
},
|
||||
onError = { errorMsg ->
|
||||
// 处理错误,可以显示Toast或其他提示
|
||||
}
|
||||
)
|
||||
}) {
|
||||
lastClickTime = System.currentTimeMillis()
|
||||
}
|
||||
}
|
||||
) {
|
||||
// 优先显示banner,如果没有banner则显示头像
|
||||
val imageUrl = if (chatRoom.banner.isNotEmpty()) chatRoom.banner else chatRoom.avatar
|
||||
|
||||
if (imageUrl.isNotEmpty()) {
|
||||
CustomAsyncImage(
|
||||
imageUrl = imageUrl,
|
||||
contentDescription = if (chatRoom.banner.isNotEmpty()) "房间banner" else "房间头像",
|
||||
modifier = Modifier
|
||||
.size(cardSize)
|
||||
.clip(RoundedCornerShape(12.dp)),
|
||||
contentScale = androidx.compose.ui.layout.ContentScale.Crop
|
||||
)
|
||||
} else {
|
||||
// 默认房间图标
|
||||
Image(
|
||||
painter = painterResource(R.mipmap.rider_pro_agent),
|
||||
contentDescription = "默认房间图标",
|
||||
modifier = Modifier.size(cardSize * 0.4f),
|
||||
colorFilter = ColorFilter.tint(AppColors.secondaryText)
|
||||
)
|
||||
}
|
||||
|
||||
// 房间名称,重叠在底部
|
||||
Box(
|
||||
modifier = Modifier
|
||||
.align(Alignment.BottomCenter)
|
||||
.fillMaxWidth()
|
||||
.background(
|
||||
color = Color.Black.copy(alpha = 0.6f),
|
||||
shape = RoundedCornerShape(bottomStart = 12.dp, bottomEnd = 12.dp)
|
||||
)
|
||||
.padding(horizontal = 8.dp, vertical = 6.dp)
|
||||
) {
|
||||
androidx.compose.material3.Text(
|
||||
text = chatRoom.name,
|
||||
fontSize = 12.sp,
|
||||
color = Color.White,
|
||||
maxLines = 1,
|
||||
overflow = androidx.compose.ui.text.style.TextOverflow.Ellipsis,
|
||||
modifier = Modifier.fillMaxWidth(),
|
||||
textAlign = androidx.compose.ui.text.style.TextAlign.Center
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,74 +1,131 @@
|
||||
package com.aiosman.ravenow.ui.index.tabs.ai
|
||||
|
||||
import android.util.Log
|
||||
import androidx.compose.runtime.getValue
|
||||
import androidx.compose.runtime.mutableStateOf
|
||||
import androidx.compose.runtime.setValue
|
||||
import androidx.lifecycle.ViewModel
|
||||
import androidx.lifecycle.viewModelScope
|
||||
import androidx.navigation.NavHostController
|
||||
import com.aiosman.ravenow.data.Agent
|
||||
import com.aiosman.ravenow.data.ListContainer
|
||||
import com.aiosman.ravenow.AppState
|
||||
import com.aiosman.ravenow.AppStore
|
||||
import com.aiosman.ravenow.ConstVars
|
||||
import com.aiosman.ravenow.data.Room
|
||||
import com.aiosman.ravenow.data.api.ApiClient
|
||||
import com.aiosman.ravenow.data.api.CategoryTemplate
|
||||
import com.aiosman.ravenow.data.api.CreateGroupChatRequestBody
|
||||
import com.aiosman.ravenow.data.api.RaveNowAPI
|
||||
import com.aiosman.ravenow.data.api.SingleChatRequestBody
|
||||
import com.aiosman.ravenow.data.api.JoinGroupChatRequestBody
|
||||
import com.aiosman.ravenow.ui.index.tabs.ai.tabs.mine.MineAgentViewModel.createGroup2ChatAi
|
||||
import com.aiosman.ravenow.ui.NavigationRoute
|
||||
import com.aiosman.ravenow.ui.index.tabs.message.MessageListViewModel.userService
|
||||
import com.aiosman.ravenow.ui.index.tabs.message.tab.GroupChatListViewModel.createGroupChat
|
||||
import com.aiosman.ravenow.ui.index.tabs.moment.tabs.expolre.AgentItem
|
||||
import com.aiosman.ravenow.ui.navigateToGroupChat
|
||||
import com.aiosman.ravenow.data.api.ApiErrorResponse
|
||||
import com.google.gson.Gson
|
||||
import android.content.Context
|
||||
import android.widget.Toast
|
||||
import kotlinx.coroutines.launch
|
||||
|
||||
object AgentViewModel: ViewModel() {
|
||||
data class ChatRoom(
|
||||
val id: Int,
|
||||
val name: String,
|
||||
val avatar: String = "",
|
||||
val banner: String = "",
|
||||
val memberCount: Int = 0
|
||||
)
|
||||
|
||||
object AgentViewModel : ViewModel() {
|
||||
|
||||
private val apiClient: RaveNowAPI = ApiClient.api
|
||||
|
||||
var agentItems by mutableStateOf<List<AgentItem>>(emptyList())
|
||||
// 顶部Agent列表数据(用于ViewPager)
|
||||
var topAgentItems by mutableStateOf<List<AgentItem>>(emptyList())
|
||||
private set
|
||||
|
||||
var categories by mutableStateOf<List<CategoryItem>>(emptyList())
|
||||
// 底部网格布局Agent数据
|
||||
var gridAgentItems by mutableStateOf<List<AgentItem>>(emptyList())
|
||||
private set
|
||||
|
||||
var chatRooms by mutableStateOf<List<ChatRoom>>(emptyList())
|
||||
private set
|
||||
|
||||
var rooms by mutableStateOf<List<Room>>(emptyList())
|
||||
private set
|
||||
|
||||
var categories by mutableStateOf<List<CategoryTemplate>>(emptyList())
|
||||
private set
|
||||
|
||||
var selectedCategoryIndex by mutableStateOf(0)
|
||||
private set
|
||||
|
||||
var errorMessage by mutableStateOf<String?>(null)
|
||||
private set
|
||||
|
||||
|
||||
var isRefreshing by mutableStateOf(false)
|
||||
private set
|
||||
|
||||
var isLoading by mutableStateOf(false)
|
||||
private set
|
||||
|
||||
var isJoiningRoom by mutableStateOf(false)
|
||||
private set
|
||||
|
||||
private var topCurrentPage = 1
|
||||
private var topHasMoreData = true
|
||||
|
||||
private var gridCurrentPage = 1
|
||||
private var gridHasMoreData = true
|
||||
|
||||
init {
|
||||
loadAgentData()
|
||||
loadTopAgentData()
|
||||
loadGridAgentData()
|
||||
loadChatRooms()
|
||||
loadCategories()
|
||||
}
|
||||
|
||||
private fun loadAgentData(categoryId: Int? = null) {
|
||||
/**
|
||||
* 加载顶部Agent列表数据(用于ViewPager)
|
||||
*/
|
||||
private fun loadTopAgentData(categoryIndex: Int = 0) {
|
||||
viewModelScope.launch {
|
||||
isLoading = true
|
||||
errorMessage = null
|
||||
topCurrentPage = 1
|
||||
topHasMoreData = true
|
||||
try {
|
||||
val response = if (categoryId != null) {
|
||||
// 根据分类ID获取智能体
|
||||
val selectedCategory =
|
||||
if (categoryIndex < categories.size) categories[categoryIndex] else null
|
||||
|
||||
val response = if (categoryIndex == 0 || selectedCategory == null) {
|
||||
// 推荐分类或无效分类,加载所有 Agent
|
||||
apiClient.getAgent(
|
||||
page = 1,
|
||||
pageSize = 20,
|
||||
withWorkflow = 1,
|
||||
categoryIds = listOf(categoryId)
|
||||
page = topCurrentPage,
|
||||
pageSize = 15,
|
||||
withWorkflow = "1",
|
||||
random = "1"
|
||||
)
|
||||
} else {
|
||||
// 获取所有智能体
|
||||
apiClient.getAgent(page = 1, pageSize = 20, withWorkflow = 1)
|
||||
// 特定分类,使用 categoryName 参数
|
||||
apiClient.getAgent(
|
||||
page = topCurrentPage,
|
||||
pageSize = 15,
|
||||
withWorkflow = "1",
|
||||
categoryName = selectedCategory.name,
|
||||
random = "1"
|
||||
)
|
||||
}
|
||||
|
||||
if (response.isSuccessful) {
|
||||
val agents = response.body()?.data?.list ?: emptyList<Agent>()
|
||||
|
||||
agentItems = agents.map { agent ->
|
||||
val agents = response.body()?.data?.list ?: emptyList()
|
||||
topAgentItems = agents.map { agent ->
|
||||
AgentItem.fromAgent(agent)
|
||||
}
|
||||
topHasMoreData = agents.size >= 15
|
||||
} else {
|
||||
errorMessage = "获取Agent数据失败: ${response.code()}"
|
||||
errorMessage = "获取顶部Agent数据失败: ${response.code()}"
|
||||
}
|
||||
} catch (e: Exception) {
|
||||
errorMessage = "网络请求失败: ${e.message}"
|
||||
@@ -78,58 +135,226 @@ object AgentViewModel: ViewModel() {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 加载底部网格布局Agent数据
|
||||
*/
|
||||
private fun loadGridAgentData(categoryIndex: Int = 0) {
|
||||
viewModelScope.launch {
|
||||
isLoading = true
|
||||
errorMessage = null
|
||||
gridCurrentPage = 1
|
||||
gridHasMoreData = true
|
||||
try {
|
||||
val selectedCategory =
|
||||
if (categoryIndex < categories.size) categories[categoryIndex] else null
|
||||
|
||||
val response = if (categoryIndex == 0 || selectedCategory == null) {
|
||||
// 推荐分类或无效分类,加载所有 Agent
|
||||
apiClient.getAgent(
|
||||
page = gridCurrentPage,
|
||||
pageSize = 20,
|
||||
withWorkflow = "1",
|
||||
random = "true"
|
||||
)
|
||||
} else {
|
||||
// 特定分类,使用 categoryName 参数
|
||||
apiClient.getAgent(
|
||||
page = gridCurrentPage,
|
||||
pageSize = 20,
|
||||
withWorkflow = "1",
|
||||
categoryName = selectedCategory.name,
|
||||
random = "true"
|
||||
)
|
||||
}
|
||||
|
||||
if (response.isSuccessful) {
|
||||
val agents = response.body()?.data?.list ?: emptyList()
|
||||
gridAgentItems = agents.map { agent ->
|
||||
AgentItem.fromAgent(agent)
|
||||
}
|
||||
gridHasMoreData = agents.size >= 20
|
||||
} else {
|
||||
errorMessage = "获取网格Agent数据失败: ${response.code()}"
|
||||
}
|
||||
} catch (e: Exception) {
|
||||
errorMessage = "网络请求失败: ${e.message}"
|
||||
} finally {
|
||||
isLoading = false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private fun loadChatRooms() {
|
||||
viewModelScope.launch {
|
||||
try {
|
||||
val response = apiClient.getRooms(
|
||||
page = 1,
|
||||
pageSize = 21,
|
||||
isRecommended = 1,
|
||||
random = 1
|
||||
) // 请求21个,确保是3的倍数
|
||||
if (response.isSuccessful) {
|
||||
val allRooms = response.body()?.list ?: emptyList()
|
||||
// 确保房间数量是3的倍数,如果不足则截取,如果超出则取前几个
|
||||
val targetCount = (allRooms.size / 3) * 3 // 向下取整到最近的3的倍数
|
||||
rooms = allRooms.take(targetCount)
|
||||
|
||||
// 转换为ChatRoom格式用于兼容现有UI
|
||||
chatRooms = rooms.map { room ->
|
||||
ChatRoom(
|
||||
id = room.id,
|
||||
name = room.name,
|
||||
avatar = room.avatar,
|
||||
banner = ConstVars.BASE_SERVER + "/api/v1/outside/" + room.recommendBanner + "?token=${AppStore.token}",
|
||||
memberCount = room.userCount
|
||||
)
|
||||
}
|
||||
} else {
|
||||
|
||||
}
|
||||
} catch (e: Exception) {
|
||||
// 如果网络请求失败,使用默认数据
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private fun loadCategories() {
|
||||
viewModelScope.launch {
|
||||
try {
|
||||
val response = apiClient.getCategories(
|
||||
pageSize = 20,
|
||||
withChildren = false,
|
||||
withParent = false,
|
||||
withCount = true,
|
||||
hideEmpty = true
|
||||
val categoriesResponse = apiClient.getCategories(
|
||||
page = 1,
|
||||
pageSize = 100,
|
||||
)
|
||||
println("分类数据请求完成,响应成功: ${response.isSuccessful}")
|
||||
if (response.isSuccessful) {
|
||||
val categoryList = response.body()?.data?.list ?: emptyList()
|
||||
println("获取到 ${categoryList.size} 个分类")
|
||||
categories = categoryList.map { category ->
|
||||
CategoryItem.fromCategoryTemplate(category)
|
||||
if (categoriesResponse.isSuccessful && categoriesResponse.body() != null) {
|
||||
// 添加一个默认的"推荐"分类在第一位
|
||||
val recommendCategory = createRecommendCategory()
|
||||
val categoriesList = categoriesResponse.body()?.list ?: emptyList()
|
||||
categories = listOf(recommendCategory) + categoriesList
|
||||
|
||||
// 分类加载完成后,重新加载当前选中分类的 Agent 数据
|
||||
if (topAgentItems.isEmpty()) {
|
||||
loadTopAgentData(selectedCategoryIndex)
|
||||
}
|
||||
if (gridAgentItems.isEmpty()) {
|
||||
loadGridAgentData(selectedCategoryIndex)
|
||||
}
|
||||
println("成功处理并映射了 ${categories.size} 个分类")
|
||||
} else {
|
||||
errorMessage = "获取分类数据失败: ${response.code()}"
|
||||
println("获取分类数据失败: ${response.code()}")
|
||||
// 如果请求失败,使用默认分类
|
||||
categories = listOf(createRecommendCategory())
|
||||
errorMessage = "获取分类失败: ${categoriesResponse.code()}"
|
||||
}
|
||||
} catch (e: Exception) {
|
||||
errorMessage = "获取分类数据失败: ${e.message}"
|
||||
println("获取分类数据异常: ${e.message}")
|
||||
e.printStackTrace()
|
||||
// 如果网络请求失败,使用默认分类
|
||||
categories = listOf(createRecommendCategory())
|
||||
errorMessage = "网络请求失败: ${e.message}"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun loadAgentsByCategory(categoryId: Int) {
|
||||
loadAgentData(categoryId)
|
||||
/**
|
||||
* 创建推荐分类
|
||||
*/
|
||||
private fun createRecommendCategory(): CategoryTemplate {
|
||||
return CategoryTemplate(
|
||||
id = 0,
|
||||
name = "推荐",
|
||||
description = "推荐内容",
|
||||
avatar = "",
|
||||
parentId = null,
|
||||
parent = null,
|
||||
children = null,
|
||||
sort = 0,
|
||||
isActive = true,
|
||||
promptCount = 0,
|
||||
createdAt = "",
|
||||
updatedAt = "",
|
||||
translations = null
|
||||
)
|
||||
}
|
||||
|
||||
fun loadAllAgents() {
|
||||
loadAgentData()
|
||||
|
||||
/**
|
||||
* 加载更多网格Agent数据
|
||||
*/
|
||||
fun loadMoreGridAgentData() {
|
||||
if (!gridHasMoreData || isLoading) return
|
||||
|
||||
viewModelScope.launch {
|
||||
isLoading = true
|
||||
try {
|
||||
val nextPage = gridCurrentPage + 1
|
||||
val selectedCategory =
|
||||
if (selectedCategoryIndex < categories.size) categories[selectedCategoryIndex] else null
|
||||
|
||||
val response = if (selectedCategoryIndex == 0 || selectedCategory == null) {
|
||||
// 推荐分类或无效分类,加载所有 Agent
|
||||
apiClient.getAgent(
|
||||
page = nextPage,
|
||||
pageSize = 20,
|
||||
withWorkflow = "1",
|
||||
random = "true"
|
||||
)
|
||||
} else {
|
||||
// 特定分类,使用 categoryName 参数
|
||||
apiClient.getAgent(
|
||||
page = nextPage,
|
||||
pageSize = 20,
|
||||
withWorkflow = "1",
|
||||
categoryName = selectedCategory.name,
|
||||
random = "true"
|
||||
)
|
||||
}
|
||||
|
||||
if (response.isSuccessful) {
|
||||
val agents = response.body()?.data?.list ?: emptyList()
|
||||
val newAgentItems = agents.map { agent ->
|
||||
AgentItem.fromAgent(agent)
|
||||
}
|
||||
gridAgentItems = gridAgentItems + newAgentItems
|
||||
gridCurrentPage = nextPage
|
||||
gridHasMoreData = agents.size >= 20
|
||||
} else {
|
||||
errorMessage = "加载更多网格Agent数据失败: ${response.code()}"
|
||||
}
|
||||
} catch (e: Exception) {
|
||||
errorMessage = "网络请求失败: ${e.message}"
|
||||
} finally {
|
||||
isLoading = false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 选择分类并加载对应的 Agent 数据
|
||||
*/
|
||||
fun selectCategory(categoryIndex: Int) {
|
||||
if (categoryIndex != selectedCategoryIndex && categoryIndex >= 0 && categoryIndex < categories.size) {
|
||||
selectedCategoryIndex = categoryIndex
|
||||
// 同时加载顶部和网格的数据
|
||||
loadTopAgentData(categoryIndex)
|
||||
loadGridAgentData(categoryIndex)
|
||||
}
|
||||
}
|
||||
|
||||
fun createSingleChat(
|
||||
openId: String,
|
||||
) {
|
||||
viewModelScope.launch {
|
||||
val response = ApiClient.api.createSingleChat(SingleChatRequestBody(agentOpenId = openId))
|
||||
val response =
|
||||
ApiClient.api.createSingleChat(SingleChatRequestBody(agentOpenId = openId))
|
||||
Log.d("debug", response.toString())
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
fun goToChatAi(
|
||||
openId: String,
|
||||
navController: NavHostController
|
||||
) {
|
||||
viewModelScope.launch {
|
||||
val profile = userService.getUserProfileByOpenId(openId)
|
||||
createGroup2ChatAi(profile.trtcUserId,"ai_group",navController,profile.id)
|
||||
createGroup2ChatAi(profile.trtcUserId, "ai_group", navController, profile.id)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -156,15 +381,98 @@ object AgentViewModel: ViewModel() {
|
||||
* 刷新推荐Agent数据
|
||||
*/
|
||||
fun refreshAgentData() {
|
||||
loadAgentData()
|
||||
loadTopAgentData()
|
||||
loadGridAgentData()
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查数据是否为空,如果为空则重新加载
|
||||
*/
|
||||
fun ensureDataLoaded() {
|
||||
if (agentItems.isEmpty() && !isLoading) {
|
||||
loadAgentData()
|
||||
if (topAgentItems.isEmpty() && !isLoading) {
|
||||
loadTopAgentData()
|
||||
}
|
||||
if (gridAgentItems.isEmpty() && !isLoading) {
|
||||
loadGridAgentData()
|
||||
}
|
||||
// 同时确保分类数据已加载
|
||||
if (categories.isEmpty() && !isLoading) {
|
||||
loadCategories()
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 加入房间
|
||||
*/
|
||||
fun joinRoom(
|
||||
id: Int,
|
||||
name: String,
|
||||
avatar: String,
|
||||
context: Context,
|
||||
navController: NavHostController,
|
||||
onSuccess: () -> Unit,
|
||||
onError: (String) -> Unit
|
||||
) {
|
||||
// 防止重复点击
|
||||
if (isJoiningRoom) return
|
||||
|
||||
viewModelScope.launch {
|
||||
try {
|
||||
isJoiningRoom = true
|
||||
val response = apiClient.joinRoom(JoinGroupChatRequestBody(roomId = id))
|
||||
if (response.isSuccessful) {
|
||||
// 打开房间
|
||||
val openRoomResponse = apiClient.createGroupChatAi(
|
||||
roomId = id
|
||||
)
|
||||
|
||||
if (openRoomResponse.isSuccessful){
|
||||
val respData = openRoomResponse.body()
|
||||
respData?.let {
|
||||
viewModelScope.launch {
|
||||
try {
|
||||
// 群聊直接使用群ID进行导航
|
||||
navController.navigateToGroupChat(
|
||||
id = respData.data.trtcRoomId,
|
||||
name = name,
|
||||
avatar = avatar
|
||||
)
|
||||
} catch (e: Exception) {
|
||||
onError("加入房间失败")
|
||||
e.printStackTrace()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
onSuccess()
|
||||
|
||||
} else {
|
||||
// 处理错误响应
|
||||
try {
|
||||
val errorBody = response.errorBody()?.string()
|
||||
if (errorBody != null) {
|
||||
val gson = Gson()
|
||||
val errorResponse = gson.fromJson(errorBody, ApiErrorResponse::class.java)
|
||||
|
||||
// 在主线程显示 Toast
|
||||
Toast.makeText(context, errorResponse.error, Toast.LENGTH_LONG).show()
|
||||
onError(errorResponse.error)
|
||||
} else {
|
||||
Toast.makeText(context, "加入房间失败", Toast.LENGTH_SHORT).show()
|
||||
onError("加入房间失败")
|
||||
}
|
||||
} catch (parseException: Exception) {
|
||||
// 如果解析错误响应失败,显示默认错误信息
|
||||
Toast.makeText(context, "加入房间失败", Toast.LENGTH_SHORT).show()
|
||||
onError("加入房间失败")
|
||||
}
|
||||
}
|
||||
} catch (e: Exception) {
|
||||
Toast.makeText(context, "网络请求失败:${e.message}", Toast.LENGTH_SHORT).show()
|
||||
onError("网络请求失败:${e.message}")
|
||||
} finally {
|
||||
isJoiningRoom = false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -172,29 +480,18 @@ object AgentViewModel: ViewModel() {
|
||||
* 重置ViewModel状态,用于登出或切换账号时清理数据
|
||||
*/
|
||||
fun ResetModel() {
|
||||
agentItems = emptyList()
|
||||
topAgentItems = emptyList()
|
||||
gridAgentItems = emptyList()
|
||||
categories = emptyList()
|
||||
selectedCategoryIndex = 0
|
||||
errorMessage = null
|
||||
isRefreshing = false
|
||||
isLoading = false
|
||||
isJoiningRoom = false
|
||||
topCurrentPage = 1
|
||||
topHasMoreData = true
|
||||
gridCurrentPage = 1
|
||||
gridHasMoreData = true
|
||||
}
|
||||
|
||||
}
|
||||
data class CategoryItem(
|
||||
val id: Int,
|
||||
val name: String,
|
||||
val description: String,
|
||||
val avatar: String,
|
||||
val promptCount: Int?
|
||||
) {
|
||||
companion object {
|
||||
fun fromCategoryTemplate(template: CategoryTemplate): CategoryItem {
|
||||
return CategoryItem(
|
||||
id = template.id,
|
||||
name = template.name,
|
||||
description = template.description,
|
||||
avatar = "${ApiClient.BASE_API_URL}${template.avatar}",
|
||||
promptCount = template.promptCount
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -2,6 +2,7 @@ package com.aiosman.ravenow.ui.index.tabs.message.tab
|
||||
|
||||
import android.content.Context
|
||||
import android.icu.util.Calendar
|
||||
import android.util.Log
|
||||
import androidx.compose.runtime.getValue
|
||||
import androidx.compose.runtime.mutableStateOf
|
||||
import androidx.compose.runtime.setValue
|
||||
@@ -167,10 +168,12 @@ object GroupChatListViewModel : ViewModel() {
|
||||
}
|
||||
|
||||
fun createGroupChat(
|
||||
trtcGroupId: String,
|
||||
trtcGroupId: String? = null,
|
||||
roomId: Int? = null
|
||||
) {
|
||||
viewModelScope.launch {
|
||||
val response = ApiClient.api.createGroupChatAi(trtcGroupId = trtcGroupId)
|
||||
val response = ApiClient.api.createGroupChatAi(trtcGroupId = trtcGroupId,roomId = roomId)
|
||||
Log.d("debug",response.toString())
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
@@ -369,6 +369,7 @@ fun Explore() {
|
||||
trtcId = roomItem.trtcId.toString(),
|
||||
name = roomItem.title,
|
||||
avatar = roomItem.avatar,
|
||||
context = context,
|
||||
navController = navController,
|
||||
onSuccess = {
|
||||
Toast.makeText(context, enterSuccessText, Toast.LENGTH_SHORT).show()
|
||||
@@ -636,6 +637,7 @@ fun Explore() {
|
||||
trtcId = bannerItem.trtcId.toString(),
|
||||
name = bannerItem.title,
|
||||
avatar = bannerItem.avatar,
|
||||
context = context,
|
||||
navController = navController,
|
||||
onSuccess = {
|
||||
Toast.makeText(context, enterSuccessText, Toast.LENGTH_SHORT).show()
|
||||
|
||||
@@ -17,6 +17,10 @@ import com.aiosman.ravenow.ui.index.tabs.message.MessageListViewModel.userServic
|
||||
import com.aiosman.ravenow.ui.index.tabs.message.tab.GroupChatListViewModel
|
||||
import com.aiosman.ravenow.ui.index.tabs.message.tab.GroupChatListViewModel.createGroupChat
|
||||
import com.aiosman.ravenow.ui.navigateToGroupChat
|
||||
import com.aiosman.ravenow.data.api.ApiErrorResponse
|
||||
import com.google.gson.Gson
|
||||
import android.content.Context
|
||||
import android.widget.Toast
|
||||
import kotlinx.coroutines.launch
|
||||
|
||||
class ExploreViewModel : ViewModel() {
|
||||
@@ -70,7 +74,7 @@ class ExploreViewModel : ViewModel() {
|
||||
isRefreshing = true
|
||||
errorMessage = null
|
||||
try {
|
||||
val response = apiClient.getAgent(page = 1, pageSize = 20, withWorkflow = 1)
|
||||
val response = apiClient.getAgent(page = 1, pageSize = 20, withWorkflow = "1")
|
||||
if (response.isSuccessful) {
|
||||
val agents = response.body()?.data?.list ?: emptyList()
|
||||
agentItems = agents.map { agent ->
|
||||
@@ -114,7 +118,7 @@ class ExploreViewModel : ViewModel() {
|
||||
isLoading = true
|
||||
errorMessage = null
|
||||
try {
|
||||
val response = apiClient.getAgent(page = 1, pageSize = 20, withWorkflow = 1)
|
||||
val response = apiClient.getAgent(page = 1, pageSize = 20, withWorkflow = "1")
|
||||
if (response.isSuccessful) {
|
||||
val agents = response.body()?.data?.list ?: emptyList()
|
||||
agentItems = agents.map { agent ->
|
||||
@@ -130,21 +134,24 @@ class ExploreViewModel : ViewModel() {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun createSingleChat(
|
||||
openId: String,
|
||||
) {
|
||||
viewModelScope.launch {
|
||||
val response = ApiClient.api.createSingleChat(SingleChatRequestBody(agentOpenId = openId))
|
||||
val response =
|
||||
ApiClient.api.createSingleChat(SingleChatRequestBody(agentOpenId = openId))
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
fun goToChatAi(
|
||||
openId: String,
|
||||
navController: NavHostController
|
||||
) {
|
||||
viewModelScope.launch {
|
||||
val profile = userService.getUserProfileByOpenId(openId)
|
||||
createGroup2ChatAi(profile.trtcUserId,"ai_group",navController,profile.id)
|
||||
createGroup2ChatAi(profile.trtcUserId, "ai_group", navController, profile.id)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -152,6 +159,7 @@ class ExploreViewModel : ViewModel() {
|
||||
trtcId: String,
|
||||
name: String,
|
||||
avatar: String,
|
||||
context: Context,
|
||||
navController: NavHostController,
|
||||
onSuccess: () -> Unit,
|
||||
onError: (String) -> Unit
|
||||
@@ -164,9 +172,11 @@ class ExploreViewModel : ViewModel() {
|
||||
try {
|
||||
createGroupChat(trtcGroupId = trtcId)
|
||||
// 群聊直接使用群ID进行导航
|
||||
navController.navigateToGroupChat( id = trtcId,
|
||||
navController.navigateToGroupChat(
|
||||
id = trtcId,
|
||||
name = name,
|
||||
avatar = avatar)
|
||||
avatar = avatar
|
||||
)
|
||||
} catch (e: Exception) {
|
||||
onError("加入房间失败")
|
||||
e.printStackTrace()
|
||||
@@ -175,9 +185,28 @@ class ExploreViewModel : ViewModel() {
|
||||
onSuccess()
|
||||
|
||||
} else {
|
||||
// 处理错误响应
|
||||
try {
|
||||
val errorBody = response.errorBody()?.string()
|
||||
if (errorBody != null) {
|
||||
val gson = Gson()
|
||||
val errorResponse = gson.fromJson(errorBody, ApiErrorResponse::class.java)
|
||||
|
||||
// 在主线程显示 Toast
|
||||
Toast.makeText(context, errorResponse.error, Toast.LENGTH_LONG).show()
|
||||
onError(errorResponse.error)
|
||||
} else {
|
||||
Toast.makeText(context, "加入房间失败", Toast.LENGTH_SHORT).show()
|
||||
onError("加入房间失败")
|
||||
}
|
||||
} catch (parseException: Exception) {
|
||||
// 如果解析错误响应失败,显示默认错误信息
|
||||
Toast.makeText(context, "加入房间失败", Toast.LENGTH_SHORT).show()
|
||||
onError("加入房间失败")
|
||||
}
|
||||
}
|
||||
} catch (e: Exception) {
|
||||
Toast.makeText(context, "网络请求失败:${e.message}", Toast.LENGTH_SHORT).show()
|
||||
onError("网络请求失败:${e.message}")
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user