feat: on create course, go to invite page while course creation loads (#4178)

pull/2245/head
ggurdin 4 weeks ago committed by GitHub
parent def99c22e5
commit 1605624006
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -37,6 +37,7 @@ import 'package:fluffychat/pangea/analytics_summary/progress_indicators_enum.dar
import 'package:fluffychat/pangea/chat_settings/pages/edit_course.dart';
import 'package:fluffychat/pangea/chat_settings/pages/pangea_invitation_selection.dart';
import 'package:fluffychat/pangea/constructs/construct_identifier.dart';
import 'package:fluffychat/pangea/course_creation/course_invite_page.dart';
import 'package:fluffychat/pangea/course_creation/selected_course_page.dart';
import 'package:fluffychat/pangea/extensions/pangea_room_extension.dart';
import 'package:fluffychat/pangea/find_your_people/find_your_people_constants.dart';
@ -245,6 +246,22 @@ abstract class AppRoutes {
),
);
},
routes: [
GoRoute(
path: 'invite',
pageBuilder: (context, state) {
return defaultPageBuilder(
context,
state,
CourseInvitePage(
state.pathParameters['courseid']!,
courseCreationCompleter:
state.extra as Completer<String>?,
),
);
},
),
],
),
],
),
@ -445,6 +462,22 @@ abstract class AppRoutes {
),
);
},
routes: [
GoRoute(
path: 'invite',
pageBuilder: (context, state) {
return defaultPageBuilder(
context,
state,
CourseInvitePage(
state.pathParameters['courseid']!,
courseCreationCompleter:
state.extra as Completer<String>?,
),
);
},
),
],
),
],
),

@ -5295,5 +5295,8 @@
"feedbackDesc": "How should the activity be improved? If you can provide some details, well make the change!",
"feedbackHint": "Your feedback",
"feedbackButton": "Submit feedback",
"directMessageBotDesc": "Talking to humans is more fun but... AI is always ready!"
"directMessageBotDesc": "Talking to humans is more fun but... AI is always ready!",
"inviteYourFriends": "Invite your friends",
"playWithAI": "Play with AI for now",
"courseStartDesc": "Pangea Bot is ready to go anytime!\n\n...but learning is better with friends!"
}

