Added top upstreams and average processing time

This commit is contained in:
Juan Gilsanz Polo 2023-11-25 19:03:26 +01:00
parent 28229311c0
commit 07bd3dcb9a
15 changed files with 441 additions and 188 deletions

View file

@ -5,7 +5,9 @@ import 'package:adguard_home_manager/constants/enums.dart';
final List<HomeTopItems> homeTopItemsDefaultOrder = [ final List<HomeTopItems> homeTopItemsDefaultOrder = [
HomeTopItems.queriedDomains, HomeTopItems.queriedDomains,
HomeTopItems.blockedDomains, HomeTopItems.blockedDomains,
HomeTopItems.recurrentClients HomeTopItems.recurrentClients,
HomeTopItems.topUpstreams,
HomeTopItems.avgUpstreamResponseTime
]; ];
final String homeTopItemsDefaultOrderString = jsonEncode( final String homeTopItemsDefaultOrderString = jsonEncode(

View file

@ -1,2 +1,2 @@
enum LoadStatus { loading, loaded, error } enum LoadStatus { loading, loaded, error }
enum HomeTopItems { queriedDomains, blockedDomains, recurrentClients } enum HomeTopItems { queriedDomains, blockedDomains, recurrentClients, topUpstreams, avgUpstreamResponseTime }

View file

@ -686,5 +686,7 @@
"unsupportedServerVersion": "Unsupported server version", "unsupportedServerVersion": "Unsupported server version",
"unsupportedServerVersionMessage": "Your AdGuard Home server version is too old and is not supported by AdGuard Home Manager. You will need to upgrade your AdGuard Home server to a newer version to use this application.", "unsupportedServerVersionMessage": "Your AdGuard Home server version is too old and is not supported by AdGuard Home Manager. You will need to upgrade your AdGuard Home server to a newer version to use this application.",
"yourVersion": "Your version: {version}", "yourVersion": "Your version: {version}",
"minimumRequiredVersion": "Minimum required version: {version}" "minimumRequiredVersion": "Minimum required version: {version}",
"topUpstreams": "Top upstreams",
"averageUpstreamResponseTime": "Average upstream response time"
} }

View file

@ -686,5 +686,7 @@
"unsupportedServerVersion": "Versión del servidor no soportada", "unsupportedServerVersion": "Versión del servidor no soportada",
"unsupportedServerVersionMessage": "La versión de tu servidor AdGuard Home es demasiado antigua y no está soportada por AdGuard Home Manager. Necesitarás actualizar tu servidor AdGuard Home a una versión más actual para utilizar esta aplicación.", "unsupportedServerVersionMessage": "La versión de tu servidor AdGuard Home es demasiado antigua y no está soportada por AdGuard Home Manager. Necesitarás actualizar tu servidor AdGuard Home a una versión más actual para utilizar esta aplicación.",
"yourVersion": "Tu versión: {version}", "yourVersion": "Tu versión: {version}",
"minimumRequiredVersion": "Versión mínima requerida: {version}" "minimumRequiredVersion": "Versión mínima requerida: {version}",
"topUpstreams": "DNS de subida más frecuentes",
"averageUpstreamResponseTime": "Tiempo promedio de respuesta upstream"
} }

View file

