Merge pull request #208 from SyncrowIOT/SP-1603-FE-Freeze-First-Row-in-All-Table-Views-Across-the-Platform

Refactor table layout to accommodate dynamic table size
This commit is contained in:
mohammadnemer1
2025-05-28 16:57:56 +03:00
committed by GitHub
4 changed files with 475 additions and 377 deletions

View File

@ -21,6 +21,7 @@ class DynamicTable extends StatefulWidget {
final List<String>? initialSelectedIds; final List<String>? initialSelectedIds;
final int uuidIndex; final int uuidIndex;
final Function(dynamic selectedRows)? onSelectionChanged; final Function(dynamic selectedRows)? onSelectionChanged;
final Function(int rowIndex)? onSettingsPressed;
const DynamicTable({ const DynamicTable({
super.key, super.key,
required this.headers, required this.headers,
@ -37,6 +38,7 @@ class DynamicTable extends StatefulWidget {
this.initialSelectedIds, this.initialSelectedIds,
required this.uuidIndex, required this.uuidIndex,
this.onSelectionChanged, this.onSelectionChanged,
this.onSettingsPressed,
}); });
@override @override
@ -48,11 +50,20 @@ class _DynamicTableState extends State<DynamicTable> {
bool _selectAll = false; bool _selectAll = false;
final ScrollController _verticalScrollController = ScrollController(); final ScrollController _verticalScrollController = ScrollController();
final ScrollController _horizontalScrollController = ScrollController(); final ScrollController _horizontalScrollController = ScrollController();
late ScrollController _horizontalHeaderScrollController;
late ScrollController _horizontalBodyScrollController;
@override @override
void initState() { void initState() {
super.initState(); super.initState();
_initializeSelection(); _initializeSelection();
_horizontalHeaderScrollController = ScrollController();
_horizontalBodyScrollController = ScrollController();
// Synchronize horizontal scrolling
_horizontalBodyScrollController.addListener(() {
_horizontalHeaderScrollController
.jumpTo(_horizontalBodyScrollController.offset);
});
} }
@override @override
@ -102,102 +113,88 @@ class _DynamicTableState extends State<DynamicTable> {
context.read<DeviceManagementBloc>().add(UpdateSelection(_selectedRows)); context.read<DeviceManagementBloc>().add(UpdateSelection(_selectedRows));
} }
@override
void dispose() {
_horizontalHeaderScrollController.dispose();
_horizontalBodyScrollController.dispose();
super.dispose();
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Container( return Container(
decoration: widget.cellDecoration, decoration: widget.cellDecoration,
child: Scrollbar(
controller: _verticalScrollController,
thumbVisibility: true,
trackVisibility: true,
child: Scrollbar(
controller: _horizontalScrollController,
thumbVisibility: true,
trackVisibility: true,
notificationPredicate: (notif) => notif.depth == 1,
child: SingleChildScrollView(
controller: _verticalScrollController,
child: SingleChildScrollView(
controller: _horizontalScrollController,
scrollDirection: Axis.horizontal,
child: SizedBox(
width: widget.size.width,
child: Column( child: Column(
children: [ children: [
Container( Container(
decoration: widget.headerDecoration ?? decoration: widget.headerDecoration ??
const BoxDecoration( const BoxDecoration(color: ColorsManager.boxColor),
color: ColorsManager.boxColor, child: SingleChildScrollView(
), scrollDirection: Axis.horizontal,
physics: const NeverScrollableScrollPhysics(),
controller: _horizontalHeaderScrollController,
child: SizedBox(
width: widget.size.width,
child: Row( child: Row(
children: [ children: [
if (widget.withCheckBox) _buildSelectAllCheckbox(), if (widget.withCheckBox) _buildSelectAllCheckbox(),
...List.generate(widget.headers.length, (index) { ...List.generate(widget.headers.length, (index) {
return _buildTableHeaderCell( return _buildTableHeaderCell(
widget.headers[index], index); widget.headers[index], index);
}) }),
//...widget.headers.map((header) => _buildTableHeaderCell(header)),
], ],
), ),
), ),
widget.isEmpty ),
? SizedBox( ),
height: widget.size.height * 0.5, Expanded(
child: Scrollbar(
controller: _verticalScrollController,
thumbVisibility: true,
trackVisibility: true,
child: SingleChildScrollView(
controller: _verticalScrollController,
child: Scrollbar(
controller: _horizontalBodyScrollController,
thumbVisibility: false,
trackVisibility: false,
notificationPredicate: (notif) => notif.depth == 1,
child: SingleChildScrollView(
scrollDirection: Axis.horizontal,
controller: _horizontalBodyScrollController,
child: SizedBox(
width: widget.size.width, width: widget.size.width,
child: Column( child: widget.isEmpty
mainAxisAlignment: MainAxisAlignment.center, ? _buildEmptyState()
children: [
Row(
crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center,
children: [
Column(
children: [
SvgPicture.asset(Assets.emptyTable),
const SizedBox(
height: 15,
),
Text(
widget.tableName == 'AccessManagement'
? 'No Password '
: 'No Devices',
style: Theme.of(context)
.textTheme
.bodySmall!
.copyWith(
color:
ColorsManager.grayColor),
)
],
),
],
),
],
),
)
: Column( : Column(
children: children:
List.generate(widget.data.length, (index) { List.generate(widget.data.length, (rowIndex) {
final row = widget.data[index]; final row = widget.data[rowIndex];
return Row( return Row(
children: [ children: [
if (widget.withCheckBox) if (widget.withCheckBox)
_buildRowCheckbox( _buildRowCheckbox(
index, widget.size.height * 0.08), rowIndex, widget.size.height * 0.08),
...row.map((cell) => _buildTableCell( ...row.asMap().entries.map((entry) {
cell.toString(), return _buildTableCell(
widget.size.height * 0.08)), entry.value.toString(),
widget.size.height * 0.08,
rowIndex: rowIndex,
columnIndex: entry.key,
);
}).toList(),
], ],
); );
}), }),
), ),
),
),
),
),
),
),
], ],
), ),
),
),
),
),
),
); );
} }
@ -218,6 +215,32 @@ class _DynamicTableState extends State<DynamicTable> {
); );
} }
Widget _buildEmptyState() => Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Row(
crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center,
children: [
Column(
children: [
SvgPicture.asset(Assets.emptyTable),
const SizedBox(height: 15),
Text(
widget.tableName == 'AccessManagement'
? 'No Password '
: 'No Devices',
style: Theme.of(context)
.textTheme
.bodySmall!
.copyWith(color: ColorsManager.grayColor),
)
],
),
],
),
],
);
Widget _buildRowCheckbox(int index, double size) { Widget _buildRowCheckbox(int index, double size) {
return Container( return Container(
width: 50, width: 50,
@ -272,13 +295,23 @@ class _DynamicTableState extends State<DynamicTable> {
); );
} }
Widget _buildTableCell(String content, double size) { Widget _buildTableCell(
String content,
double size, {
required int rowIndex,
required int columnIndex,
}) {
bool isBatteryLevel = content.endsWith('%'); bool isBatteryLevel = content.endsWith('%');
double? batteryLevel; double? batteryLevel;
if (isBatteryLevel) { if (isBatteryLevel) {
batteryLevel = double.tryParse(content.replaceAll('%', '').trim()); batteryLevel = double.tryParse(content.replaceAll('%', '').trim());
} }
bool isSettingsColumn = widget.headers[columnIndex] == 'Settings';
if (isSettingsColumn) {
return _buildSettingsIcon(rowIndex, size);
}
Color? statusColor; Color? statusColor;
switch (content) { switch (content) {
@ -330,4 +363,23 @@ class _DynamicTableState extends State<DynamicTable> {
), ),
); );
} }
Widget _buildSettingsIcon(int rowIndex, double size) {
return Container(
height: size,
width: 120,
padding: const EdgeInsets.all(5.0),
decoration: const BoxDecoration(
border: Border(
bottom: BorderSide(color: ColorsManager.boxDivider, width: 1.0),
),
color: Colors.white,
),
alignment: Alignment.center,
child: IconButton(
icon: SvgPicture.asset(Assets.settings),
onPressed: () => widget.onSettingsPressed?.call(rowIndex),
),
);
}
} }