@ -0,0 +1,233 @@
import 'dart:async';
import 'dart:math';
import 'package:flutter/material.dart';
import 'package:go_router/go_router.dart';
import 'package:matrix/matrix.dart';
import 'package:fluffychat/config/app_config.dart';
import 'package:fluffychat/l10n/l10n.dart';
import 'package:fluffychat/pangea/bot/utils/bot_name.dart';
import 'package:fluffychat/pangea/course_creation/course_info_chip_widget.dart';
import 'package:fluffychat/pangea/course_plans/course_plan_builder.dart';
import 'package:fluffychat/widgets/avatar.dart';
import 'package:fluffychat/widgets/future_loading_dialog.dart';
import 'package:fluffychat/widgets/matrix.dart';
class CourseInvitePage extends StatefulWidget {
final String courseId;
final Completer<String>? courseCreationCompleter;
const CourseInvitePage(
this.courseId, {
super.key,
this.courseCreationCompleter,
});
@override
CourseInvitePageController createState() => CourseInvitePageController();
}
class CourseInvitePageController extends State<CourseInvitePage> {
Future<String> getSpaceId() async {
if (widget.courseCreationCompleter == null) {
throw Exception("No course creation completer provided");
}
return widget.courseCreationCompleter!.future;
}
@override
Widget build(BuildContext context) {
const avatarSize = 44.0;
final theme = Theme.of(context);
final client = Matrix.of(context).client;
return CoursePlanBuilder(
courseId: widget.courseId,
builder: (context, courseController) {
return Scaffold(
body: SafeArea(
child: Center(
child: Container(
padding: const EdgeInsets.all(30.0),
constraints: const BoxConstraints(
maxWidth: 750,
),
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
courseController.course != null
? Container(
decoration: BoxDecoration(
border: Border.all(color: AppConfig.gold),
borderRadius: BorderRadius.circular(12),
),
padding: const EdgeInsets.all(16.0),
child: Column(
spacing: 16.0,
mainAxisSize: MainAxisSize.min,
children: [
Row(
spacing: 10.0,
mainAxisAlignment: MainAxisAlignment.center,
children: [
const Icon(
Icons.map_outlined,
size: 40.0,
),
Flexible(
child: Text(
courseController.course!.title,
style: theme.textTheme.titleLarge
?.copyWith(
fontWeight: FontWeight.bold,
),
overflow: TextOverflow.ellipsis,
),
),
],
),
CourseInfoChips(
courseController.course!,
fontSize: 12.0,
iconSize: 12.0,
),
],
),
)
: const CircularProgressIndicator.adaptive(),
Padding(
padding: const EdgeInsets.all(8.0),
child: Column(
spacing: 16.0,
mainAxisSize: MainAxisSize.min,
children: [
LayoutBuilder(
builder: (context, constraints) {
const avatarSpace = avatarSize + 8.0;
final availableSpace =
constraints.maxWidth - 24.0;
final visibleAvatars = min(
3,
(availableSpace / avatarSpace).floor() - 2,
);
return Row(
spacing: 8.0,
mainAxisSize: MainAxisSize.min,
children: [
FutureBuilder(
future: client
.getProfileFromUserId(client.userID!),
builder: (context, snapshot) {
return Avatar(
size: avatarSize,
mxContent: snapshot.data?.avatarUrl,
name: snapshot.data?.displayName ??
client.userID!.localpart,
userId: client.userID!,
);
},
),
Avatar(
userId: BotName.byEnvironment,
size: avatarSize,
),
...List.generate(visibleAvatars, (index) {
return CircleAvatar(
radius: avatarSize / 2,
backgroundColor:
AppConfig.gold.withAlpha(80),
child: const Icon(
Icons.question_mark,
size: 20.0,
),
);
}),
const Icon(
Icons.more_horiz,
size: 24.0,
),
],
);
},
),
Text(
L10n.of(context).courseStartDesc,
style: theme.textTheme.titleMedium,
),
],
),
),
Column(
spacing: 24.0,
mainAxisSize: MainAxisSize.min,
children: [
ElevatedButton(
onPressed: () async {
final resp = await showFutureLoadingDialog(
context: context,
future: getSpaceId,
);
if (mounted && !resp.isError) {
context.go("/rooms/spaces/${resp.result}/invite");
}
},
style: ElevatedButton.styleFrom(
backgroundColor: theme.colorScheme.surface,
foregroundColor: theme.colorScheme.onSurface,
side: BorderSide(
width: 1,
color: theme.colorScheme.onSurface,
),
),
child: Row(
spacing: 8.0,
mainAxisAlignment: MainAxisAlignment.center,
children: [
const Icon(Icons.upload_file),
Text(L10n.of(context).inviteYourFriends),
],
),
),
ElevatedButton(
onPressed: () async {
final resp = await showFutureLoadingDialog(
context: context,
future: getSpaceId,
);
if (mounted && !resp.isError) {
context.go("/rooms/spaces/${resp.result}");
}
},
style: ElevatedButton.styleFrom(
backgroundColor: theme.colorScheme.surface,
foregroundColor: theme.colorScheme.onSurface,
side: BorderSide(
width: 1,
color: theme.colorScheme.onSurface,
),
),
child: Row(
spacing: 8.0,
mainAxisAlignment: MainAxisAlignment.center,
children: [
Text(L10n.of(context).playWithAI),
],
),
),
],
),
],
),
),
),
),
);
},
);
}
}

@ -1,3 +1,5 @@
import 'dart:async';
import 'package:flutter/material.dart';
import 'package:go_router/go_router.dart';
@ -23,29 +25,33 @@ class SelectedCourse extends StatefulWidget {
class SelectedCourseController extends State<SelectedCourse> {
Future<void> launchCourse(CoursePlanModel course) async {
final client = Matrix.of(context).client;
final roomId = await client.createPangeaSpace(
name: course.title,
topic: course.description,
introChatName: L10n.of(context).introductions,
announcementsChatName: L10n.of(context).announcements,
visibility: sdk.Visibility.private,
joinRules: sdk.JoinRules.knock,
initialState: [
sdk.StateEvent(
type: PangeaEventTypes.coursePlan,
content: {
"uuid": course.uuid,
},
),
],
avatarUrl: course.imageUrl,
spaceChild: 0,
);
final Completer<String> completer = Completer<String>();
client
.createPangeaSpace(
name: course.title,
topic: course.description,
introChatName: L10n.of(context).introductions,
announcementsChatName: L10n.of(context).announcements,
visibility: sdk.Visibility.private,
joinRules: sdk.JoinRules.knock,
initialState: [
sdk.StateEvent(
type: PangeaEventTypes.coursePlan,
content: {
"uuid": course.uuid,
},
),
],
avatarUrl: course.imageUrl,
spaceChild: 0,
)
.then((spaceId) => completer.complete(spaceId))
.catchError((error) => completer.completeError(error));
if (!mounted) return;
final room = client.getRoomById(roomId);
if (room == null) return;
context.go("/rooms/spaces/${room.id}/details");
context.go(
"/rooms/course/own/${widget.courseId}/invite",
extra: completer,
);
}
Future<void> addCourseToSpace(CoursePlanModel course) async {

Loading…
Cancel
Save