feat: show indicator when send chat message with attachment/mention etc (#5919)

* chore: adjust line height

* chore: send stream message

* chore: index file

* chore: clippy
This commit is contained in:
Nathan.fooo 2024-08-10 17:23:37 +08:00 committed by GitHub
parent 758c304a74
commit 7abe9f4661
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
64 changed files with 676 additions and 172 deletions

View File

@ -102,6 +102,7 @@ class _TypeOptionMenuItem<T> extends StatelessWidget {
value.text, value.text,
fontSize: 14.0, fontSize: 14.0,
maxLines: 2, maxLines: 2,
lineHeight: 1.0,
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
textAlign: TextAlign.center, textAlign: TextAlign.center,
), ),

View File

@ -176,6 +176,7 @@ class DatabaseViewSettingTile extends StatelessWidget {
return Row( return Row(
children: [ children: [
FlowyText( FlowyText(
lineHeight: 1.0,
databaseLayoutFromViewLayout(view.layout).layoutName, databaseLayoutFromViewLayout(view.layout).layoutName,
color: Theme.of(context).hintColor, color: Theme.of(context).hintColor,
), ),

View File

@ -29,6 +29,7 @@ class FontSetting extends StatelessWidget {
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
children: [ children: [
FlowyText( FlowyText(
lineHeight: 1.0,
name, name,
color: theme.colorScheme.onSurface, color: theme.colorScheme.onSurface,
), ),

View File

@ -38,6 +38,7 @@ class _LanguageSettingGroupState extends State<LanguageSettingGroup> {
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
children: [ children: [
FlowyText( FlowyText(
lineHeight: 1.0,
languageFromLocale(locale), languageFromLocale(locale),
color: theme.colorScheme.onSurface, color: theme.colorScheme.onSurface,
), ),

View File

@ -39,7 +39,8 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
final String chatId; final String chatId;
/// The last streaming message id /// The last streaming message id
String lastStreamMessageId = ''; String answerStreamMessageId = '';
String questionStreamMessageId = '';
/// Using a temporary map to associate the real message ID with the last streaming message ID. /// Using a temporary map to associate the real message ID with the last streaming message ID.
/// ///
@ -127,21 +128,11 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
); );
}, },
// streaming message // streaming message
streaming: (Message message) { finishAnswerStreaming: () {
final allMessages = _perminentMessages();
allMessages.insert(0, message);
emit(
state.copyWith(
messages: allMessages,
streamingState: const StreamingState.streaming(),
canSendMessage: false,
),
);
},
finishStreaming: () {
emit( emit(
state.copyWith( state.copyWith(
streamingState: const StreamingState.done(), streamingState: const StreamingState.done(),
acceptRelatedQuestion: true,
canSendMessage: canSendMessage:
state.sendingState == const SendMessageState.done(), state.sendingState == const SendMessageState.done(),
), ),
@ -162,9 +153,9 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
// If the streaming is not started, remove the message from the list // If the streaming is not started, remove the message from the list
if (!state.answerStream!.hasStarted) { if (!state.answerStream!.hasStarted) {
allMessages.removeWhere( allMessages.removeWhere(
(element) => element.id == lastStreamMessageId, (element) => element.id == answerStreamMessageId,
); );
lastStreamMessageId = ""; answerStreamMessageId = "";
} }
// when stop stream, we will set the answer stream to null. Which means the streaming // when stop stream, we will set the answer stream to null. Which means the streaming
@ -189,22 +180,26 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
), ),
); );
}, },
startAnswerStreaming: (Message message) {
final allMessages = _perminentMessages();
allMessages.insert(0, message);
emit(
state.copyWith(
messages: allMessages,
streamingState: const StreamingState.streaming(),
canSendMessage: false,
),
);
},
sendMessage: (String message, Map<String, dynamic>? metadata) async { sendMessage: (String message, Map<String, dynamic>? metadata) async {
unawaited(_startStreamingMessage(message, metadata, emit)); unawaited(_startStreamingMessage(message, metadata, emit));
final allMessages = _perminentMessages(); final allMessages = _perminentMessages();
// allMessages.insert(
// 0,
// CustomMessage(
// metadata: OnetimeShotType.sendingMessage.toMap(),
// author: User(id: state.userProfile.id.toString()),
// id: state.userProfile.id.toString(),
// ),
// );
emit( emit(
state.copyWith( state.copyWith(
lastSentMessage: null, lastSentMessage: null,
messages: allMessages, messages: allMessages,
relatedQuestions: [], relatedQuestions: [],
acceptRelatedQuestion: false,
sendingState: const SendMessageState.sending(), sendingState: const SendMessageState.sending(),
canSendMessage: false, canSendMessage: false,
), ),
@ -257,10 +252,17 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
chatMessageCallback: (pb) { chatMessageCallback: (pb) {
if (!isClosed) { if (!isClosed) {
// 3 mean message response from AI // 3 mean message response from AI
if (pb.authorType == 3 && lastStreamMessageId.isNotEmpty) { if (pb.authorType == 3 && answerStreamMessageId.isNotEmpty) {
temporaryMessageIDMap[pb.messageId.toString()] = temporaryMessageIDMap[pb.messageId.toString()] =
lastStreamMessageId; answerStreamMessageId;
lastStreamMessageId = ""; answerStreamMessageId = "";
}
// 1 mean message response from User
if (pb.authorType == 1 && questionStreamMessageId.isNotEmpty) {
temporaryMessageIDMap[pb.messageId.toString()] =
questionStreamMessageId;
questionStreamMessageId = "";
} }
final message = _createTextMessage(pb); final message = _createTextMessage(pb);
@ -270,7 +272,7 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
chatErrorMessageCallback: (err) { chatErrorMessageCallback: (err) {
if (!isClosed) { if (!isClosed) {
Log.error("chat error: ${err.errorMessage}"); Log.error("chat error: ${err.errorMessage}");
add(const ChatEvent.finishStreaming()); add(const ChatEvent.finishAnswerStreaming());
} }
}, },
latestMessageCallback: (list) { latestMessageCallback: (list) {
@ -287,7 +289,7 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
}, },
finishStreamingCallback: () { finishStreamingCallback: () {
if (!isClosed) { if (!isClosed) {
add(const ChatEvent.finishStreaming()); add(const ChatEvent.finishAnswerStreaming());
// The answer strema will bet set to null after the streaming is finished or canceled. // The answer strema will bet set to null after the streaming is finished or canceled.
// so if the answer stream is null, we will not get related question. // so if the answer stream is null, we will not get related question.
if (state.lastSentMessage != null && state.answerStream != null) { if (state.lastSentMessage != null && state.answerStream != null) {
@ -300,7 +302,9 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
if (!isClosed) { if (!isClosed) {
result.fold( result.fold(
(list) { (list) {
add(ChatEvent.didReceiveRelatedQuestion(list.items)); if (state.acceptRelatedQuestion) {
add(ChatEvent.didReceiveRelatedQuestion(list.items));
}
}, },
(err) { (err) {
Log.error("Failed to get related question: $err"); Log.error("Failed to get related question: $err");
@ -358,16 +362,21 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
} }
final answerStream = AnswerStream(); final answerStream = AnswerStream();
final questionStream = QuestionStream();
add(ChatEvent.didUpdateAnswerStream(answerStream)); add(ChatEvent.didUpdateAnswerStream(answerStream));
final payload = StreamChatPayloadPB( final payload = StreamChatPayloadPB(
chatId: state.view.id, chatId: state.view.id,
message: message, message: message,
messageType: ChatMessageTypePB.User, messageType: ChatMessageTypePB.User,
textStreamPort: Int64(answerStream.nativePort), questionStreamPort: Int64(questionStream.nativePort),
answerStreamPort: Int64(answerStream.nativePort),
metadata: await metadataPBFromMetadata(metadata), metadata: await metadataPBFromMetadata(metadata),
); );
final questionStreamMessage = _createQuestionStreamMessage(questionStream);
add(ChatEvent.receveMessage(questionStreamMessage));
// Stream message to the server // Stream message to the server
final result = await AIEventStreamMessage(payload).send(); final result = await AIEventStreamMessage(payload).send();
result.fold( result.fold(
@ -375,13 +384,12 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
if (!isClosed) { if (!isClosed) {
add(ChatEvent.finishSending(question)); add(ChatEvent.finishSending(question));
final questionMessageId = question.messageId; // final message = _createTextMessage(question);
final message = _createTextMessage(question); // add(ChatEvent.receveMessage(message));
add(ChatEvent.receveMessage(message));
final streamAnswer = final streamAnswer =
_createStreamMessage(answerStream, questionMessageId); _createAnswerStreamMessage(answerStream, question.messageId);
add(ChatEvent.streaming(streamAnswer)); add(ChatEvent.startAnswerStreaming(streamAnswer));
} }
}, },
(err) { (err) {
@ -404,9 +412,12 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
); );
} }
Message _createStreamMessage(AnswerStream stream, Int64 questionMessageId) { Message _createAnswerStreamMessage(
AnswerStream stream,
Int64 questionMessageId,
) {
final streamMessageId = (questionMessageId + 1).toString(); final streamMessageId = (questionMessageId + 1).toString();
lastStreamMessageId = streamMessageId; answerStreamMessageId = streamMessageId;
return TextMessage( return TextMessage(
author: User(id: "streamId:${nanoid()}"), author: User(id: "streamId:${nanoid()}"),
@ -421,6 +432,20 @@ class ChatBloc extends Bloc<ChatEvent, ChatState> {
); );
} }
Message _createQuestionStreamMessage(QuestionStream stream) {
questionStreamMessageId = nanoid();
return TextMessage(
author: User(id: state.userProfile.id.toString()),
metadata: {
"$QuestionStream": stream,
"chatId": chatId,
},
id: questionStreamMessageId,
createdAt: DateTime.now().millisecondsSinceEpoch,
text: '',
);
}
Message _createTextMessage(ChatMessagePB message) { Message _createTextMessage(ChatMessagePB message) {
String messageId = message.messageId.toString(); String messageId = message.messageId.toString();
@ -454,9 +479,10 @@ class ChatEvent with _$ChatEvent {
_FinishSendMessage; _FinishSendMessage;
// receive message // receive message
const factory ChatEvent.streaming(Message message) = _StreamingMessage; const factory ChatEvent.startAnswerStreaming(Message message) =
_StartAnswerStreaming;
const factory ChatEvent.receveMessage(Message message) = _ReceiveMessage; const factory ChatEvent.receveMessage(Message message) = _ReceiveMessage;
const factory ChatEvent.finishStreaming() = _FinishStreamingMessage; const factory ChatEvent.finishAnswerStreaming() = _FinishAnswerStreaming;
// loading messages // loading messages
const factory ChatEvent.startLoadingPrevMessage() = _StartLoadPrevMessage; const factory ChatEvent.startLoadingPrevMessage() = _StartLoadPrevMessage;
@ -499,6 +525,7 @@ class ChatState with _$ChatState {
required bool hasMorePrevMessage, required bool hasMorePrevMessage,
// The related questions that are received after the user message is sent. // The related questions that are received after the user message is sent.
required List<RelatedQuestionPB> relatedQuestions, required List<RelatedQuestionPB> relatedQuestions,
@Default(false) bool acceptRelatedQuestion,
// The last user message that is sent to the server. // The last user message that is sent to the server.
ChatMessagePB? lastSentMessage, ChatMessagePB? lastSentMessage,
AnswerStream? answerStream, AnswerStream? answerStream,

View File

@ -92,3 +92,102 @@ class AnswerStream {
} }
} }
} }
class QuestionStream {
QuestionStream() {
_port.handler = _controller.add;
_subscription = _controller.stream.listen(
(event) {
if (event.startsWith("data:")) {
_hasStarted = true;
final newText = event.substring(5);
_text += newText;
if (_onData != null) {
_onData!(_text);
}
} else if (event.startsWith("message_id:")) {
final messageId = event.substring(11);
_onMessageId?.call(messageId);
} else if (event.startsWith("start_index_file:")) {
final indexName = event.substring(17);
_onFileIndexStart?.call(indexName);
} else if (event.startsWith("end_index_file:")) {
final indexName = event.substring(10);
_onFileIndexEnd?.call(indexName);
} else if (event.startsWith("index_file_error:")) {
final indexName = event.substring(16);
_onFileIndexError?.call(indexName);
} else if (event.startsWith("index_start:")) {
_onIndexStart?.call();
} else if (event.startsWith("index_end:")) {
_onIndexEnd?.call();
} else if (event.startsWith("done:")) {
_onDone?.call();
} else if (event.startsWith("error:")) {
_error = event.substring(5);
if (_onError != null) {
_onError!(_error!);
}
}
},
onError: (error) {
if (_onError != null) {
_onError!(error.toString());
}
},
);
}
final RawReceivePort _port = RawReceivePort();
final StreamController<String> _controller = StreamController.broadcast();
late StreamSubscription<String> _subscription;
bool _hasStarted = false;
String? _error;
String _text = "";
// Callbacks
void Function(String text)? _onData;
void Function(String error)? _onError;
void Function(String messageId)? _onMessageId;
void Function(String indexName)? _onFileIndexStart;
void Function(String indexName)? _onFileIndexEnd;
void Function(String indexName)? _onFileIndexError;
void Function()? _onIndexStart;
void Function()? _onIndexEnd;
void Function()? _onDone;
int get nativePort => _port.sendPort.nativePort;
bool get hasStarted => _hasStarted;
String? get error => _error;
String get text => _text;
Future<void> dispose() async {
await _controller.close();
await _subscription.cancel();
_port.close();
}
void listen({
void Function(String text)? onData,
void Function(String error)? onError,
void Function(String messageId)? onMessageId,
void Function(String indexName)? onFileIndexStart,
void Function(String indexName)? onFileIndexEnd,
void Function(String indexName)? onFileIndexFail,
void Function()? onIndexStart,
void Function()? onIndexEnd,
void Function()? onDone,
}) {
_onData = onData;
_onError = onError;
_onMessageId = onMessageId;
_onFileIndexStart = onFileIndexStart;
_onFileIndexEnd = onFileIndexEnd;
_onFileIndexError = onFileIndexFail;
_onIndexStart = onIndexStart;
_onIndexEnd = onIndexEnd;
_onDone = onDone;
}
}

View File

@ -1,27 +1,91 @@
import 'package:appflowy/plugins/ai_chat/application/chat_entity.dart'; import 'package:appflowy/plugins/ai_chat/application/chat_message_stream.dart';
import 'package:appflowy_backend/log.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:flutter_chat_types/flutter_chat_types.dart';
import 'package:freezed_annotation/freezed_annotation.dart'; import 'package:freezed_annotation/freezed_annotation.dart';
import 'chat_message_service.dart';
part 'chat_user_message_bloc.freezed.dart'; part 'chat_user_message_bloc.freezed.dart';
class ChatUserMessageBloc class ChatUserMessageBloc
extends Bloc<ChatUserMessageEvent, ChatUserMessageState> { extends Bloc<ChatUserMessageEvent, ChatUserMessageState> {
ChatUserMessageBloc({ ChatUserMessageBloc({
required Message message, required dynamic message,
required String? metadata,
}) : super( }) : super(
ChatUserMessageState.initial( ChatUserMessageState.initial(
message, message,
chatFilesFromMetadataString(metadata),
), ),
) { ) {
on<ChatUserMessageEvent>( on<ChatUserMessageEvent>(
(event, emit) async { (event, emit) async {
event.when( event.when(
initial: () {}, initial: () {
if (state.stream != null) {
add(ChatUserMessageEvent.updateText(state.stream!.text));
}
state.stream?.listen(
onData: (text) {
if (!isClosed) {
add(ChatUserMessageEvent.updateText(text));
}
},
onMessageId: (messageId) {
if (!isClosed) {
add(ChatUserMessageEvent.updateMessageId(messageId));
}
},
onError: (error) {
if (!isClosed) {
add(ChatUserMessageEvent.receiveError(error.toString()));
}
},
onFileIndexStart: (indexName) {
Log.debug("index start: $indexName");
},
onFileIndexEnd: (indexName) {
Log.info("index end: $indexName");
},
onFileIndexFail: (indexName) {
Log.debug("index fail: $indexName");
},
onIndexStart: () {
if (!isClosed) {
add(
const ChatUserMessageEvent.updateQuestionState(
QuestionMessageState.indexStart(),
),
);
}
},
onIndexEnd: () {
if (!isClosed) {
add(
const ChatUserMessageEvent.updateQuestionState(
QuestionMessageState.indexEnd(),
),
);
}
},
onDone: () {
if (!isClosed) {
add(
const ChatUserMessageEvent.updateQuestionState(
QuestionMessageState.finish(),
),
);
}
},
);
},
updateText: (String text) {
emit(state.copyWith(text: text));
},
updateMessageId: (String messageId) {
emit(state.copyWith(messageId: messageId));
},
receiveError: (String error) {},
updateQuestionState: (QuestionMessageState newState) {
emit(state.copyWith(messageState: newState));
},
); );
}, },
); );
@ -31,18 +95,47 @@ class ChatUserMessageBloc
@freezed @freezed
class ChatUserMessageEvent with _$ChatUserMessageEvent { class ChatUserMessageEvent with _$ChatUserMessageEvent {
const factory ChatUserMessageEvent.initial() = Initial; const factory ChatUserMessageEvent.initial() = Initial;
const factory ChatUserMessageEvent.updateText(String text) = _UpdateText;
const factory ChatUserMessageEvent.updateQuestionState(
QuestionMessageState newState,
) = _UpdateQuestionState;
const factory ChatUserMessageEvent.updateMessageId(String messageId) =
_UpdateMessageId;
const factory ChatUserMessageEvent.receiveError(String error) = _ReceiveError;
} }
@freezed @freezed
class ChatUserMessageState with _$ChatUserMessageState { class ChatUserMessageState with _$ChatUserMessageState {
const factory ChatUserMessageState({ const factory ChatUserMessageState({
required Message message, required String text,
required List<ChatFile> files, QuestionStream? stream,
String? messageId,
@Default(QuestionMessageState.finish()) QuestionMessageState messageState,
}) = _ChatUserMessageState; }) = _ChatUserMessageState;
factory ChatUserMessageState.initial( factory ChatUserMessageState.initial(
Message message, dynamic message,
List<ChatFile> files,
) => ) =>
ChatUserMessageState(message: message, files: files); ChatUserMessageState(
text: message is String ? message : "",
stream: message is QuestionStream ? message : null,
);
}
@freezed
class QuestionMessageState with _$QuestionMessageState {
const factory QuestionMessageState.indexFileStart(String fileName) =
_IndexFileStart;
const factory QuestionMessageState.indexFileEnd(String fileName) =
_IndexFileEnd;
const factory QuestionMessageState.indexFileFail(String fileName) =
_IndexFileFail;
const factory QuestionMessageState.indexStart() = _IndexStart;
const factory QuestionMessageState.indexEnd() = _IndexEnd;
const factory QuestionMessageState.finish() = _Finish;
}
extension QuestionMessageStateX on QuestionMessageState {
bool get isFinish => this is _Finish;
} }

View File

@ -0,0 +1,48 @@
import 'package:appflowy/plugins/ai_chat/application/chat_entity.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:flutter_chat_types/flutter_chat_types.dart';
import 'package:freezed_annotation/freezed_annotation.dart';
import 'chat_message_service.dart';
part 'chat_user_message_bubble_bloc.freezed.dart';
class ChatUserMessageBubbleBloc
extends Bloc<ChatUserMessageBubbleEvent, ChatUserMessageBubbleState> {
ChatUserMessageBubbleBloc({
required Message message,
required String? metadata,
}) : super(
ChatUserMessageBubbleState.initial(
message,
chatFilesFromMetadataString(metadata),
),
) {
on<ChatUserMessageBubbleEvent>(
(event, emit) async {
event.when(
initial: () {},
);
},
);
}
}
@freezed
class ChatUserMessageBubbleEvent with _$ChatUserMessageBubbleEvent {
const factory ChatUserMessageBubbleEvent.initial() = Initial;
}
@freezed
class ChatUserMessageBubbleState with _$ChatUserMessageBubbleState {
const factory ChatUserMessageBubbleState({
required Message message,
required List<ChatFile> files,
}) = _ChatUserMessageBubbleState;
factory ChatUserMessageBubbleState.initial(
Message message,
List<ChatFile> files,
) =>
ChatUserMessageBubbleState(message: message, files: files);
}

View File

@ -319,21 +319,22 @@ class _ChatContentPageState extends State<_ChatContentPage> {
Widget _buildTextMessage(BuildContext context, TextMessage message) { Widget _buildTextMessage(BuildContext context, TextMessage message) {
if (message.author.id == _user.id) { if (message.author.id == _user.id) {
final stream = message.metadata?["$QuestionStream"];
final metadata = message.metadata?[messageMetadataKey] as String?; final metadata = message.metadata?[messageMetadataKey] as String?;
return ChatUserTextMessageWidget( return ChatUserMessageWidget(
key: ValueKey(message.id),
user: message.author, user: message.author,
messageUserId: message.id, message: stream is QuestionStream ? stream : message.text,
message: message,
metadata: metadata, metadata: metadata,
); );
} else { } else {
final stream = message.metadata?["$AnswerStream"]; final stream = message.metadata?["$AnswerStream"];
final questionId = message.metadata?[messageQuestionIdKey]; final questionId = message.metadata?[messageQuestionIdKey];
final metadata = message.metadata?[messageMetadataKey] as String?; final metadata = message.metadata?[messageMetadataKey] as String?;
return ChatAITextMessageWidget( return ChatAIMessageWidget(
user: message.author, user: message.author,
messageUserId: message.id, messageUserId: message.id,
text: stream is AnswerStream ? stream : message.text, message: stream is AnswerStream ? stream : message.text,
key: ValueKey(message.id), key: ValueKey(message.id),
questionId: questionId, questionId: questionId,
chatId: widget.view.id, chatId: widget.view.id,

View File

@ -156,6 +156,7 @@ class _ActionItem extends StatelessWidget {
margin: const EdgeInsets.symmetric(horizontal: 6), margin: const EdgeInsets.symmetric(horizontal: 6),
iconPadding: 10.0, iconPadding: 10.0,
text: FlowyText.regular( text: FlowyText.regular(
lineHeight: 1.0,
item.title, item.title,
), ),
onTap: onTap, onTap: onTap,

View File

@ -49,6 +49,8 @@ class AIMessageMetadata extends StatelessWidget {
child: FlowyText( child: FlowyText(
m.name, m.name,
fontSize: 14, fontSize: 14,
lineHeight: 1.0,
overflow: TextOverflow.ellipsis,
), ),
), ),
onTap: () => onSelectedMetadata(m), onTap: () => onSelectedMetadata(m),

View File

@ -14,12 +14,12 @@ import 'package:flutter_chat_types/flutter_chat_types.dart';
import 'ai_metadata.dart'; import 'ai_metadata.dart';
class ChatAITextMessageWidget extends StatelessWidget { class ChatAIMessageWidget extends StatelessWidget {
const ChatAITextMessageWidget({ const ChatAIMessageWidget({
super.key, super.key,
required this.user, required this.user,
required this.messageUserId, required this.messageUserId,
required this.text, required this.message,
required this.questionId, required this.questionId,
required this.chatId, required this.chatId,
required this.metadata, required this.metadata,
@ -28,7 +28,9 @@ class ChatAITextMessageWidget extends StatelessWidget {
final User user; final User user;
final String messageUserId; final String messageUserId;
final dynamic text;
/// message can be a striing or Stream<String>
final dynamic message;
final Int64? questionId; final Int64? questionId;
final String chatId; final String chatId;
final String? metadata; final String? metadata;
@ -38,7 +40,7 @@ class ChatAITextMessageWidget extends StatelessWidget {
Widget build(BuildContext context) { Widget build(BuildContext context) {
return BlocProvider( return BlocProvider(
create: (context) => ChatAIMessageBloc( create: (context) => ChatAIMessageBloc(
message: text, message: message,
metadata: metadata, metadata: metadata,
chatId: chatId, chatId: chatId,
questionId: questionId, questionId: questionId,
@ -59,7 +61,6 @@ class ChatAITextMessageWidget extends StatelessWidget {
return FlowyText( return FlowyText(
LocaleKeys.sideBar_askOwnerToUpgradeToAIMax.tr(), LocaleKeys.sideBar_askOwnerToUpgradeToAIMax.tr(),
maxLines: 10, maxLines: 10,
lineHeight: 1.5,
); );
}, },
ready: () { ready: () {

View File

@ -1,6 +1,6 @@
import 'package:appflowy/plugins/ai_chat/application/chat_entity.dart'; import 'package:appflowy/plugins/ai_chat/application/chat_entity.dart';
import 'package:appflowy/plugins/ai_chat/application/chat_member_bloc.dart'; import 'package:appflowy/plugins/ai_chat/application/chat_member_bloc.dart';
import 'package:appflowy/plugins/ai_chat/application/chat_user_message_bloc.dart'; import 'package:appflowy/plugins/ai_chat/application/chat_user_message_bubble_bloc.dart';
import 'package:appflowy/plugins/ai_chat/presentation/chat_avatar.dart'; import 'package:appflowy/plugins/ai_chat/presentation/chat_avatar.dart';
import 'package:flowy_infra_ui/style_widget/text.dart'; import 'package:flowy_infra_ui/style_widget/text.dart';
import 'package:flowy_infra_ui/widget/spacing.dart'; import 'package:flowy_infra_ui/widget/spacing.dart';
@ -32,11 +32,11 @@ class ChatUserMessageBubble extends StatelessWidget {
final metadata = message.metadata?[messageMetadataKey] as String?; final metadata = message.metadata?[messageMetadataKey] as String?;
return BlocProvider( return BlocProvider(
create: (context) => ChatUserMessageBloc( create: (context) => ChatUserMessageBubbleBloc(
message: message, message: message,
metadata: metadata, metadata: metadata,
), ),
child: BlocBuilder<ChatUserMessageBloc, ChatUserMessageState>( child: BlocBuilder<ChatUserMessageBubbleBloc, ChatUserMessageBubbleState>(
builder: (context, state) { builder: (context, state) {
return Column( return Column(
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,

View File

@ -1,26 +1,50 @@
import 'package:appflowy/plugins/ai_chat/application/chat_user_message_bloc.dart';
import 'package:flowy_infra/theme_extension.dart'; import 'package:flowy_infra/theme_extension.dart';
import 'package:flowy_infra_ui/style_widget/text.dart'; import 'package:flowy_infra_ui/flowy_infra_ui.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:flutter_chat_types/flutter_chat_types.dart'; import 'package:flutter_chat_types/flutter_chat_types.dart';
class ChatUserTextMessageWidget extends StatelessWidget { class ChatUserMessageWidget extends StatelessWidget {
const ChatUserTextMessageWidget({ const ChatUserMessageWidget({
super.key, super.key,
required this.user, required this.user,
required this.messageUserId,
required this.message, required this.message,
required this.metadata, required this.metadata,
}); });
final User user; final User user;
final String messageUserId; final dynamic message;
final TextMessage message;
final String? metadata; final String? metadata;
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return TextMessageText( return BlocProvider(
text: message.text, create: (context) => ChatUserMessageBloc(message: message)
..add(const ChatUserMessageEvent.initial()),
child: BlocBuilder<ChatUserMessageBloc, ChatUserMessageState>(
builder: (context, state) {
final List<Widget> children = [];
children.add(
Flexible(
child: TextMessageText(
text: state.text,
),
),
);
if (!state.messageState.isFinish) {
children.add(const HSpace(6));
children.add(const CircularProgressIndicator.adaptive());
}
return Row(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.end,
children: children,
);
},
),
); );
} }
} }

View File

@ -237,6 +237,7 @@ class _BoardColumnHeaderState extends State<BoardColumnHeader> {
leftIcon: FlowySvg(action.icon), leftIcon: FlowySvg(action.icon),
text: FlowyText.medium( text: FlowyText.medium(
action.text, action.text,
lineHeight: 1.0,
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
), ),
onTap: () { onTap: () {

View File

@ -173,7 +173,10 @@ class LayoutDateField extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(fieldInfo.name), text: FlowyText.medium(
fieldInfo.name,
lineHeight: 1.0,
),
onTap: () { onTap: () {
onUpdated(fieldInfo.id); onUpdated(fieldInfo.id);
popoverMutex.close(); popoverMutex.close();
@ -206,6 +209,7 @@ class LayoutDateField extends StatelessWidget {
child: FlowyButton( child: FlowyButton(
margin: const EdgeInsets.symmetric(vertical: 2.0, horizontal: 10.0), margin: const EdgeInsets.symmetric(vertical: 2.0, horizontal: 10.0),
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
LocaleKeys.calendar_settings_layoutDateField.tr(), LocaleKeys.calendar_settings_layoutDateField.tr(),
), ),
), ),
@ -307,6 +311,7 @@ class FirstDayOfWeek extends StatelessWidget {
child: FlowyButton( child: FlowyButton(
margin: const EdgeInsets.symmetric(vertical: 2.0, horizontal: 10.0), margin: const EdgeInsets.symmetric(vertical: 2.0, horizontal: 10.0),
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
LocaleKeys.calendar_settings_firstDayOfWeek.tr(), LocaleKeys.calendar_settings_firstDayOfWeek.tr(),
), ),
), ),
@ -367,7 +372,10 @@ class StartFromButton extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(title), text: FlowyText.medium(
title,
lineHeight: 1.0,
),
onTap: () => onTap(dayIndex), onTap: () => onTap(dayIndex),
rightIcon: isSelected ? const FlowySvg(FlowySvgs.check_s) : null, rightIcon: isSelected ? const FlowySvg(FlowySvgs.check_s) : null,
), ),

View File

@ -167,6 +167,7 @@ class _CalculateCellState extends State<CalculateCell> {
mainAxisAlignment: MainAxisAlignment.end, mainAxisAlignment: MainAxisAlignment.end,
children: [ children: [
FlowyText( FlowyText(
lineHeight: 1.0,
widget.calculation!.calculationType.shortLabel widget.calculation!.calculationType.shortLabel
.toUpperCase(), .toUpperCase(),
color: Theme.of(context).hintColor, color: Theme.of(context).hintColor,
@ -175,6 +176,7 @@ class _CalculateCellState extends State<CalculateCell> {
if (widget.calculation!.value.isNotEmpty) ...[ if (widget.calculation!.value.isNotEmpty) ...[
const HSpace(8), const HSpace(8),
FlowyText( FlowyText(
lineHeight: 1.0,
calculateValue, calculateValue,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,

View File

@ -22,7 +22,11 @@ class CalculationTypeItem extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(type.label, overflow: TextOverflow.ellipsis), text: FlowyText.medium(
type.label,
overflow: TextOverflow.ellipsis,
lineHeight: 1.0,
),
onTap: () { onTap: () {
onTap(); onTap();
PopoverContainer.of(context).close(); PopoverContainer.of(context).close();

View File

@ -39,6 +39,7 @@ class ChoiceChipButton extends StatelessWidget {
decoration: decoration, decoration: decoration,
useIntrinsicWidth: true, useIntrinsicWidth: true,
text: FlowyText( text: FlowyText(
lineHeight: 1.0,
filterInfo.fieldInfo.field.name, filterInfo.fieldInfo.field.name,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
), ),

View File

@ -31,6 +31,7 @@ class ConditionButton extends StatelessWidget {
child: FlowyButton( child: FlowyButton(
useIntrinsicWidth: true, useIntrinsicWidth: true,
text: FlowyText( text: FlowyText(
lineHeight: 1.0,
conditionName, conditionName,
fontSize: 10, fontSize: 10,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,

View File

@ -163,6 +163,7 @@ class GridFilterPropertyCell extends StatelessWidget {
return FlowyButton( return FlowyButton(
hoverColor: AFThemeExtension.of(context).lightGreyHover, hoverColor: AFThemeExtension.of(context).lightGreyHover,
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
fieldInfo.field.name, fieldInfo.field.name,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
), ),

View File

@ -87,6 +87,7 @@ class _AddFilterButtonState extends State<AddFilterButton> {
height: 28, height: 28,
child: FlowyButton( child: FlowyButton(
text: FlowyText( text: FlowyText(
lineHeight: 1.0,
LocaleKeys.grid_settings_addFilter.tr(), LocaleKeys.grid_settings_addFilter.tr(),
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
), ),

View File

@ -23,6 +23,7 @@ class GridAddRowButton extends StatelessWidget {
), ),
), ),
text: FlowyText( text: FlowyText(
lineHeight: 1.0,
LocaleKeys.grid_row_newRow.tr(), LocaleKeys.grid_row_newRow.tr(),
color: Theme.of(context).hintColor, color: Theme.of(context).hintColor,
), ),

View File

@ -229,6 +229,7 @@ class FieldCellButton extends StatelessWidget {
radius: radius, radius: radius,
text: FlowyText.medium( text: FlowyText.medium(
field.name, field.name,
lineHeight: 1.0,
maxLines: maxLines, maxLines: maxLines,
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,

View File

@ -195,6 +195,7 @@ class _CreateFieldButtonState extends State<CreateFieldButton> {
margin: GridSize.cellContentInsets, margin: GridSize.cellContentInsets,
radius: BorderRadius.zero, radius: BorderRadius.zero,
text: FlowyText( text: FlowyText(
lineHeight: 1.0,
LocaleKeys.grid_field_newProperty.tr(), LocaleKeys.grid_field_newProperty.tr(),
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
), ),

View File

@ -51,7 +51,11 @@ class RowActionMenu extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(action.text, overflow: TextOverflow.ellipsis), text: FlowyText.medium(
action.text,
overflow: TextOverflow.ellipsis,
lineHeight: 1.0,
),
onTap: () { onTap: () {
if (action == RowAction.delete) { if (action == RowAction.delete) {
NavigatorOkCancelDialog( NavigatorOkCancelDialog(

View File

@ -120,6 +120,7 @@ class GridSortPropertyCell extends StatelessWidget {
hoverColor: AFThemeExtension.of(context).lightGreyHover, hoverColor: AFThemeExtension.of(context).lightGreyHover,
text: FlowyText.medium( text: FlowyText.medium(
fieldInfo.name, fieldInfo.name,
lineHeight: 1.0,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
), ),
onTap: onTap, onTap: onTap,

View File

@ -34,6 +34,7 @@ class SortChoiceButton extends StatelessWidget {
useIntrinsicWidth: true, useIntrinsicWidth: true,
text: FlowyText( text: FlowyText(
text, text,
lineHeight: 1.0,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
), ),

View File

@ -222,6 +222,7 @@ class TabBarItemButton extends StatelessWidget {
), ),
text: FlowyText( text: FlowyText(
view.name, view.name,
lineHeight: 1.0,
fontSize: FontSizes.s11, fontSize: FontSizes.s11,
textAlign: TextAlign.center, textAlign: TextAlign.center,
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,

View File

@ -158,6 +158,7 @@ class _EditFieldButton extends StatelessWidget {
child: FlowyButton( child: FlowyButton(
leftIcon: const FlowySvg(FlowySvgs.edit_s), leftIcon: const FlowySvg(FlowySvgs.edit_s),
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
LocaleKeys.grid_field_editProperty.tr(), LocaleKeys.grid_field_editProperty.tr(),
), ),
onTap: onTap, onTap: onTap,
@ -193,6 +194,7 @@ class FieldActionCell extends StatelessWidget {
disable: !enable, disable: !enable,
text: FlowyText.medium( text: FlowyText.medium(
action.title(fieldInfo), action.title(fieldInfo),
lineHeight: 1.0,
color: enable ? null : Theme.of(context).disabledColor, color: enable ? null : Theme.of(context).disabledColor,
), ),
onHover: (_) => popoverMutex?.close(), onHover: (_) => popoverMutex?.close(),
@ -613,6 +615,7 @@ class _SwitchFieldButtonState extends State<SwitchFieldButton> {
}, },
text: FlowyText.medium( text: FlowyText.medium(
state.field.fieldType.i18n, state.field.fieldType.i18n,
lineHeight: 1.0,
color: isPrimary ? Theme.of(context).disabledColor : null, color: isPrimary ? Theme.of(context).disabledColor : null,
), ),
leftIcon: FlowySvg( leftIcon: FlowySvg(

View File

@ -75,9 +75,7 @@ class FieldTypeCell extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium( text: FlowyText.medium(fieldType.i18n, lineHeight: 1.0),
fieldType.i18n,
),
onTap: () => onSelectField(fieldType), onTap: () => onSelectField(fieldType),
leftIcon: FlowySvg( leftIcon: FlowySvg(
fieldType.svgData, fieldType.svgData,

View File

@ -23,7 +23,10 @@ class DateFormatButton extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(LocaleKeys.grid_field_dateFormat.tr()), text: FlowyText.medium(
LocaleKeys.grid_field_dateFormat.tr(),
lineHeight: 1.0,
),
onTap: onTap, onTap: onTap,
onHover: onHover, onHover: onHover,
rightIcon: const FlowySvg(FlowySvgs.more_s), rightIcon: const FlowySvg(FlowySvgs.more_s),
@ -47,7 +50,10 @@ class TimeFormatButton extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(LocaleKeys.grid_field_timeFormat.tr()), text: FlowyText.medium(
LocaleKeys.grid_field_timeFormat.tr(),
lineHeight: 1.0,
),
onTap: onTap, onTap: onTap,
onHover: onHover, onHover: onHover,
rightIcon: const FlowySvg(FlowySvgs.more_s), rightIcon: const FlowySvg(FlowySvgs.more_s),
@ -114,7 +120,10 @@ class DateFormatCell extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(dateFormat.title()), text: FlowyText.medium(
dateFormat.title(),
lineHeight: 1.0,
),
rightIcon: checkmark, rightIcon: checkmark,
onTap: () => onSelected(dateFormat), onTap: () => onSelected(dateFormat),
), ),
@ -199,7 +208,10 @@ class TimeFormatCell extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(timeFormat.title()), text: FlowyText.medium(
timeFormat.title(),
lineHeight: 1.0,
),
rightIcon: checkmark, rightIcon: checkmark,
onTap: () => onSelected(timeFormat), onTap: () => onSelected(timeFormat),
), ),

View File

@ -32,6 +32,7 @@ class NumberTypeOptionEditorFactory implements TypeOptionEditorFactory {
child: FlowyButton( child: FlowyButton(
rightIcon: const FlowySvg(FlowySvgs.more_s), rightIcon: const FlowySvg(FlowySvgs.more_s),
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
typeOption.format.title(), typeOption.format.title(),
), ),
), ),
@ -167,7 +168,10 @@ class NumberFormatCell extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(format.title()), text: FlowyText.medium(
format.title(),
lineHeight: 1.0,
),
onTap: () => onSelected(format), onTap: () => onSelected(format),
rightIcon: checkmark, rightIcon: checkmark,
), ),

View File

@ -61,6 +61,7 @@ class RelationTypeOptionEditorFactory implements TypeOptionEditorFactory {
(meta) => meta.databaseId == typeOption.databaseId, (meta) => meta.databaseId == typeOption.databaseId,
); );
return FlowyText( return FlowyText(
lineHeight: 1.0,
databaseMeta == null databaseMeta == null
? LocaleKeys ? LocaleKeys
.grid_relation_relatedDatabasePlaceholder .grid_relation_relatedDatabasePlaceholder
@ -134,6 +135,7 @@ class _DatabaseList extends StatelessWidget {
child: FlowyButton( child: FlowyButton(
onTap: () => onSelectDatabase(meta.databaseId), onTap: () => onSelectDatabase(meta.databaseId),
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
meta.databaseName, meta.databaseName,
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
), ),

View File

@ -181,6 +181,7 @@ class _AddOptionButton extends StatelessWidget {
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
LocaleKeys.grid_field_addSelectOption.tr(), LocaleKeys.grid_field_addSelectOption.tr(),
), ),
onTap: () { onTap: () {

View File

@ -107,6 +107,7 @@ class _DeleteTag extends StatelessWidget {
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
LocaleKeys.grid_selectOption_deleteTag.tr(), LocaleKeys.grid_selectOption_deleteTag.tr(),
), ),
leftIcon: const FlowySvg(FlowySvgs.delete_s), leftIcon: const FlowySvg(FlowySvgs.delete_s),
@ -230,6 +231,7 @@ class _SelectOptionColorCell extends StatelessWidget {
child: FlowyButton( child: FlowyButton(
hoverColor: AFThemeExtension.of(context).lightGreyHover, hoverColor: AFThemeExtension.of(context).lightGreyHover,
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
color.colorName(), color.colorName(),
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
), ),

View File

@ -97,7 +97,12 @@ class SelectLanguageButton extends StatelessWidget {
Widget build(BuildContext context) { Widget build(BuildContext context) {
return SizedBox( return SizedBox(
height: 30, height: 30,
child: FlowyButton(text: FlowyText(language)), child: FlowyButton(
text: FlowyText(
language,
lineHeight: 1.0,
),
),
); );
} }
} }
@ -159,7 +164,10 @@ class LanguageCell extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(languageTypeToLanguage(languageType)), text: FlowyText.medium(
languageTypeToLanguage(languageType),
lineHeight: 1.0,
),
rightIcon: checkmark, rightIcon: checkmark,
onTap: () => onSelected(languageType), onTap: () => onSelected(languageType),
), ),

View File

@ -193,6 +193,7 @@ class _GridGroupCell extends StatelessWidget {
text: FlowyText.medium( text: FlowyText.medium(
name, name,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
lineHeight: 1.0,
), ),
leftIcon: icon != null leftIcon: icon != null
? FlowySvg( ? FlowySvg(

View File

@ -50,7 +50,10 @@ class RowDetailPageDeleteButton extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.regular(LocaleKeys.grid_row_delete.tr()), text: FlowyText.regular(
LocaleKeys.grid_row_delete.tr(),
lineHeight: 1.0,
),
leftIcon: const FlowySvg(FlowySvgs.trash_m), leftIcon: const FlowySvg(FlowySvgs.trash_m),
onTap: () { onTap: () {
RowBackendService.deleteRows(viewId, [rowId]); RowBackendService.deleteRows(viewId, [rowId]);
@ -76,7 +79,10 @@ class RowDetailPageDuplicateButton extends StatelessWidget {
return SizedBox( return SizedBox(
height: GridSize.popoverItemHeight, height: GridSize.popoverItemHeight,
child: FlowyButton( child: FlowyButton(
text: FlowyText.regular(LocaleKeys.grid_row_duplicate.tr()), text: FlowyText.regular(
LocaleKeys.grid_row_duplicate.tr(),
lineHeight: 1.0,
),
leftIcon: const FlowySvg(FlowySvgs.copy_s), leftIcon: const FlowySvg(FlowySvgs.copy_s),
onTap: () { onTap: () {
RowBackendService.duplicateRow(viewId, rowId); RowBackendService.duplicateRow(viewId, rowId);

View File

@ -220,6 +220,7 @@ class AddEmojiButton extends StatelessWidget {
child: FlowyButton( child: FlowyButton(
useIntrinsicWidth: true, useIntrinsicWidth: true,
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
LocaleKeys.document_plugins_cover_addIcon.tr(), LocaleKeys.document_plugins_cover_addIcon.tr(),
), ),
leftIcon: const FlowySvg(FlowySvgs.emoji_s), leftIcon: const FlowySvg(FlowySvgs.emoji_s),
@ -242,6 +243,7 @@ class RemoveEmojiButton extends StatelessWidget {
child: FlowyButton( child: FlowyButton(
useIntrinsicWidth: true, useIntrinsicWidth: true,
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
LocaleKeys.document_plugins_cover_removeIcon.tr(), LocaleKeys.document_plugins_cover_removeIcon.tr(),
), ),
leftIcon: const FlowySvg(FlowySvgs.emoji_s), leftIcon: const FlowySvg(FlowySvgs.emoji_s),

View File

@ -294,7 +294,11 @@ class ToggleHiddenFieldsVisibilityButton extends StatelessWidget {
return SizedBox( return SizedBox(
height: 30, height: 30,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(text, color: Theme.of(context).hintColor), text: FlowyText.medium(
text,
lineHeight: 1.0,
color: Theme.of(context).hintColor,
),
hoverColor: AFThemeExtension.of(context).lightGreyHover, hoverColor: AFThemeExtension.of(context).lightGreyHover,
leftIcon: RotatedBox( leftIcon: RotatedBox(
quarterTurns: quarterTurns, quarterTurns: quarterTurns,
@ -381,6 +385,7 @@ class _CreateRowFieldButtonState extends State<CreateRowFieldButton> {
child: FlowyButton( child: FlowyButton(
margin: const EdgeInsets.symmetric(horizontal: 4, vertical: 6), margin: const EdgeInsets.symmetric(horizontal: 4, vertical: 6),
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
LocaleKeys.grid_field_newProperty.tr(), LocaleKeys.grid_field_newProperty.tr(),
color: Theme.of(context).hintColor, color: Theme.of(context).hintColor,
), ),

View File

@ -80,6 +80,7 @@ class DatabaseViewLayoutCell extends StatelessWidget {
child: FlowyButton( child: FlowyButton(
hoverColor: AFThemeExtension.of(context).lightGreyHover, hoverColor: AFThemeExtension.of(context).lightGreyHover,
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
databaseLayout.layoutName, databaseLayout.layoutName,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
), ),

View File

@ -82,6 +82,7 @@ extension DatabaseSettingActionExtension on DatabaseSettingAction {
hoverColor: AFThemeExtension.of(context).lightGreyHover, hoverColor: AFThemeExtension.of(context).lightGreyHover,
text: FlowyText.medium( text: FlowyText.medium(
title(), title(),
lineHeight: 1.0,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
), ),
leftIcon: FlowySvg( leftIcon: FlowySvg(

View File

@ -150,6 +150,7 @@ class _DatabasePropertyCellState extends State<DatabasePropertyCell> {
child: FlowyButton( child: FlowyButton(
hoverColor: AFThemeExtension.of(context).lightGreyHover, hoverColor: AFThemeExtension.of(context).lightGreyHover,
text: FlowyText.medium( text: FlowyText.medium(
lineHeight: 1.0,
widget.fieldInfo.name, widget.fieldInfo.name,
color: AFThemeExtension.of(context).textColor, color: AFThemeExtension.of(context).textColor,
), ),

View File

@ -53,7 +53,10 @@ class SelectableItem extends StatelessWidget {
return SizedBox( return SizedBox(
height: 32, height: 32,
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(item), text: FlowyText.medium(
item,
lineHeight: 1.0,
),
rightIcon: isSelected ? const FlowySvg(FlowySvgs.check_s) : null, rightIcon: isSelected ? const FlowySvg(FlowySvgs.check_s) : null,
onTap: onTap, onTap: onTap,
), ),

View File

@ -163,7 +163,6 @@ class _FileUploadLocalState extends State<_FileUploadLocal> {
fontSize: 16, fontSize: 16,
maxLines: 2, maxLines: 2,
textAlign: TextAlign.center, textAlign: TextAlign.center,
lineHeight: 1.5,
), ),
], ],
], ],

View File

@ -247,6 +247,7 @@ class _CoverImagePreviewWidgetState extends State<CoverImagePreviewWidget> {
size: Size(20, 20), size: Size(20, 20),
), ),
text: FlowyText( text: FlowyText(
lineHeight: 1.0,
LocaleKeys.document_plugins_cover_pickFromFiles.tr(), LocaleKeys.document_plugins_cover_pickFromFiles.tr(),
), ),
), ),

View File

@ -161,7 +161,10 @@ class _ExportButton extends StatelessWidget {
borderRadius: radius, borderRadius: radius,
), ),
radius: radius, radius: radius,
text: FlowyText(title), text: FlowyText(
title,
lineHeight: 1.0,
),
leftIcon: FlowySvg(svg), leftIcon: FlowySvg(svg),
onTap: onTap, onTap: onTap,
); );

View File

@ -173,6 +173,7 @@ class _PublishedWidgetState extends State<_PublishedWidget> {
), ),
radius: BorderRadius.circular(10), radius: BorderRadius.circular(10),
text: FlowyText.regular( text: FlowyText.regular(
lineHeight: 1.0,
LocaleKeys.shareAction_unPublish.tr(), LocaleKeys.shareAction_unPublish.tr(),
textAlign: TextAlign.center, textAlign: TextAlign.center,
), ),

View File

@ -101,7 +101,10 @@ class _TrashPageState extends State<TrashPage> {
const Spacer(), const Spacer(),
IntrinsicWidth( IntrinsicWidth(
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(LocaleKeys.trash_restoreAll.tr()), text: FlowyText.medium(
LocaleKeys.trash_restoreAll.tr(),
lineHeight: 1.0,
),
leftIcon: const FlowySvg(FlowySvgs.restore_s), leftIcon: const FlowySvg(FlowySvgs.restore_s),
onTap: () { onTap: () {
NavigatorAlertDialog( NavigatorAlertDialog(
@ -118,7 +121,10 @@ class _TrashPageState extends State<TrashPage> {
const HSpace(6), const HSpace(6),
IntrinsicWidth( IntrinsicWidth(
child: FlowyButton( child: FlowyButton(
text: FlowyText.medium(LocaleKeys.trash_deleteAll.tr()), text: FlowyText.medium(
LocaleKeys.trash_deleteAll.tr(),
lineHeight: 1.0,
),
leftIcon: const FlowySvg(FlowySvgs.delete_s), leftIcon: const FlowySvg(FlowySvgs.delete_s),
onTap: () { onTap: () {
NavigatorAlertDialog( NavigatorAlertDialog(

View File

@ -257,6 +257,7 @@ class ViewMoreActionTypeWrapper extends CustomActionCell {
textBuilder: (onHover) => FlowyText.regular( textBuilder: (onHover) => FlowyText.regular(
inner.name, inner.name,
fontSize: 14.0, fontSize: 14.0,
lineHeight: 1.0,
figmaLineHeight: 18.0, figmaLineHeight: 18.0,
color: inner == ViewMoreActionType.delete && onHover color: inner == ViewMoreActionType.delete && onHover
? Theme.of(context).colorScheme.error ? Theme.of(context).colorScheme.error

View File

@ -204,7 +204,6 @@ class _UpgradeToAILocalPlanState extends State<_UpgradeToAILocalPlan> {
FlowyText.medium( FlowyText.medium(
LocaleKeys.sideBar_upgradeToAILocal.tr(), LocaleKeys.sideBar_upgradeToAILocal.tr(),
maxLines: 10, maxLines: 10,
lineHeight: 1.5,
), ),
const VSpace(4), const VSpace(4),
Opacity( Opacity(
@ -213,7 +212,6 @@ class _UpgradeToAILocalPlanState extends State<_UpgradeToAILocalPlan> {
LocaleKeys.sideBar_upgradeToAILocalDesc.tr(), LocaleKeys.sideBar_upgradeToAILocalDesc.tr(),
fontSize: 12, fontSize: 12,
maxLines: 10, maxLines: 10,
lineHeight: 1.5,
), ),
), ),
], ],

View File

@ -363,7 +363,6 @@ class _CurrentPathState extends State<_CurrentPath> {
resetHoverOnRebuild: false, resetHoverOnRebuild: false,
builder: (_, isHovering) => FlowyText.regular( builder: (_, isHovering) => FlowyText.regular(
widget.path, widget.path,
lineHeight: 1.5,
maxLines: 2, maxLines: 2,
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
decoration: isHovering ? TextDecoration.underline : null, decoration: isHovering ? TextDecoration.underline : null,

View File

@ -626,7 +626,6 @@ class _Heading extends StatelessWidget {
description!, description!,
fontSize: 12, fontSize: 12,
maxLines: 5, maxLines: 5,
lineHeight: 1.5,
), ),
), ),
], ],

View File

@ -42,7 +42,7 @@ class FlowyText extends StatelessWidget {
this.fontFamily, this.fontFamily,
this.fallbackFontFamily, this.fallbackFontFamily,
// https://api.flutter.dev/flutter/painting/TextStyle/height.html // https://api.flutter.dev/flutter/painting/TextStyle/height.html
this.lineHeight = 1, this.lineHeight = 1.5,
this.figmaLineHeight, this.figmaLineHeight,
this.withTooltip = false, this.withTooltip = false,
this.isEmoji = false, this.isEmoji = false,
@ -61,7 +61,7 @@ class FlowyText extends StatelessWidget {
this.selectable = false, this.selectable = false,
this.fontFamily, this.fontFamily,
this.fallbackFontFamily, this.fallbackFontFamily,
this.lineHeight, this.lineHeight = 1.5,
this.withTooltip = false, this.withTooltip = false,
this.isEmoji = false, this.isEmoji = false,
this.strutStyle, this.strutStyle,
@ -82,7 +82,7 @@ class FlowyText extends StatelessWidget {
this.selectable = false, this.selectable = false,
this.fontFamily, this.fontFamily,
this.fallbackFontFamily, this.fallbackFontFamily,
this.lineHeight, this.lineHeight = 1.5,
this.withTooltip = false, this.withTooltip = false,
this.isEmoji = false, this.isEmoji = false,
this.strutStyle, this.strutStyle,
@ -102,7 +102,7 @@ class FlowyText extends StatelessWidget {
this.selectable = false, this.selectable = false,
this.fontFamily, this.fontFamily,
this.fallbackFontFamily, this.fallbackFontFamily,
this.lineHeight = 1, this.lineHeight = 1.5,
this.withTooltip = false, this.withTooltip = false,
this.isEmoji = false, this.isEmoji = false,
this.strutStyle, this.strutStyle,
@ -122,7 +122,7 @@ class FlowyText extends StatelessWidget {
this.selectable = false, this.selectable = false,
this.fontFamily, this.fontFamily,
this.fallbackFontFamily, this.fallbackFontFamily,
this.lineHeight = 1, this.lineHeight = 1.5,
this.withTooltip = false, this.withTooltip = false,
this.isEmoji = false, this.isEmoji = false,
this.strutStyle, this.strutStyle,

View File

@ -1510,10 +1510,10 @@ packages:
dependency: transitive dependency: transitive
description: description:
name: pdf name: pdf
sha256: "81d5522bddc1ef5c28e8f0ee40b71708761753c163e0c93a40df56fd515ea0f0" sha256: "05df53f8791587402493ac97b9869d3824eccbc77d97855f4545cf72df3cae07"
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "3.11.0" version: "3.11.1"
pdf_widget_wrapper: pdf_widget_wrapper:
dependency: transitive dependency: transitive
description: description:
@ -1694,10 +1694,10 @@ packages:
dependency: transitive dependency: transitive
description: description:
name: qr name: qr
sha256: "64957a3930367bf97cc211a5af99551d630f2f4625e38af10edd6b19131b64b3" sha256: "5a1d2586170e172b8a8c8470bbbffd5eb0cd38a66c0d77155ea138d3af3a4445"
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "3.0.1" version: "3.0.2"
realtime_client: realtime_client:
dependency: transitive dependency: transitive
description: description:

View File

@ -144,12 +144,19 @@ impl AIManager {
chat_id: &str, chat_id: &str,
message: &str, message: &str,
message_type: ChatMessageType, message_type: ChatMessageType,
text_stream_port: i64, answer_stream_port: i64,
question_stream_port: i64,
metadata: Vec<ChatMessageMetadata>, metadata: Vec<ChatMessageMetadata>,
) -> Result<ChatMessagePB, FlowyError> { ) -> Result<ChatMessagePB, FlowyError> {
let chat = self.get_or_create_chat_instance(chat_id).await?; let chat = self.get_or_create_chat_instance(chat_id).await?;
let question = chat let question = chat
.stream_chat_message(message, message_type, text_stream_port, metadata) .stream_chat_message(
message,
message_type,
answer_stream_port,
question_stream_port,
metadata,
)
.await?; .await?;
Ok(question) Ok(question)
} }

View File

@ -5,6 +5,7 @@ use crate::entities::{
use crate::middleware::chat_service_mw::AICloudServiceMiddleware; use crate::middleware::chat_service_mw::AICloudServiceMiddleware;
use crate::notification::{make_notification, ChatNotification}; use crate::notification::{make_notification, ChatNotification};
use crate::persistence::{insert_chat_messages, select_chat_messages, ChatMessageTable}; use crate::persistence::{insert_chat_messages, select_chat_messages, ChatMessageTable};
use crate::stream_message::StreamMessage;
use allo_isolate::Isolate; use allo_isolate::Isolate;
use flowy_ai_pub::cloud::{ use flowy_ai_pub::cloud::{
ChatCloudService, ChatMessage, ChatMessageMetadata, ChatMessageType, MessageCursor, ChatCloudService, ChatMessage, ChatMessageMetadata, ChatMessageType, MessageCursor,
@ -81,7 +82,8 @@ impl Chat {
&self, &self,
message: &str, message: &str,
message_type: ChatMessageType, message_type: ChatMessageType,
text_stream_port: i64, answer_stream_port: i64,
question_stream_port: i64,
metadata: Vec<ChatMessageMetadata>, metadata: Vec<ChatMessageMetadata>,
) -> Result<ChatMessagePB, FlowyError> { ) -> Result<ChatMessagePB, FlowyError> {
if message.len() > 2000 { if message.len() > 2000 {
@ -93,10 +95,19 @@ impl Chat {
.store(false, std::sync::atomic::Ordering::SeqCst); .store(false, std::sync::atomic::Ordering::SeqCst);
self.stream_buffer.lock().await.clear(); self.stream_buffer.lock().await.clear();
let stream_buffer = self.stream_buffer.clone(); let mut question_sink = IsolateSink::new(Isolate::new(question_stream_port));
let answer_stream_buffer = self.stream_buffer.clone();
let uid = self.user_service.user_id()?; let uid = self.user_service.user_id()?;
let workspace_id = self.user_service.workspace_id()?; let workspace_id = self.user_service.workspace_id()?;
let _ = question_sink
.send(
StreamMessage::Text {
text: message.to_string(),
}
.to_string(),
)
.await;
let question = self let question = self
.chat_service .chat_service
.create_question( .create_question(
@ -112,15 +123,31 @@ impl Chat {
FlowyError::server_error() FlowyError::server_error()
})?; })?;
if self.chat_service.is_local_ai_enabled() { let _ = question_sink
.send(
StreamMessage::MessageId {
message_id: question.message_id,
}
.to_string(),
)
.await;
if self.chat_service.is_local_ai_enabled() && !metadata.is_empty() {
let _ = question_sink
.send(StreamMessage::IndexStart.to_string())
.await;
if let Err(err) = self if let Err(err) = self
.chat_service .chat_service
.index_message_metadata(&self.chat_id, &metadata) .index_message_metadata(&self.chat_id, &metadata, &mut question_sink)
.await .await
{ {
error!("Failed to index file: {}", err); error!("Failed to index file: {}", err);
} }
let _ = question_sink
.send(StreamMessage::IndexEnd.to_string())
.await;
} }
let _ = question_sink.send(StreamMessage::Done.to_string()).await;
save_chat_message( save_chat_message(
self.user_service.sqlite_connection(uid)?, self.user_service.sqlite_connection(uid)?,
@ -134,7 +161,7 @@ impl Chat {
let cloud_service = self.chat_service.clone(); let cloud_service = self.chat_service.clone();
let user_service = self.user_service.clone(); let user_service = self.user_service.clone();
tokio::spawn(async move { tokio::spawn(async move {
let mut text_sink = IsolateSink::new(Isolate::new(text_stream_port)); let mut answer_sink = IsolateSink::new(Isolate::new(answer_stream_port));
match cloud_service match cloud_service
.stream_answer(&workspace_id, &chat_id, question_id) .stream_answer(&workspace_id, &chat_id, question_id)
.await .await
@ -149,20 +176,20 @@ impl Chat {
} }
match message { match message {
QuestionStreamValue::Answer { value } => { QuestionStreamValue::Answer { value } => {
stream_buffer.lock().await.push_str(&value); answer_stream_buffer.lock().await.push_str(&value);
let _ = text_sink.send(format!("data:{}", value)).await; let _ = answer_sink.send(format!("data:{}", value)).await;
}, },
QuestionStreamValue::Metadata { value } => { QuestionStreamValue::Metadata { value } => {
if let Ok(s) = serde_json::to_string(&value) { if let Ok(s) = serde_json::to_string(&value) {
stream_buffer.lock().await.set_metadata(value); answer_stream_buffer.lock().await.set_metadata(value);
let _ = text_sink.send(format!("metadata:{}", s)).await; let _ = answer_sink.send(format!("metadata:{}", s)).await;
} }
}, },
} }
}, },
Err(err) => { Err(err) => {
error!("[Chat] failed to stream answer: {}", err); error!("[Chat] failed to stream answer: {}", err);
let _ = text_sink.send(format!("error:{}", err)).await; let _ = answer_sink.send(format!("error:{}", err)).await;
let pb = ChatMessageErrorPB { let pb = ChatMessageErrorPB {
chat_id: chat_id.clone(), chat_id: chat_id.clone(),
error_message: err.to_string(), error_message: err.to_string(),
@ -178,9 +205,9 @@ impl Chat {
Err(err) => { Err(err) => {
error!("[Chat] failed to stream answer: {}", err); error!("[Chat] failed to stream answer: {}", err);
if err.is_ai_response_limit_exceeded() { if err.is_ai_response_limit_exceeded() {
let _ = text_sink.send("AI_RESPONSE_LIMIT".to_string()).await; let _ = answer_sink.send("AI_RESPONSE_LIMIT".to_string()).await;
} else { } else {
let _ = text_sink.send(format!("error:{}", err)).await; let _ = answer_sink.send(format!("error:{}", err)).await;
} }
let pb = ChatMessageErrorPB { let pb = ChatMessageErrorPB {
@ -195,11 +222,11 @@ impl Chat {
} }
make_notification(&chat_id, ChatNotification::FinishStreaming).send(); make_notification(&chat_id, ChatNotification::FinishStreaming).send();
if stream_buffer.lock().await.is_empty() { if answer_stream_buffer.lock().await.is_empty() {
return Ok(()); return Ok(());
} }
let content = stream_buffer.lock().await.take_content(); let content = answer_stream_buffer.lock().await.take_content();
let metadata = stream_buffer.lock().await.take_metadata(); let metadata = answer_stream_buffer.lock().await.take_metadata();
let answer = cloud_service let answer = cloud_service
.create_answer(&workspace_id, &chat_id, &content, question_id, metadata) .create_answer(&workspace_id, &chat_id, &content, question_id, metadata)

View File

@ -61,9 +61,12 @@ pub struct StreamChatPayloadPB {
pub message_type: ChatMessageTypePB, pub message_type: ChatMessageTypePB,
#[pb(index = 4)] #[pb(index = 4)]
pub text_stream_port: i64, pub answer_stream_port: i64,
#[pb(index = 5)] #[pb(index = 5)]
pub question_stream_port: i64,
#[pb(index = 6)]
pub metadata: Vec<ChatMessageMetaPB>, pub metadata: Vec<ChatMessageMetaPB>,
} }

View File

@ -72,7 +72,8 @@ pub(crate) async fn stream_chat_message_handler(
&data.chat_id, &data.chat_id,
&data.message, &data.message,
message_type, message_type,
data.text_stream_port, data.answer_stream_port,
data.question_stream_port,
metadata, metadata,
) )
.await; .await;

View File

@ -10,3 +10,4 @@ mod middleware;
pub mod notification; pub mod notification;
mod persistence; mod persistence;
mod protobuf; mod protobuf;
mod stream_message;

View File

@ -16,11 +16,13 @@ use futures::Sink;
use lib_infra::async_trait::async_trait; use lib_infra::async_trait::async_trait;
use std::collections::HashMap; use std::collections::HashMap;
use crate::stream_message::StreamMessage;
use futures_util::SinkExt;
use parking_lot::Mutex; use parking_lot::Mutex;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use serde_json::json; use serde_json::json;
use std::ops::Deref; use std::ops::Deref;
use std::path::Path; use std::path::{Path, PathBuf};
use std::sync::Arc; use std::sync::Arc;
use tokio::select; use tokio::select;
use tokio_stream::StreamExt; use tokio_stream::StreamExt;
@ -339,11 +341,11 @@ impl LocalAIController {
.set_bool(APPFLOWY_LOCAL_AI_CHAT_RAG_ENABLED, enabled)?; .set_bool(APPFLOWY_LOCAL_AI_CHAT_RAG_ENABLED, enabled)?;
Ok(enabled) Ok(enabled)
} }
pub async fn index_message_metadata( pub async fn index_message_metadata(
&self, &self,
chat_id: &str, chat_id: &str,
metadata_list: &[ChatMessageMetadata], metadata_list: &[ChatMessageMetadata],
index_process_sink: &mut (impl Sink<String> + Unpin),
) -> FlowyResult<()> { ) -> FlowyResult<()> {
for metadata in metadata_list { for metadata in metadata_list {
let mut index_metadata = HashMap::new(); let mut index_metadata = HashMap::new();
@ -351,52 +353,94 @@ impl LocalAIController {
index_metadata.insert("at_name".to_string(), json!(format!("@{}", &metadata.name))); index_metadata.insert("at_name".to_string(), json!(format!("@{}", &metadata.name)));
index_metadata.insert("source".to_string(), json!(&metadata.source)); index_metadata.insert("source".to_string(), json!(&metadata.source));
match &metadata.data.url { if let Some(url) = &metadata.data.url {
None => match &metadata.data.content_type { let file_path = Path::new(url);
ChatMetadataContentType::Text | ChatMetadataContentType::Markdown => { if file_path.exists() {
if metadata.data.validate() { self
if let Err(err) = self .process_index_file(
.index_file( chat_id,
chat_id, Some(file_path.to_path_buf()),
None, None,
Some(metadata.data.content.clone()), metadata,
Some(index_metadata), &index_metadata,
) index_process_sink,
.await )
{ .await?;
error!("[AI Plugin] failed to index file: {:?}", err); }
} } else if matches!(
} metadata.data.content_type,
}, ChatMetadataContentType::Text | ChatMetadataContentType::Markdown
_ => { ) && metadata.data.validate()
error!( {
"[AI Plugin] unsupported content type: {:?}", self
metadata.data.content_type .process_index_file(
); chat_id,
}, None,
}, Some(metadata.data.content.clone()),
Some(url) => { metadata,
let file_path = Path::new(url); &index_metadata,
if file_path.exists() { index_process_sink,
if let Err(err) = self )
.index_file( .await?;
chat_id, } else {
Some(file_path.to_path_buf()), error!(
None, "[AI Plugin] unsupported content type: {:?}",
Some(index_metadata), metadata.data.content_type
) );
.await
{
error!("[AI Plugin] failed to index file: {:?}", err);
}
}
},
} }
} }
Ok(()) Ok(())
} }
async fn process_index_file(
&self,
chat_id: &str,
file_path: Option<PathBuf>,
content: Option<String>,
metadata: &ChatMessageMetadata,
index_metadata: &HashMap<String, serde_json::Value>,
index_process_sink: &mut (impl Sink<String> + Unpin),
) -> Result<(), FlowyError> {
let _ = index_process_sink
.send(
StreamMessage::StartIndexFile {
file_name: metadata.name.clone(),
}
.to_string(),
)
.await;
let result = self
.index_file(chat_id, file_path, content, Some(index_metadata.clone()))
.await;
match result {
Ok(_) => {
let _ = index_process_sink
.send(
StreamMessage::EndIndexFile {
file_name: metadata.name.clone(),
}
.to_string(),
)
.await;
},
Err(err) => {
let _ = index_process_sink
.send(
StreamMessage::IndexFileError {
file_name: metadata.name.clone(),
}
.to_string(),
)
.await;
error!("[AI Plugin] failed to index file: {:?}", err);
},
}
Ok(())
}
async fn enable_chat_plugin(&self, enabled: bool) -> FlowyResult<()> { async fn enable_chat_plugin(&self, enabled: bool) -> FlowyResult<()> {
info!("[AI Plugin] enable chat plugin: {}", enabled); info!("[AI Plugin] enable chat plugin: {}", enabled);
if enabled { if enabled {

View File

@ -11,7 +11,7 @@ use flowy_ai_pub::cloud::{
RepeatedRelatedQuestion, StreamAnswer, StreamComplete, RepeatedRelatedQuestion, StreamAnswer, StreamComplete,
}; };
use flowy_error::{FlowyError, FlowyResult}; use flowy_error::{FlowyError, FlowyResult};
use futures::{stream, StreamExt, TryStreamExt}; use futures::{stream, Sink, StreamExt, TryStreamExt};
use lib_infra::async_trait::async_trait; use lib_infra::async_trait::async_trait;
use lib_infra::future::FutureResult; use lib_infra::future::FutureResult;
@ -48,10 +48,11 @@ impl AICloudServiceMiddleware {
&self, &self,
chat_id: &str, chat_id: &str,
metadata_list: &[ChatMessageMetadata], metadata_list: &[ChatMessageMetadata],
index_process_sink: &mut (impl Sink<String> + Unpin),
) -> Result<(), FlowyError> { ) -> Result<(), FlowyError> {
self self
.local_llm_controller .local_llm_controller
.index_message_metadata(chat_id, metadata_list) .index_message_metadata(chat_id, metadata_list, index_process_sink)
.await?; .await?;
Ok(()) Ok(())
} }

View File

@ -0,0 +1,34 @@
use std::fmt::Display;
pub enum StreamMessage {
MessageId { message_id: i64 },
IndexStart,
IndexEnd,
Text { text: String },
Done,
StartIndexFile { file_name: String },
EndIndexFile { file_name: String },
IndexFileError { file_name: String },
}
impl Display for StreamMessage {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
StreamMessage::MessageId { message_id } => write!(f, "message_id:{}", message_id),
StreamMessage::IndexStart => write!(f, "index_start:"),
StreamMessage::IndexEnd => write!(f, "index_end"),
StreamMessage::Text { text } => {
write!(f, "data:{}", text)
},
StreamMessage::Done => write!(f, "done:"),
StreamMessage::StartIndexFile { file_name } => {
write!(f, "start_index_file:{}", file_name)
},
StreamMessage::EndIndexFile { file_name } => {
write!(f, "end_index_file:{}", file_name)
},
StreamMessage::IndexFileError { file_name } => {
write!(f, "index_file_error:{}", file_name)
},
}
}
}