@ -9,6 +9,8 @@ class DnsStatistics {
final List<Map<String, int>> topQueriedDomains; final List<Map<String, int>> topQueriedDomains;
final List<Map<String, int>> topClients; final List<Map<String, int>> topClients;
final List<Map<String, int>> topBlockedDomains; final List<Map<String, int>> topBlockedDomains;
final List<Map<String, int>>? topUpstreamResponses;
final List<Map<String, double>>? topUpstreamsAvgTime;
final List<int> dnsQueries; final List<int> dnsQueries;
final List<int> blockedFiltering; final List<int> blockedFiltering;
final List<int> replacedSafebrowsing; final List<int> replacedSafebrowsing;
@ -25,6 +27,8 @@ class DnsStatistics {
required this.topQueriedDomains, required this.topQueriedDomains,
required this.topClients, required this.topClients,
required this.topBlockedDomains, required this.topBlockedDomains,
required this.topUpstreamResponses,
required this.topUpstreamsAvgTime,
required this.dnsQueries, required this.dnsQueries,
required this.blockedFiltering, required this.blockedFiltering,
required this.replacedSafebrowsing, required this.replacedSafebrowsing,
@ -42,6 +46,8 @@ class DnsStatistics {
topQueriedDomains: List<Map<String, int>>.from(json["top_queried_domains"].map((x) => Map.from(x).map((k, v) => MapEntry<String, int>(k, v)))), topQueriedDomains: List<Map<String, int>>.from(json["top_queried_domains"].map((x) => Map.from(x).map((k, v) => MapEntry<String, int>(k, v)))),
topClients: List<Map<String, int>>.from(json["top_clients"].map((x) => Map.from(x).map((k, v) => MapEntry<String, int>(k, v)))), topClients: List<Map<String, int>>.from(json["top_clients"].map((x) => Map.from(x).map((k, v) => MapEntry<String, int>(k, v)))),
topBlockedDomains: List<Map<String, int>>.from(json["top_blocked_domains"].map((x) => Map.from(x).map((k, v) => MapEntry<String, int>(k, v)))), topBlockedDomains: List<Map<String, int>>.from(json["top_blocked_domains"].map((x) => Map.from(x).map((k, v) => MapEntry<String, int>(k, v)))),
topUpstreamResponses: json["top_upstreams_responses"] != null ? List<Map<String, int>>.from(json["top_upstreams_responses"].map((x) => Map.from(x).map((k, v) => MapEntry<String, int>(k, v)))) : null,
topUpstreamsAvgTime: json["top_upstreams_avg_time"] != null ? List<Map<String, double>>.from(json["top_upstreams_avg_time"].map((x) => Map.from(x).map((k, v) => MapEntry<String, double>(k, v)))) : null,
dnsQueries: List<int>.from(json["dns_queries"].map((x) => x)), dnsQueries: List<int>.from(json["dns_queries"].map((x) => x)),
blockedFiltering: List<int>.from(json["blocked_filtering"].map((x) => x)), blockedFiltering: List<int>.from(json["blocked_filtering"].map((x) => x)),
replacedSafebrowsing: List<int>.from(json["replaced_safebrowsing"].map((x) => x)), replacedSafebrowsing: List<int>.from(json["replaced_safebrowsing"].map((x) => x)),
@ -59,6 +65,8 @@ class DnsStatistics {
"top_queried_domains": List<dynamic>.from(topQueriedDomains.map((x) => Map.from(x).map((k, v) => MapEntry<String, dynamic>(k, v)))), "top_queried_domains": List<dynamic>.from(topQueriedDomains.map((x) => Map.from(x).map((k, v) => MapEntry<String, dynamic>(k, v)))),
"top_clients": List<dynamic>.from(topClients.map((x) => Map.from(x).map((k, v) => MapEntry<String, dynamic>(k, v)))), "top_clients": List<dynamic>.from(topClients.map((x) => Map.from(x).map((k, v) => MapEntry<String, dynamic>(k, v)))),
"top_blocked_domains": List<dynamic>.from(topBlockedDomains.map((x) => Map.from(x).map((k, v) => MapEntry<String, dynamic>(k, v)))), "top_blocked_domains": List<dynamic>.from(topBlockedDomains.map((x) => Map.from(x).map((k, v) => MapEntry<String, dynamic>(k, v)))),
"top_upstreams_responses": topUpstreamResponses != null ? List<dynamic>.from(topUpstreamResponses!.map((x) => Map.from(x).map((k, v) => MapEntry<String, dynamic>(k, v)))) : null,
"top_upstreams_avg_time": topUpstreamsAvgTime != null ? List<dynamic>.from(topUpstreamsAvgTime!.map((x) => Map.from(x).map((k, v) => MapEntry<String, dynamic>(k, v)))) : null,
"dns_queries": List<dynamic>.from(dnsQueries.map((x) => x)), "dns_queries": List<dynamic>.from(dnsQueries.map((x) => x)),
"blocked_filtering": List<dynamic>.from(blockedFiltering.map((x) => x)), "blocked_filtering": List<dynamic>.from(blockedFiltering.map((x) => x)),
"replaced_safebrowsing": List<dynamic>.from(replacedSafebrowsing.map((x) => x)), "replaced_safebrowsing": List<dynamic>.from(replacedSafebrowsing.map((x) => x)),

View file

@ -449,7 +449,7 @@ class AppConfigProvider with ChangeNotifier {
_showTopItemsChart = dbData['showTopItemsChart']; _showTopItemsChart = dbData['showTopItemsChart'];
if (dbData['homeTopItemsOrder'] != null) { if (dbData['homeTopItemsOrder'] != null) {
try { try {
_homeTopItemsOrder = List<HomeTopItems>.from( final itemsOrder = List<HomeTopItems>.from(
List<String>.from(jsonDecode(dbData['homeTopItemsOrder'])).map((e) { List<String>.from(jsonDecode(dbData['homeTopItemsOrder'])).map((e) {
switch (e) { switch (e) {
case 'queriedDomains': case 'queriedDomains':
@ -461,11 +461,22 @@ class AppConfigProvider with ChangeNotifier {
case 'recurrentClients': case 'recurrentClients':
return HomeTopItems.recurrentClients; return HomeTopItems.recurrentClients;
case 'topUpstreams':
return HomeTopItems.topUpstreams;
case 'avgUpstreamResponseTime':
return HomeTopItems.avgUpstreamResponseTime;
default: default:
return null; return null;
} }
}).where((e) => e != null).toList() }).where((e) => e != null).toList()
); );
final missingItems = homeTopItemsDefaultOrder.where((e) => !itemsOrder.contains(e));
_homeTopItemsOrder = [
...itemsOrder,
...missingItems
];
} catch (e) { } catch (e) {
Sentry.captureException(e); Sentry.captureException(e);
_homeTopItemsOrder = homeTopItemsDefaultOrder; _homeTopItemsOrder = homeTopItemsDefaultOrder;

View file

@ -237,7 +237,7 @@ class StatusProvider with ChangeNotifier {
} }
Future<bool> getServerStatus({ Future<bool> getServerStatus({
bool? withLoadingIndicator, bool? withLoadingIndicator = true,
bool? overrideCheckServerVersion bool? overrideCheckServerVersion
}) async { }) async {
if (withLoadingIndicator == true) { if (withLoadingIndicator == true) {

View file

@ -32,8 +32,11 @@ class _HomeState extends State<Home> {
late bool isVisible; late bool isVisible;
@override @override
initState(){ initState() {
Provider.of<StatusProvider>(context, listen: false).getServerStatus(); final statusProvider = Provider.of<StatusProvider>(context, listen: false);
statusProvider.getServerStatus(
withLoadingIndicator: statusProvider.serverStatus != null ? false : true
);
super.initState(); super.initState();
@ -239,9 +242,9 @@ class TopItemsLists extends StatelessWidget {
final List<HomeTopItems> order; final List<HomeTopItems> order;
const TopItemsLists({ const TopItemsLists({
Key? key, super.key,
required this.order, required this.order,
}) : super(key: key); });
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -266,8 +269,7 @@ class TopItemsLists extends StatelessWidget {
children: [ children: [
TopItems( TopItems(
label: AppLocalizations.of(context)!.topQueriedDomains, label: AppLocalizations.of(context)!.topQueriedDomains,
data: statusProvider.serverStatus!.stats.topQueriedDomains, type: HomeTopItems.queriedDomains,
type: 'topQueriedDomains',
), ),
if (item.key < order.length - 1) ...bottom if (item.key < order.length - 1) ...bottom
], ],
@ -278,8 +280,7 @@ class TopItemsLists extends StatelessWidget {
children: [ children: [
TopItems( TopItems(
label: AppLocalizations.of(context)!.topBlockedDomains, label: AppLocalizations.of(context)!.topBlockedDomains,
data: statusProvider.serverStatus!.stats.topBlockedDomains, type: HomeTopItems.blockedDomains,
type: 'topBlockedDomains',
), ),
if (item.key < order.length - 1) ...bottom if (item.key < order.length - 1) ...bottom
], ],
@ -290,13 +291,37 @@ class TopItemsLists extends StatelessWidget {
children: [ children: [
TopItems( TopItems(
label: AppLocalizations.of(context)!.topClients, label: AppLocalizations.of(context)!.topClients,
data: statusProvider.serverStatus!.stats.topClients, type: HomeTopItems.recurrentClients,
type: 'topClients',
clients: true,
), ),
if (item.key < order.length - 1) ...bottom if (item.key < order.length - 1) ...bottom
], ],
); );
case HomeTopItems.topUpstreams:
return statusProvider.serverStatus!.stats.topUpstreamResponses != null
? Column(
children: [
TopItems(
label: AppLocalizations.of(context)!.topUpstreams,
type: HomeTopItems.topUpstreams,
),
if (item.key < order.length - 1) ...bottom
],
)
: const SizedBox();
case HomeTopItems.avgUpstreamResponseTime:
return statusProvider.serverStatus!.stats.topUpstreamsAvgTime != null
? Column(
children: [
TopItems(
label: AppLocalizations.of(context)!.averageUpstreamResponseTime,
type: HomeTopItems.avgUpstreamResponseTime,
),
if (item.key < order.length - 1) ...bottom
],
)
: const SizedBox();
default: default:
return const SizedBox(); return const SizedBox();

View file

@ -4,28 +4,31 @@ import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:adguard_home_manager/widgets/domain_options.dart'; import 'package:adguard_home_manager/widgets/domain_options.dart';
import 'package:adguard_home_manager/constants/enums.dart';
import 'package:adguard_home_manager/models/applied_filters.dart'; import 'package:adguard_home_manager/models/applied_filters.dart';
import 'package:adguard_home_manager/providers/app_config_provider.dart'; import 'package:adguard_home_manager/providers/app_config_provider.dart';
import 'package:adguard_home_manager/providers/logs_provider.dart'; import 'package:adguard_home_manager/providers/logs_provider.dart';
import 'package:adguard_home_manager/providers/status_provider.dart'; import 'package:adguard_home_manager/providers/status_provider.dart';
class RowItem extends StatefulWidget { class RowItem extends StatefulWidget {
final String type; final HomeTopItems type;
final Color chartColor; final Color chartColor;
final String domain; final String domain;
final String number; final String number;
final bool clients; final bool clients;
final bool showColor; final bool showColor;
final String? unit;
const RowItem({ const RowItem({
Key? key, super.key,
required this.type, required this.type,
required this.chartColor, required this.chartColor,
required this.domain, required this.domain,
required this.number, required this.number,
required this.clients, required this.clients,
required this.showColor, required this.showColor,
}) : super(key: key); this.unit,
});
@override @override
State<RowItem> createState() => _RowItemState(); State<RowItem> createState() => _RowItemState();
@ -93,10 +96,10 @@ class _RowItemState extends State<RowItem> with TickerProviderStateMixin {
color: Colors.transparent, color: Colors.transparent,
child: DomainOptions( child: DomainOptions(
item: widget.domain, item: widget.domain,
isClient: widget.type == 'topClients', isDomain: widget.type == HomeTopItems.queriedDomains || widget.type == HomeTopItems.blockedDomains,
isBlocked: widget.type == 'topBlockedDomains', isBlocked: widget.type == HomeTopItems.blockedDomains,
onTap: () { onTap: () {
if (widget.type == 'topQueriedDomains' || widget.type == 'topBlockedDomains') { if (widget.type == HomeTopItems.queriedDomains || widget.type == HomeTopItems.blockedDomains) {
logsProvider.setSearchText(widget.domain); logsProvider.setSearchText(widget.domain);
logsProvider.setSelectedClients(null); logsProvider.setSelectedClients(null);
logsProvider.setAppliedFilters( logsProvider.setAppliedFilters(
@ -108,7 +111,7 @@ class _RowItemState extends State<RowItem> with TickerProviderStateMixin {
); );
appConfigProvider.setSelectedScreen(2); appConfigProvider.setSelectedScreen(2);
} }
else if (widget.type == 'topClients') { else if (widget.type == HomeTopItems.recurrentClients) {
logsProvider.setSearchText(null); logsProvider.setSearchText(null);
logsProvider.setSelectedClients([widget.domain]); logsProvider.setSelectedClients([widget.domain]);
logsProvider.setAppliedFilters( logsProvider.setAppliedFilters(
@ -195,10 +198,10 @@ class OthersRowItem extends StatefulWidget {
final bool showColor; final bool showColor;
const OthersRowItem({ const OthersRowItem({
Key? key, super.key,
required this.items, required this.items,
required this.showColor, required this.showColor,
}) : super(key: key); });
@override @override
State<OthersRowItem> createState() => _OthersRowItemState(); State<OthersRowItem> createState() => _OthersRowItemState();

View file

@ -0,0 +1,193 @@
import 'package:flutter/material.dart';
import 'package:adguard_home_manager/widgets/custom_pie_chart.dart';
import 'package:adguard_home_manager/screens/home/top_items/row_item.dart';
import 'package:adguard_home_manager/constants/enums.dart';
class TopItemExpansionPanel extends StatefulWidget {
final HomeTopItems type;
final String label;
final List<Map<String, dynamic>> data;
final Map<String, double> chartData;
final bool withChart;
const TopItemExpansionPanel({
super.key,
required this.type,
required this.label,
required this.data,
required this.chartData,
required this.withChart
});
@override
State<TopItemExpansionPanel> createState() => _TopItemExpansionPanelState();
}
class _TopItemExpansionPanelState extends State<TopItemExpansionPanel> {
bool _showChart = true;
final colors = [
Colors.red,
Colors.green,
Colors.blue,
Colors.orange,
Colors.teal,
Colors.grey
];
@override
Widget build(BuildContext context) {
final width = MediaQuery.of(context).size.width;
if (widget.withChart == true) {
return Column(
children: [
ExpansionPanelList(
expandedHeaderPadding: const EdgeInsets.all(0),
elevation: 0,
expansionCallback: (_, isExpanded) => setState(() => _showChart = isExpanded),
animationDuration: const Duration(milliseconds: 250),
children: [
ExpansionPanel(
headerBuilder: (context, isExpanded) => Padding(
padding: const EdgeInsets.symmetric(horizontal: 20),
child: Row(
mainAxisAlignment: width <= 700
? MainAxisAlignment.spaceBetween
: MainAxisAlignment.center,
children: [
Flexible(
child: Text(
widget.label,
overflow: TextOverflow.ellipsis,
style: TextStyle(
fontSize: 18,
fontWeight: FontWeight.w500,
color: Theme.of(context).colorScheme.onSurface
),
),
),
],
),
),
body: Padding(
padding: const EdgeInsets.symmetric(vertical: 16),
child: Column(
children: [
SizedBox(
height: 150,
child: CustomPieChart(
data: widget.chartData,
colors: colors
)
),
const SizedBox(height: 16),
],
),
),
isExpanded: _showChart
),
],
),
Padding(
padding: const EdgeInsets.only(top: 8),
child: _ItemsList(
colors: colors,
data: widget.data,
clients: widget.type == HomeTopItems.recurrentClients,
type: widget.type,
showChart: _showChart,
unit: widget.type == HomeTopItems.avgUpstreamResponseTime ? 'ms' : null,
),
),
if (widget.type != HomeTopItems.avgUpstreamResponseTime) OthersRowItem(
items: widget.data,
showColor: _showChart,
),
const SizedBox(height: 16),
],
);
}
else {
return Column(
children: [
Padding(
padding: const EdgeInsets.symmetric(horizontal: 18),
child: Row(
mainAxisAlignment: width <= 700
? MainAxisAlignment.spaceBetween
: MainAxisAlignment.center,
children: [
Flexible(
child: Text(
widget.label,
overflow: TextOverflow.ellipsis,
style: TextStyle(
fontSize: 18,
fontWeight: FontWeight.w500,
color: Theme.of(context).colorScheme.onSurface
),
),
),
],
),
),
Padding(
padding: const EdgeInsets.only(top: 16),
child: _ItemsList(
colors: colors,
data: widget.data,
clients: widget.type == HomeTopItems.recurrentClients,
type: widget.type,
showChart: false,
unit: widget.type == HomeTopItems.avgUpstreamResponseTime ? 'ms' : null,
),
),
if (widget.type != HomeTopItems.avgUpstreamResponseTime) OthersRowItem(
items: widget.data,
showColor: false,
),
const SizedBox(height: 16),
],
);
}
}
}
class _ItemsList extends StatelessWidget {
final List<Color> colors;
final List<Map<String, dynamic>> data;
final bool? clients;
final HomeTopItems type;
final bool showChart;
final String? unit;
const _ItemsList({
required this.colors,
required this.data,
required this.clients,
required this.type,
required this.showChart,
this.unit,
});
@override
Widget build(BuildContext context) {
return Column(
children: data.sublist(
0, data.length > 5 ? 5 : data.length
).asMap().entries.map((e) => RowItem(
clients: clients ?? false,
domain: e.value.keys.toList()[0],
number: e.value.values.toList()[0].runtimeType == double
? "${e.value.values.toList()[0].toStringAsFixed(2)}${unit != null ? ' $unit' : ''}"
: "${e.value.values.toList()[0].toString()}${unit != null ? ' $unit' : ''}",
type: type,
chartColor: colors[e.key],
showColor: showChart,
)).toList()
);
}
}

View file

@ -7,26 +7,24 @@ import 'package:provider/provider.dart';
import 'package:flutter_gen/gen_l10n/app_localizations.dart'; import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:adguard_home_manager/screens/home/top_items/row_item.dart'; import 'package:adguard_home_manager/screens/home/top_items/row_item.dart';
import 'package:adguard_home_manager/widgets/custom_pie_chart.dart'; import 'package:adguard_home_manager/screens/home/top_items/top_item_expansion_panel.dart';
import 'package:adguard_home_manager/screens/top_items/top_items_modal.dart'; import 'package:adguard_home_manager/screens/top_items/top_items_modal.dart';
import 'package:adguard_home_manager/screens/top_items/top_items.dart'; import 'package:adguard_home_manager/screens/top_items/top_items.dart';
import 'package:adguard_home_manager/widgets/custom_pie_chart.dart';
import 'package:adguard_home_manager/constants/enums.dart';
import 'package:adguard_home_manager/providers/status_provider.dart'; import 'package:adguard_home_manager/providers/status_provider.dart';
import 'package:adguard_home_manager/providers/app_config_provider.dart'; import 'package:adguard_home_manager/providers/app_config_provider.dart';
class TopItems extends StatefulWidget { class TopItems extends StatefulWidget {
final String type; final HomeTopItems type;
final String label; final String label;
final List<Map<String, dynamic>> data;
final bool? clients;
const TopItems({ const TopItems({
Key? key, super.key,
required this.type, required this.type,
required this.label, required this.label,
required this.data, });
this.clients
}) : super(key: key);
@override @override
State<TopItems> createState() => _TopItemsState(); State<TopItems> createState() => _TopItemsState();
@ -58,31 +56,41 @@ class _TopItemsState extends State<TopItems> {
List<Map<String, dynamic>> generateData() { List<Map<String, dynamic>> generateData() {
switch (widget.type) { switch (widget.type) {
case 'topQueriedDomains': case HomeTopItems.queriedDomains:
return statusProvider.serverStatus!.stats.topQueriedDomains; return statusProvider.serverStatus!.stats.topQueriedDomains;
case 'topBlockedDomains': case HomeTopItems.blockedDomains:
return statusProvider.serverStatus!.stats.topBlockedDomains; return statusProvider.serverStatus!.stats.topBlockedDomains;
case 'topClients': case HomeTopItems.recurrentClients:
return statusProvider.serverStatus!.stats.topClients; return statusProvider.serverStatus!.stats.topClients;
case HomeTopItems.topUpstreams:
return statusProvider.serverStatus!.stats.topUpstreamResponses ?? [];
case HomeTopItems.avgUpstreamResponseTime:
return statusProvider.serverStatus!.stats.topUpstreamsAvgTime ?? [];
default: default:
return []; return [];
} }
} }
final data = generateData();
final withChart = widget.type != HomeTopItems.avgUpstreamResponseTime;
Map<String, double> chartData() { Map<String, double> chartData() {
Map<String, double> values = {}; Map<String, double> values = {};
widget.data.sublist(0, widget.data.length > 5 ? 5 : widget.data.length).forEach((element) { data.sublist(0, data.length > 5 ? 5 : data.length).forEach((element) {
values = { values = {
...values, ...values,
element.keys.first: element.values.first.toDouble() element.keys.first: element.values.first.toDouble()
}; };
}); });
if (widget.data.length > 5) { if (data.length > 5) {
final int rest = List<int>.from( final int rest = List<int>.from(
widget.data.sublist(5, widget.data.length).map((e) => e.values.first.toInt()) data.sublist(5, data.length).map((e) => e.values.first.toInt())
).reduce((a, b) => a + b); ).reduce((a, b) => a + b);
values = { values = {
...values, ...values,
@ -109,108 +117,71 @@ class _TopItemsState extends State<TopItems> {
return SizedBox( return SizedBox(
child: Column( child: Column(
children: [ children: [
if (widget.data.isEmpty) noItems, if (data.isEmpty) noItems,
if (widget.data.isNotEmpty && width > 700) Row( if (data.isNotEmpty && width > 700) Padding(
mainAxisAlignment: MainAxisAlignment.spaceEvenly, padding: EdgeInsets.only(bottom: withChart == false ? 16 : 0),
children: [ child: Row(
Expanded( mainAxisAlignment: MainAxisAlignment.spaceEvenly,
flex: 1,
child: ConstrainedBox(
constraints: const BoxConstraints(
maxHeight: 250
),
child: Padding(
padding: const EdgeInsets.all(16),
child: CustomPieChart(
data: chartData(),
colors: colors
)
),
)
),
Expanded(
flex: 2,
child: Column(
children: [
ItemsList(
colors: colors,
data: widget.data,
clients: widget.clients,
type: widget.type,
showChart: _showChart
),
OthersRowItem(
items: widget.data,
showColor: true,
)
]
),
)
],
),
if (widget.data.isNotEmpty && width <= 700) ...[
ExpansionPanelList(
expandedHeaderPadding: const EdgeInsets.all(0),
elevation: 0,
expansionCallback: (_, isExpanded) => setState(() => _showChart = isExpanded),
animationDuration: const Duration(milliseconds: 250),
children: [ children: [
ExpansionPanel( if (withChart == true) Expanded(
headerBuilder: (context, isExpanded) => Padding( flex: 1,
padding: const EdgeInsets.symmetric(horizontal: 24), child: ConstrainedBox(
child: Row( constraints: const BoxConstraints(
mainAxisAlignment: width <= 700 maxHeight: 250
? MainAxisAlignment.spaceBetween ),
: MainAxisAlignment.center, child: Padding(
children: [ padding: const EdgeInsets.all(16),
Text( child: CustomPieChart(
data: chartData(),
colors: colors
)
),
)
),
Expanded(
flex: 2,
child: Column(
children: [
Padding(
padding: const EdgeInsets.only(
top: 8,
bottom: 16
),
child: Text(
widget.label, widget.label,
style: TextStyle( style: const TextStyle(
fontSize: 18, fontSize: 18,
fontWeight: FontWeight.w500, fontWeight: FontWeight.w500
color: Theme.of(context).colorScheme.onSurface
), ),
), ),
], ),
), _ItemsList(
colors: colors,
data: data,
clients: widget.type == HomeTopItems.recurrentClients,
type: widget.type,
showChart: withChart == true ? _showChart : false,
unit: widget.type == HomeTopItems.avgUpstreamResponseTime ? 'ms' : null,
),
if (withChart == true) OthersRowItem(
items: data,
showColor: true,
)
]
), ),
body: Padding( )
padding: const EdgeInsets.symmetric(vertical: 16),
child: Column(
children: [
SizedBox(
height: 150,
child: CustomPieChart(
data: chartData(),
colors: colors
)
),
const SizedBox(height: 16),
],
),
),
isExpanded: _showChart
),
], ],
), ),
Padding( ),
padding: const EdgeInsets.only(top: 8), if (data.isNotEmpty && width <= 700) TopItemExpansionPanel(
child: ItemsList( type: widget.type,
colors: colors, label: widget.label,
data: widget.data, data: data,
clients: widget.clients, chartData: chartData(),
type: widget.type, withChart: withChart
showChart: _showChart ),
),
),
OthersRowItem(
items: widget.data,
showColor: _showChart,
),
const SizedBox(height: 16),
],
if (widget.data.length > 5) ...[ if (data.length > 5) ...[
Padding( Padding(
padding: const EdgeInsets.only(right: 20), padding: const EdgeInsets.only(right: 20),
child: Row( child: Row(
@ -225,8 +196,10 @@ class _TopItemsState extends State<TopItems> {
builder: (context) => TopItemsModal( builder: (context) => TopItemsModal(
type: widget.type, type: widget.type,
title: widget.label, title: widget.label,
isClient: widget.clients, isClient: widget.type == HomeTopItems.recurrentClients,
data: generateData(), data: generateData(),
withProgressBar: widget.type != HomeTopItems.avgUpstreamResponseTime,
unit: widget.type == HomeTopItems.avgUpstreamResponseTime ? 'ms' : null,
) )
) )
} }
@ -236,8 +209,10 @@ class _TopItemsState extends State<TopItems> {
builder: (context) => TopItemsScreen( builder: (context) => TopItemsScreen(
type: widget.type, type: widget.type,
title: widget.label, title: widget.label,
isClient: widget.clients, isClient: widget.type == HomeTopItems.recurrentClients,
data: generateData(), data: generateData(),
withProgressBar: widget.type != HomeTopItems.avgUpstreamResponseTime,
unit: widget.type == HomeTopItems.avgUpstreamResponseTime ? 'ms' : null,
) )
) )
) )
@ -266,21 +241,22 @@ class _TopItemsState extends State<TopItems> {
} }
} }
class ItemsList extends StatelessWidget { class _ItemsList extends StatelessWidget {
final List<Color> colors; final List<Color> colors;
final List<Map<String, dynamic>> data; final List<Map<String, dynamic>> data;
final bool? clients; final bool? clients;
final String type; final HomeTopItems type;
final bool showChart; final bool showChart;
final String? unit;
const ItemsList({ const _ItemsList({
Key? key,
required this.colors, required this.colors,
required this.data, required this.data,
required this.clients, required this.clients,
required this.type, required this.type,
required this.showChart, required this.showChart,
}) : super(key: key); this.unit,
});
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -290,7 +266,9 @@ class ItemsList extends StatelessWidget {
).asMap().entries.map((e) => RowItem( ).asMap().entries.map((e) => RowItem(
clients: clients ?? false, clients: clients ?? false,
domain: e.value.keys.toList()[0], domain: e.value.keys.toList()[0],
number: e.value.values.toList()[0].toString(), number: e.value.values.toList()[0].runtimeType == double
? "${e.value.values.toList()[0].toStringAsFixed(2)}${unit != null ? ' $unit' : ''}"
: "${e.value.values.toList()[0].toString()}${unit != null ? ' $unit' : ''}",
type: type, type: type,
chartColor: colors[e.key], chartColor: colors[e.key],
showColor: showChart, showColor: showChart,

View file

@ -1,6 +1,5 @@
// ignore_for_file: use_build_context_synchronously // ignore_for_file: use_build_context_synchronously
import 'package:adguard_home_manager/functions/desktop_mode.dart';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
@ -9,15 +8,16 @@ import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:adguard_home_manager/widgets/custom_list_tile.dart'; import 'package:adguard_home_manager/widgets/custom_list_tile.dart';
import 'package:adguard_home_manager/functions/desktop_mode.dart';
import 'package:adguard_home_manager/functions/snackbar.dart'; import 'package:adguard_home_manager/functions/snackbar.dart';
import 'package:adguard_home_manager/constants/enums.dart'; import 'package:adguard_home_manager/constants/enums.dart';
import 'package:adguard_home_manager/providers/app_config_provider.dart'; import 'package:adguard_home_manager/providers/app_config_provider.dart';
class ItemData { class _ItemData {
final HomeTopItems title; final HomeTopItems title;
final Key key; final Key key;
const ItemData({ const _ItemData({
required this.title, required this.title,
required this.key required this.key
}); });
@ -29,7 +29,7 @@ enum DraggingMode {
} }
class ReorderableTopItemsHome extends StatefulWidget { class ReorderableTopItemsHome extends StatefulWidget {
const ReorderableTopItemsHome({Key? key}) : super(key: key); const ReorderableTopItemsHome({super.key});
@override @override
State<ReorderableTopItemsHome> createState() => _ReorderableTopItemsHomeState(); State<ReorderableTopItemsHome> createState() => _ReorderableTopItemsHomeState();
@ -38,10 +38,10 @@ class ReorderableTopItemsHome extends StatefulWidget {
class _ReorderableTopItemsHomeState extends State<ReorderableTopItemsHome> { class _ReorderableTopItemsHomeState extends State<ReorderableTopItemsHome> {
List<HomeTopItems> homeTopItemsList = []; List<HomeTopItems> homeTopItemsList = [];
List<HomeTopItems> persistHomeTopItemsList = []; List<HomeTopItems> persistHomeTopItemsList = [];
List<ItemData> renderItems = []; List<_ItemData> renderItems = [];
int _indexOfKey(Key key) { int _indexOfKey(Key key) {
return renderItems.indexWhere((ItemData d) => d.key == key); return renderItems.indexWhere((_ItemData d) => d.key == key);
} }
bool _reorderCallback(Key item, Key newPosition) { bool _reorderCallback(Key item, Key newPosition) {
@ -79,7 +79,7 @@ class _ReorderableTopItemsHomeState extends State<ReorderableTopItemsHome> {
homeTopItemsList = appConfigProvider.homeTopItemsOrder; homeTopItemsList = appConfigProvider.homeTopItemsOrder;
persistHomeTopItemsList = appConfigProvider.homeTopItemsOrder; persistHomeTopItemsList = appConfigProvider.homeTopItemsOrder;
renderItems = appConfigProvider.homeTopItemsOrder.asMap().entries.map( renderItems = appConfigProvider.homeTopItemsOrder.asMap().entries.map(
(e) => ItemData( (e) => _ItemData(
key: ValueKey(e.key), key: ValueKey(e.key),
title: e.value, title: e.value,
) )
@ -117,16 +117,31 @@ class _ReorderableTopItemsHomeState extends State<ReorderableTopItemsHome> {
padding: const EdgeInsets.all(16) padding: const EdgeInsets.all(16)
); );
case HomeTopItems.topUpstreams:
return CustomListTile(
title: AppLocalizations.of(context)!.topUpstreams,
icon: Icons.upload_file_rounded,
padding: const EdgeInsets.all(16)
);
case HomeTopItems.avgUpstreamResponseTime:
return CustomListTile(
title: AppLocalizations.of(context)!.averageUpstreamResponseTime,
icon: Icons.timer_rounded,
padding: const EdgeInsets.all(16)
);
default: default:
return const SizedBox(); return const SizedBox();
} }
} }
Future<bool> onWillPopScope() async { Future<bool> onWillPopScope(bool popInvoked) async {
if (!listEquals(appConfigProvider.homeTopItemsOrder, persistHomeTopItemsList)) { if (!listEquals(appConfigProvider.homeTopItemsOrder, persistHomeTopItemsList)) {
showDialog( await showDialog(
context: context, context: context,
builder: (dialogContext) => AlertDialog( useRootNavigator: false,
builder: (ctx) => AlertDialog(
title: Text(AppLocalizations.of(context)!.discardChanges), title: Text(AppLocalizations.of(context)!.discardChanges),
content: Text(AppLocalizations.of(context)!.discardChangesDescription), content: Text(AppLocalizations.of(context)!.discardChangesDescription),
actions: [ actions: [
@ -135,14 +150,14 @@ class _ReorderableTopItemsHomeState extends State<ReorderableTopItemsHome> {
children: [ children: [
TextButton( TextButton(
onPressed: () { onPressed: () {
Navigator.pop(dialogContext); Navigator.pop(context);
Navigator.pop(context); Navigator.pop(context);
}, },
child: Text(AppLocalizations.of(context)!.confirm) child: Text(AppLocalizations.of(context)!.confirm)
), ),
const SizedBox(width: 8), const SizedBox(width: 8),
TextButton( TextButton(
onPressed: () => Navigator.pop(dialogContext), onPressed: () => Navigator.pop(context),
child: Text(AppLocalizations.of(context)!.cancel) child: Text(AppLocalizations.of(context)!.cancel)
), ),
], ],
@ -175,8 +190,9 @@ class _ReorderableTopItemsHomeState extends State<ReorderableTopItemsHome> {
} }
} }
return WillPopScope( return PopScope(
onWillPop: onWillPopScope, canPop: listEquals(appConfigProvider.homeTopItemsOrder, persistHomeTopItemsList),
onPopInvoked: onWillPopScope,
child: Scaffold( child: Scaffold(
appBar: AppBar( appBar: AppBar(
title: Text(AppLocalizations.of(context)!.topItemsOrder), title: Text(AppLocalizations.of(context)!.topItemsOrder),
@ -219,7 +235,7 @@ class _ReorderableTopItemsHomeState extends State<ReorderableTopItemsHome> {
child: ListView.builder( child: ListView.builder(
itemBuilder: (context, index) => reorderable_list_library.ReorderableItem( itemBuilder: (context, index) => reorderable_list_library.ReorderableItem(
key: renderItems[index].key, key: renderItems[index].key,
childBuilder: (context, state) => Item( childBuilder: (context, state) => _Item(
tileWidget: tile(renderItems[index].title), tileWidget: tile(renderItems[index].title),
isFirst: index == 0, isFirst: index == 0,
isLast: index == renderItems.length - 1, isLast: index == renderItems.length - 1,
@ -237,19 +253,18 @@ class _ReorderableTopItemsHomeState extends State<ReorderableTopItemsHome> {
} }
} }
class Item extends StatelessWidget { class _Item extends StatelessWidget {
final Widget tileWidget; final Widget tileWidget;
final bool isFirst; final bool isFirst;
final bool isLast; final bool isLast;
final reorderable_list_library.ReorderableItemState state; final reorderable_list_library.ReorderableItemState state;
const Item({ const _Item({
Key? key,
required this.tileWidget, required this.tileWidget,
required this.isFirst, required this.isFirst,
required this.isLast, required this.isLast,
required this.state, required this.state,
}) : super(key: key); });
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {

View file

@ -10,6 +10,7 @@ import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:adguard_home_manager/widgets/domain_options.dart'; import 'package:adguard_home_manager/widgets/domain_options.dart';
import 'package:adguard_home_manager/widgets/custom_list_tile.dart'; import 'package:adguard_home_manager/widgets/custom_list_tile.dart';
import 'package:adguard_home_manager/constants/enums.dart';
import 'package:adguard_home_manager/models/applied_filters.dart'; import 'package:adguard_home_manager/models/applied_filters.dart';
import 'package:adguard_home_manager/providers/logs_provider.dart'; import 'package:adguard_home_manager/providers/logs_provider.dart';
import 'package:adguard_home_manager/providers/status_provider.dart'; import 'package:adguard_home_manager/providers/status_provider.dart';
@ -18,18 +19,22 @@ import 'package:adguard_home_manager/functions/number_format.dart';
import 'package:adguard_home_manager/providers/app_config_provider.dart'; import 'package:adguard_home_manager/providers/app_config_provider.dart';
class TopItemsScreen extends StatefulWidget { class TopItemsScreen extends StatefulWidget {
final String type; final HomeTopItems type;
final String title; final String title;
final bool? isClient; final bool? isClient;
final List<Map<String, dynamic>> data; final List<Map<String, dynamic>> data;
final bool withProgressBar;
final String? unit;
const TopItemsScreen({ const TopItemsScreen({
Key? key, super.key,
required this.type, required this.type,
required this.title, required this.title,
this.isClient, this.isClient,
required this.data, required this.data,
}) : super(key: key); required this.withProgressBar,
this.unit,
});
@override @override
State<TopItemsScreen> createState() => _TopItemsScreenState(); State<TopItemsScreen> createState() => _TopItemsScreenState();
@ -60,9 +65,9 @@ class _TopItemsScreenState extends State<TopItemsScreen> {
final appConfigProvider = Provider.of<AppConfigProvider>(context); final appConfigProvider = Provider.of<AppConfigProvider>(context);
final logsProvider = Provider.of<LogsProvider>(context); final logsProvider = Provider.of<LogsProvider>(context);
int total = 0; double total = 0;
for (var element in data) { for (var element in data) {
total = total + int.parse(element.values.toList()[0].toString()); total = total + double.parse(element.values.toList()[0].toString());
} }
return Scaffold( return Scaffold(
@ -154,10 +159,10 @@ class _TopItemsScreenState extends State<TopItemsScreen> {
return DomainOptions( return DomainOptions(
item: screenData[index].keys.toList()[0], item: screenData[index].keys.toList()[0],
isBlocked: widget.type == 'topBlockedDomains', isBlocked: widget.type == HomeTopItems.blockedDomains,
isClient: widget.type == 'topClients', isDomain: widget.type == HomeTopItems.queriedDomains || widget.type == HomeTopItems.blockedDomains,
onTap: () { onTap: () {
if (widget.type == 'topQueriedDomains' || widget.type == 'topBlockedDomains') { if (widget.type == HomeTopItems.queriedDomains || widget.type == HomeTopItems.blockedDomains) {
logsProvider.setSearchText(screenData[index].keys.toList()[0]); logsProvider.setSearchText(screenData[index].keys.toList()[0]);
logsProvider.setSelectedClients(null); logsProvider.setSelectedClients(null);
logsProvider.setAppliedFilters( logsProvider.setAppliedFilters(
@ -170,7 +175,7 @@ class _TopItemsScreenState extends State<TopItemsScreen> {
appConfigProvider.setSelectedScreen(2); appConfigProvider.setSelectedScreen(2);
Navigator.pop(context); Navigator.pop(context);
} }
else if (widget.type == 'topClients') { else if (widget.type == HomeTopItems.recurrentClients) {
logsProvider.setSearchText(null); logsProvider.setSearchText(null);
logsProvider.setSelectedClients([screenData[index].keys.toList()[0]]); logsProvider.setSelectedClients([screenData[index].keys.toList()[0]]);
logsProvider.setAppliedFilters( logsProvider.setAppliedFilters(
@ -187,7 +192,9 @@ class _TopItemsScreenState extends State<TopItemsScreen> {
child: CustomListTile( child: CustomListTile(
title: screenData[index].keys.toList()[0], title: screenData[index].keys.toList()[0],
trailing: Text( trailing: Text(
screenData[index].values.toList()[0].toString(), screenData[index].values.toList()[0].runtimeType == double
? "${screenData[index].values.toList()[0].toStringAsFixed(2)}${widget.unit != null ? ' ${widget.unit}' : ''}"
: "${screenData[index].values.toList()[0].toString()}${widget.unit != null ? ' ${widget.unit}' : ''}",
style: TextStyle( style: TextStyle(
color: Theme.of(context).colorScheme.onSurfaceVariant color: Theme.of(context).colorScheme.onSurfaceVariant
), ),
@ -205,7 +212,7 @@ class _TopItemsScreenState extends State<TopItemsScreen> {
), ),
const SizedBox(height: 5), const SizedBox(height: 5),
], ],
Row( if (widget.withProgressBar == true) Row(
children: [ children: [
SizedBox( SizedBox(
width: 50, width: 50,

View file

@ -10,6 +10,7 @@ import 'package:flutter_gen/gen_l10n/app_localizations.dart';
import 'package:adguard_home_manager/widgets/domain_options.dart'; import 'package:adguard_home_manager/widgets/domain_options.dart';
import 'package:adguard_home_manager/widgets/custom_list_tile.dart'; import 'package:adguard_home_manager/widgets/custom_list_tile.dart';
import 'package:adguard_home_manager/constants/enums.dart';
import 'package:adguard_home_manager/models/applied_filters.dart'; import 'package:adguard_home_manager/models/applied_filters.dart';
import 'package:adguard_home_manager/providers/logs_provider.dart'; import 'package:adguard_home_manager/providers/logs_provider.dart';
import 'package:adguard_home_manager/functions/number_format.dart'; import 'package:adguard_home_manager/functions/number_format.dart';
@ -17,18 +18,22 @@ import 'package:adguard_home_manager/providers/status_provider.dart';
import 'package:adguard_home_manager/providers/app_config_provider.dart'; import 'package:adguard_home_manager/providers/app_config_provider.dart';
class TopItemsModal extends StatefulWidget { class TopItemsModal extends StatefulWidget {
final String type; final HomeTopItems type;
final String title; final String title;
final bool? isClient; final bool? isClient;
final List<Map<String, dynamic>> data; final List<Map<String, dynamic>> data;
final bool withProgressBar;
final String? unit;
const TopItemsModal({ const TopItemsModal({
Key? key, super.key,
required this.type, required this.type,
required this.title, required this.title,
this.isClient, this.isClient,
required this.data, required this.data,
}) : super(key: key); required this.withProgressBar,
this.unit,
});
@override @override
State<TopItemsModal> createState() => _TopItemsModalState(); State<TopItemsModal> createState() => _TopItemsModalState();
@ -59,9 +64,9 @@ class _TopItemsModalState extends State<TopItemsModal> {
final appConfigProvider = Provider.of<AppConfigProvider>(context); final appConfigProvider = Provider.of<AppConfigProvider>(context);
final logsProvider = Provider.of<LogsProvider>(context); final logsProvider = Provider.of<LogsProvider>(context);
int total = 0; double total = 0;
for (var element in data) { for (var element in data) {
total = total + int.parse(element.values.toList()[0].toString()); total = total + double.parse(element.values.toList()[0].toString());
} }
return Dialog( return Dialog(
@ -129,11 +134,11 @@ class _TopItemsModalState extends State<TopItemsModal> {
} }
return DomainOptions( return DomainOptions(
isBlocked: widget.type == 'topBlockedDomains', isBlocked: widget.type == HomeTopItems.blockedDomains,
isClient: widget.type == 'topClients', isDomain: widget.type == HomeTopItems.queriedDomains || widget.type == HomeTopItems.blockedDomains,
item: screenData[index].keys.toList()[0], item: screenData[index].keys.toList()[0],
onTap: () { onTap: () {
if (widget.type == 'topQueriedDomains' || widget.type == 'topBlockedDomains') { if (widget.type == HomeTopItems.queriedDomains || widget.type == HomeTopItems.blockedDomains) {
logsProvider.setSearchText(screenData[index].keys.toList()[0]); logsProvider.setSearchText(screenData[index].keys.toList()[0]);
logsProvider.setSelectedClients(null); logsProvider.setSelectedClients(null);
logsProvider.setAppliedFilters( logsProvider.setAppliedFilters(
@ -146,7 +151,7 @@ class _TopItemsModalState extends State<TopItemsModal> {
appConfigProvider.setSelectedScreen(2); appConfigProvider.setSelectedScreen(2);
Navigator.pop(context); Navigator.pop(context);
} }
else if (widget.type == 'topClients') { else if (widget.type == HomeTopItems.recurrentClients) {
logsProvider.setSearchText(null); logsProvider.setSearchText(null);
logsProvider.setSelectedClients([screenData[index].keys.toList()[0]]); logsProvider.setSelectedClients([screenData[index].keys.toList()[0]]);
logsProvider.setAppliedFilters( logsProvider.setAppliedFilters(
@ -163,7 +168,9 @@ class _TopItemsModalState extends State<TopItemsModal> {
child: CustomListTile( child: CustomListTile(
title: screenData[index].keys.toList()[0], title: screenData[index].keys.toList()[0],
trailing: Text( trailing: Text(
screenData[index].values.toList()[0].toString(), screenData[index].values.toList()[0].runtimeType == double
? "${screenData[index].values.toList()[0].toStringAsFixed(2)}${widget.unit != null ? ' ${widget.unit}' : ''}"
: "${screenData[index].values.toList()[0].toString()}${widget.unit != null ? ' ${widget.unit}' : ''}",
style: TextStyle( style: TextStyle(
color: Theme.of(context).colorScheme.onSurfaceVariant color: Theme.of(context).colorScheme.onSurfaceVariant
), ),
@ -181,7 +188,7 @@ class _TopItemsModalState extends State<TopItemsModal> {
), ),
const SizedBox(height: 5), const SizedBox(height: 5),
], ],
Row( if (widget.withProgressBar == true) Row(
children: [ children: [
SizedBox( SizedBox(
width: 50, width: 50,

View file

@ -17,21 +17,21 @@ import 'package:adguard_home_manager/models/menu_option.dart';
class DomainOptions extends StatelessWidget { class DomainOptions extends StatelessWidget {
final bool isBlocked; final bool isBlocked;
final bool? isClient; final bool? isDomain;
final String? item; final String? item;
final Widget child; final Widget child;
final void Function() onTap; final void Function() onTap;
final BorderRadius? borderRadius; final BorderRadius? borderRadius;
const DomainOptions({ const DomainOptions({
Key? key, super.key,
required this.isBlocked, required this.isBlocked,
this.isClient, this.isDomain,
required this.item, required this.item,
required this.child, required this.child,
required this.onTap, required this.onTap,
this.borderRadius this.borderRadius
}) : super(key: key); });
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@ -80,12 +80,12 @@ class DomainOptions extends StatelessWidget {
List<MenuOption> generateOptions() { List<MenuOption> generateOptions() {
return [ return [
if (isClient != true && isBlocked == true) MenuOption( if (isDomain == true && isBlocked == true) MenuOption(
title: AppLocalizations.of(context)!.unblock, title: AppLocalizations.of(context)!.unblock,
icon: Icons.check, icon: Icons.check,
action: () => blockUnblock(item!, 'unblock') action: () => blockUnblock(item!, 'unblock')
), ),
if (isClient != true && isBlocked == false) MenuOption( if (isDomain == true && isBlocked == false) MenuOption(
title: AppLocalizations.of(context)!.block, title: AppLocalizations.of(context)!.block,
icon: Icons.block, icon: Icons.block,
action: () => blockUnblock(item!, 'block') action: () => blockUnblock(item!, 'block')