View File

@ -5,7 +5,7 @@ import 'package:syncrow_web/utils/constants/assets.dart';
Future<void> showNameMenu({ Future<void> showNameMenu({
required BuildContext context, required BuildContext context,
Function()? aToZTap, Function()? aToZTap,
Function()? zToaTap, Function()? zToATap,
String? isSelected, String? isSelected,
}) async { }) async {
final RenderBox overlay = final RenderBox overlay =
@ -46,7 +46,7 @@ Future<void> showNameMenu({
), ),
), ),
PopupMenuItem( PopupMenuItem(
onTap: zToaTap, onTap: zToATap,
child: ListTile( child: ListTile(
leading: Image.asset( leading: Image.asset(
Assets.ZtoAIcon, Assets.ZtoAIcon,

View File

@ -95,7 +95,7 @@ class _TableRow extends StatelessWidget {
], ],
), ),
if (!isLast) if (!isLast)
Divider( const Divider(
height: 1, height: 1,
thickness: 1, thickness: 1,
color: ColorsManager.boxDivider, color: ColorsManager.boxDivider,
@ -110,12 +110,14 @@ class DynamicTableScreen extends StatefulWidget {
final List<String> titles; final List<String> titles;
final List<List<Widget>> rows; final List<List<Widget>> rows;
final void Function(int columnIndex)? onFilter; final void Function(int columnIndex)? onFilter;
final double tableSize;
const DynamicTableScreen({ const DynamicTableScreen({
required this.titles, required this.titles,
required this.rows, required this.rows,
required this.onFilter, required this.onFilter,
Key? key, Key? key,
required this.tableSize,
}) : super(key: key); }) : super(key: key);
@override @override
@ -205,7 +207,8 @@ class _DynamicTableScreenState extends State<DynamicTableScreen> {
bottomRight: Radius.circular(15), bottomRight: Radius.circular(15),
), ),
), ),
child: Column( child: ListView(
shrinkWrap: true,
children: [ children: [
for (int rowIndex = 0; rowIndex < widget.rows.length; rowIndex++) for (int rowIndex = 0; rowIndex < widget.rows.length; rowIndex++)
_TableRow( _TableRow(
@ -253,7 +256,7 @@ class _DynamicTableScreenState extends State<DynamicTableScreen> {
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
_buildHeader(), _buildHeader(),
_buildBody(), Container(height: widget.tableSize - 37, child: _buildBody()),
], ],
), ),
), ),

View File

@ -27,7 +27,8 @@ class UsersPage extends StatelessWidget {
Widget build(BuildContext context) { Widget build(BuildContext context) {
final TextEditingController searchController = TextEditingController(); final TextEditingController searchController = TextEditingController();
Widget actionButton({bool isActive = false, required String title, Function()? onTap}) { Widget actionButton(
{bool isActive = false, required String title, Function()? onTap}) {
return InkWell( return InkWell(
onTap: onTap, onTap: onTap,
child: Padding( child: Padding(
@ -60,7 +61,8 @@ class UsersPage extends StatelessWidget {
: ColorsManager.disabledPink.withOpacity(0.5), : ColorsManager.disabledPink.withOpacity(0.5),
), ),
child: Padding( child: Padding(
padding: const EdgeInsets.only(left: 10, right: 10, bottom: 5, top: 5), padding:
const EdgeInsets.only(left: 10, right: 10, bottom: 5, top: 5),
child: Row( child: Row(
crossAxisAlignment: CrossAxisAlignment.center, crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
@ -84,12 +86,15 @@ class UsersPage extends StatelessWidget {
} }
Widget changeIconStatus( Widget changeIconStatus(
{required String userId, required String status, required Function()? onTap}) { {required String userId,
required String status,
required Function()? onTap}) {
return Center( return Center(
child: InkWell( child: InkWell(
onTap: onTap, onTap: onTap,
child: Padding( child: Padding(
padding: const EdgeInsets.only(left: 5, right: 5, bottom: 5, top: 5), padding:
const EdgeInsets.only(left: 5, right: 5, bottom: 5, top: 5),
child: SvgPicture.asset( child: SvgPicture.asset(
status == "invited" status == "invited"
? Assets.invitedIcon ? Assets.invitedIcon
@ -114,8 +119,7 @@ class UsersPage extends StatelessWidget {
padding: const EdgeInsets.all(20), padding: const EdgeInsets.all(20),
child: Align( child: Align(
alignment: Alignment.topCenter, alignment: Alignment.topCenter,
child: ListView( child: Column(
shrinkWrap: true,
children: [ children: [
Row( Row(
children: [ children: [
@ -188,18 +192,25 @@ class UsersPage extends StatelessWidget {
), ),
], ],
), ),
const SizedBox(height: 25), const SizedBox(height: 20),
DynamicTableScreen( Container(
height: screenSize.height * 0.65,
child: DynamicTableScreen(
tableSize: screenSize.height * 0.65,
onFilter: (columnIndex) { onFilter: (columnIndex) {
if (columnIndex == 0) { if (columnIndex == 0) {
showNameMenu( showNameMenu(
context: context, context: context,
isSelected: _blocRole.currentSortOrder, isSelected: _blocRole.currentSortOrder,
aToZTap: () { aToZTap: () {
context.read<UserTableBloc>().add(const SortUsersByNameAsc()); context
.read<UserTableBloc>()
.add(const SortUsersByNameAsc());
}, },
zToaTap: () { zToATap: () {
context.read<UserTableBloc>().add(const SortUsersByNameDesc()); context
.read<UserTableBloc>()
.add(const SortUsersByNameDesc());
}, },
); );
} }
@ -208,8 +219,9 @@ class UsersPage extends StatelessWidget {
for (var item in _blocRole.jobTitle) for (var item in _blocRole.jobTitle)
item: _blocRole.selectedJobTitles.contains(item), item: _blocRole.selectedJobTitles.contains(item),
}; };
final RenderBox overlay = final RenderBox overlay = Overlay.of(context)
Overlay.of(context).context.findRenderObject() as RenderBox; .context
.findRenderObject() as RenderBox;
showPopUpFilterMenu( showPopUpFilterMenu(
position: RelativeRect.fromLTRB( position: RelativeRect.fromLTRB(
@ -249,8 +261,9 @@ class UsersPage extends StatelessWidget {
for (var item in _blocRole.roleTypes) for (var item in _blocRole.roleTypes)
item: _blocRole.selectedRoles.contains(item), item: _blocRole.selectedRoles.contains(item),
}; };
final RenderBox overlay = final RenderBox overlay = Overlay.of(context)
Overlay.of(context).context.findRenderObject() as RenderBox; .context
.findRenderObject() as RenderBox;
showPopUpFilterMenu( showPopUpFilterMenu(
position: RelativeRect.fromLTRB( position: RelativeRect.fromLTRB(
overlay.size.width / 4, overlay.size.width / 4,
@ -270,7 +283,8 @@ class UsersPage extends StatelessWidget {
.map((entry) => entry.key) .map((entry) => entry.key)
.toList(); .toList();
Navigator.of(context).pop(); Navigator.of(context).pop();
context.read<UserTableBloc>().add(FilterUsersByRoleEvent( context.read<UserTableBloc>().add(
FilterUsersByRoleEvent(
selectedRoles: selectedItems, selectedRoles: selectedItems,
sortOrder: _blocRole.currentSortRole)); sortOrder: _blocRole.currentSortRole));
}, },
@ -287,10 +301,14 @@ class UsersPage extends StatelessWidget {
context: context, context: context,
isSelected: _blocRole.currentSortOrder, isSelected: _blocRole.currentSortOrder,
aToZTap: () { aToZTap: () {
context.read<UserTableBloc>().add(const DateNewestToOldestEvent()); context
.read<UserTableBloc>()
.add(const DateNewestToOldestEvent());
}, },
zToaTap: () { zToaTap: () {
context.read<UserTableBloc>().add(const DateOldestToNewestEvent()); context
.read<UserTableBloc>()
.add(const DateOldestToNewestEvent());
}, },
); );
} }
@ -299,8 +317,9 @@ class UsersPage extends StatelessWidget {
for (var item in _blocRole.createdBy) for (var item in _blocRole.createdBy)
item: _blocRole.selectedCreatedBy.contains(item), item: _blocRole.selectedCreatedBy.contains(item),
}; };
final RenderBox overlay = final RenderBox overlay = Overlay.of(context)
Overlay.of(context).context.findRenderObject() as RenderBox; .context
.findRenderObject() as RenderBox;
showPopUpFilterMenu( showPopUpFilterMenu(
position: RelativeRect.fromLTRB( position: RelativeRect.fromLTRB(
overlay.size.width / 1, overlay.size.width / 1,
@ -338,8 +357,9 @@ class UsersPage extends StatelessWidget {
item: _blocRole.selectedStatuses.contains(item), item: _blocRole.selectedStatuses.contains(item),
}; };
final RenderBox overlay = final RenderBox overlay = Overlay.of(context)
Overlay.of(context).context.findRenderObject() as RenderBox; .context
.findRenderObject() as RenderBox;
showPopUpFilterMenu( showPopUpFilterMenu(
position: RelativeRect.fromLTRB( position: RelativeRect.fromLTRB(
overlay.size.width / 0, overlay.size.width / 0,
@ -376,10 +396,14 @@ class UsersPage extends StatelessWidget {
context: context, context: context,
isSelected: _blocRole.currentSortOrderDate, isSelected: _blocRole.currentSortOrderDate,
aToZTap: () { aToZTap: () {
context.read<UserTableBloc>().add(const DateNewestToOldestEvent()); context
.read<UserTableBloc>()
.add(const DateNewestToOldestEvent());
}, },
zToaTap: () { zToaTap: () {
context.read<UserTableBloc>().add(const DateOldestToNewestEvent()); context
.read<UserTableBloc>()
.add(const DateOldestToNewestEvent());
}, },
); );
} }
@ -406,17 +430,23 @@ class UsersPage extends StatelessWidget {
Text(user.createdTime ?? ''), Text(user.createdTime ?? ''),
Text(user.invitedBy), Text(user.invitedBy),
status( status(
status: user.isEnabled == false ? 'disabled' : user.status, status: user.isEnabled == false
? 'disabled'
: user.status,
), ),
changeIconStatus( changeIconStatus(
status: user.isEnabled == false ? 'disabled' : user.status, status: user.isEnabled == false
? 'disabled'
: user.status,
userId: user.uuid, userId: user.uuid,
onTap: user.status != "invited" onTap: user.status != "invited"
? () { ? () {
context.read<UserTableBloc>().add(ChangeUserStatus( context.read<UserTableBloc>().add(
ChangeUserStatus(
userId: user.uuid, userId: user.uuid,
newStatus: newStatus: user.isEnabled == false
user.isEnabled == false ? 'disabled' : user.status)); ? 'disabled'
: user.status));
} }
: null, : null,
), ),
@ -427,12 +457,15 @@ class UsersPage extends StatelessWidget {
isActive: true, isActive: true,
title: "Edit", title: "Edit",
onTap: () { onTap: () {
context.read<SpaceTreeBloc>().add(ClearCachedData()); context
.read<SpaceTreeBloc>()
.add(ClearCachedData());
showDialog( showDialog(
context: context, context: context,
barrierDismissible: false, barrierDismissible: false,
builder: (BuildContext context) { builder: (BuildContext context) {
return EditUserDialog(userId: user.uuid); return EditUserDialog(
userId: user.uuid);
}, },
).then((v) { ).then((v) {
if (v != null) { if (v != null) {
@ -453,10 +486,13 @@ class UsersPage extends StatelessWidget {
context: context, context: context,
barrierDismissible: false, barrierDismissible: false,
builder: (BuildContext context) { builder: (BuildContext context) {
return DeleteUserDialog(onTapDelete: () async { return DeleteUserDialog(
onTapDelete: () async {
try { try {
_blocRole.add(DeleteUserEvent(user.uuid, context)); _blocRole.add(DeleteUserEvent(
await Future.delayed(const Duration(seconds: 2)); user.uuid, context));
await Future.delayed(
const Duration(seconds: 2));
return true; return true;
} catch (e) { } catch (e) {
return false; return false;
@ -475,6 +511,7 @@ class UsersPage extends StatelessWidget {
]; ];
}).toList(), }).toList(),
), ),
),
Padding( Padding(
padding: const EdgeInsets.all(8.0), padding: const EdgeInsets.all(8.0),
child: Row( child: Row(
@ -486,14 +523,20 @@ class UsersPage extends StatelessWidget {
visiblePagesCount: 4, visiblePagesCount: 4,
buttonRadius: 10, buttonRadius: 10,
selectedButtonColor: ColorsManager.secondaryColor, selectedButtonColor: ColorsManager.secondaryColor,
buttonUnSelectedBorderColor: ColorsManager.grayBorder, buttonUnSelectedBorderColor:
lastPageIcon: const Icon(Icons.keyboard_double_arrow_right), ColorsManager.grayBorder,
firstPageIcon: const Icon(Icons.keyboard_double_arrow_left), lastPageIcon:
totalPages: const Icon(Icons.keyboard_double_arrow_right),
(_blocRole.totalUsersCount.length / _blocRole.itemsPerPage).ceil(), firstPageIcon:
const Icon(Icons.keyboard_double_arrow_left),
totalPages: (_blocRole.totalUsersCount.length /
_blocRole.itemsPerPage)
.ceil(),
currentPage: _blocRole.currentPage, currentPage: _blocRole.currentPage,
onPageChanged: (int pageNumber) { onPageChanged: (int pageNumber) {
context.read<UserTableBloc>().add(ChangePage(pageNumber)); context
.read<UserTableBloc>()
.add(ChangePage(pageNumber));
}, },
), ),
), ),