Compare commits
2 Commits
95b2e0bf11
...
4517ec3078
| Author | SHA1 | Date | |
|---|---|---|---|
| 4517ec3078 | |||
| d72ee93f29 |
@ -28,6 +28,7 @@ class Business {
|
||||
int id;
|
||||
String name;
|
||||
String description;
|
||||
BusinessType? type;
|
||||
String? website;
|
||||
String? contactName;
|
||||
String? contactEmail;
|
||||
@ -40,6 +41,7 @@ class Business {
|
||||
{required this.id,
|
||||
required this.name,
|
||||
required this.description,
|
||||
this.type,
|
||||
this.website,
|
||||
this.contactName,
|
||||
this.contactEmail,
|
||||
@ -49,11 +51,21 @@ class Business {
|
||||
this.locationAddress});
|
||||
|
||||
factory Business.fromJson(Map<String, dynamic> json) {
|
||||
bool typeValid = true;
|
||||
try {
|
||||
BusinessType.values.byName(json['type']);
|
||||
} catch (e) {
|
||||
typeValid = false;
|
||||
}
|
||||
|
||||
return Business(
|
||||
id: json['id'],
|
||||
name: json['name'],
|
||||
description: json['description'],
|
||||
website: json['website'],
|
||||
type: typeValid
|
||||
? BusinessType.values.byName(json['type'])
|
||||
: BusinessType.other,
|
||||
contactName: json['contactName'],
|
||||
contactEmail: json['contactEmail'],
|
||||
contactPhone: json['contactPhone'],
|
||||
@ -151,12 +163,64 @@ void main() async {
|
||||
headers: {'Access-Control-Allow-Origin': '*'},
|
||||
);
|
||||
});
|
||||
app.get('/fbla-api/businessdata/overview', (Request request) async {
|
||||
app.get('/fbla-api/businessdata/overview/jobs', (Request request) async {
|
||||
print('business overview request received');
|
||||
|
||||
var filters = request.url.queryParameters['filters']?.split(',') ??
|
||||
JobType.values.asNameMap().keys;
|
||||
|
||||
Map<String, dynamic> output = {};
|
||||
|
||||
for (int i = 0; i < filters.length; i++) {
|
||||
var postgresResult = (await postgres.query('''
|
||||
SELECT json_agg(
|
||||
json_build_object(
|
||||
'id', b.id,
|
||||
'name', b.name,
|
||||
'contactName', b."contactName",
|
||||
'contactEmail', b."contactEmail",
|
||||
'contactPhone', b."contactPhone",
|
||||
'locationName', b."locationName",
|
||||
'listings', (
|
||||
SELECT json_agg(
|
||||
json_build_object(
|
||||
'id', l.id,
|
||||
'name', l.name,
|
||||
'description', l.description,
|
||||
'type', l.type,
|
||||
'wage', l.wage,
|
||||
'link', l.link
|
||||
)
|
||||
)
|
||||
FROM listings l
|
||||
WHERE l."businessId" = b.id AND l.type = '${filters.elementAt(i)}'
|
||||
)
|
||||
)
|
||||
)
|
||||
FROM businesses b
|
||||
WHERE b.id IN (SELECT "businessId" FROM public.listings WHERE type='${filters.elementAt(i)}')
|
||||
GROUP BY b.id;
|
||||
'''));
|
||||
|
||||
if (postgresResult.isNotEmpty) {
|
||||
output.addAll({filters.elementAt(i): postgresResult[0][0]});
|
||||
}
|
||||
}
|
||||
|
||||
return Response.ok(
|
||||
json.encode(output),
|
||||
headers: {
|
||||
'Access-Control-Allow-Origin': '*',
|
||||
'Content-Type': 'text/plain'
|
||||
},
|
||||
);
|
||||
});
|
||||
app.get('/fbla-api/businessdata/overview/types', (Request request) async {
|
||||
print('business overview request received');
|
||||
|
||||
var filters = request.url.queryParameters['filters']?.split(',') ??
|
||||
BusinessType.values.asNameMap().keys;
|
||||
|
||||
// List<Map<String, List<Map<String, dynamic>>>> this is the real type lol
|
||||
Map<String, dynamic> output = {};
|
||||
|
||||
@ -172,7 +236,7 @@ void main() async {
|
||||
'contactPhone', "contactPhone",
|
||||
'locationName', "locationName"
|
||||
)
|
||||
) FROM public.businesses WHERE id IN (SELECT "businessId" FROM public.listings WHERE type='${filters.elementAt(i)}')
|
||||
) FROM public.businesses WHERE type='${filters.elementAt(i)}'
|
||||
'''))[0][0];
|
||||
|
||||
if (postgresResult != null) {
|
||||
@ -180,6 +244,7 @@ void main() async {
|
||||
}
|
||||
}
|
||||
|
||||
// await Future.delayed(Duration(seconds: 5));
|
||||
return Response.ok(
|
||||
json.encode(output),
|
||||
headers: {
|
||||
@ -218,6 +283,7 @@ void main() async {
|
||||
'id', b.id,
|
||||
'name', b.name,
|
||||
'description', b.description,
|
||||
'type', b.type,
|
||||
'website', b.website,
|
||||
'contactName', b."contactName",
|
||||
'contactEmail', b."contactEmail",
|
||||
@ -226,7 +292,9 @@ void main() async {
|
||||
'locationName', b."locationName",
|
||||
'locationAddress', b."locationAddress",
|
||||
'listings',
|
||||
json_agg(
|
||||
CASE
|
||||
WHEN COUNT(l.id) = 0 THEN 'null'
|
||||
ELSE json_agg(
|
||||
json_build_object(
|
||||
'id', l.id,
|
||||
'businessId', l."businessId",
|
||||
@ -237,6 +305,7 @@ void main() async {
|
||||
'link', l.link
|
||||
)
|
||||
)
|
||||
END
|
||||
)
|
||||
FROM businesses b
|
||||
LEFT JOIN listings l ON b.id = l."businessId"
|
||||
@ -273,14 +342,16 @@ void main() async {
|
||||
'name', b.name,
|
||||
'description', b.description,
|
||||
'website', b.website,
|
||||
'type', b.type,
|
||||
'contactName', b."contactName",
|
||||
'contactEmail', b."contactEmail",
|
||||
'contactPhone', b."contactPhone",
|
||||
'notes', b.notes,
|
||||
'locationName', b."locationName",
|
||||
'locationAddress', b."locationAddress",
|
||||
'listings',
|
||||
json_agg(
|
||||
'listings', CASE
|
||||
WHEN COUNT(l.id) = 0 THEN 'null'
|
||||
ELSE json_agg(
|
||||
json_build_object(
|
||||
'id', l.id,
|
||||
'businessId', l."businessId",
|
||||
@ -291,6 +362,7 @@ void main() async {
|
||||
'link', l.link
|
||||
)
|
||||
)
|
||||
END
|
||||
)
|
||||
FROM businesses b
|
||||
LEFT JOIN listings l ON b.id = l."businessId"
|
||||
@ -362,9 +434,10 @@ void main() async {
|
||||
Business business = Business.fromJson(json);
|
||||
|
||||
await postgres.query('''
|
||||
INSERT INTO businesses (name, description, website, "contactName", "contactPhone", "contactEmail", notes, "locationName", "locationAddress")
|
||||
VALUES ('${business.name.replaceAll("'", "''")}', '${business.description.replaceAll("'", "''")}', '${business.website ?? 'NULL'}', '${business.contactName?.replaceAll("'", "''") ?? 'NULL'}', '${business.contactPhone ?? 'NULL'}', '${business.contactEmail ?? 'NULL'}', '${business.notes?.replaceAll("'", "''") ?? 'NULL'}', '${business.locationName?.replaceAll("'", "''") ?? 'NULL'}', '${business.locationAddress?.replaceAll("'", "''") ?? 'NULL'}')
|
||||
''');
|
||||
INSERT INTO businesses (name, description, website, type, "contactName", "contactPhone", "contactEmail", notes, "locationName", "locationAddress")
|
||||
VALUES ('${business.name.replaceAll("'", "''")}', '${business.description.replaceAll("'", "''")}', '${business.website ?? 'NULL'}', '${business.type?.name}', '${business.contactName?.replaceAll("'", "''") ?? 'NULL'}', '${business.contactPhone ?? 'NULL'}', '${business.contactEmail ?? 'NULL'}', '${business.notes?.replaceAll("'", "''") ?? 'NULL'}', '${business.locationName?.replaceAll("'", "''") ?? 'NULL'}', '${business.locationAddress?.replaceAll("'", "''") ?? 'NULL'}')
|
||||
'''
|
||||
.replaceAll("'null'", 'NULL'));
|
||||
|
||||
final dbBusiness = await postgres.query('''SELECT * FROM public.businesses
|
||||
ORDER BY id DESC LIMIT 1''');
|
||||
@ -403,8 +476,9 @@ void main() async {
|
||||
|
||||
await postgres.query('''
|
||||
INSERT INTO listings ("businessId", name, description, type, wage, link)
|
||||
VALUES ('${listing.businessId}' '${listing.name.replaceAll("'", "''")}', '${listing.description.replaceAll("'", "''")}', '${listing.type.name}', '${listing.wage ?? 'NULL'}', '${listing.link?.replaceAll("'", "''") ?? 'NULL'}')
|
||||
''');
|
||||
VALUES ('${listing.businessId}', '${listing.name.replaceAll("'", "''")}', '${listing.description.replaceAll("'", "''")}', '${listing.type.name}', '${listing.wage ?? 'NULL'}', '${listing.link?.replaceAll("'", "''") ?? 'NULL'}')
|
||||
'''
|
||||
.replaceAll("'null'", 'NULL'));
|
||||
|
||||
final dbListing = await postgres.query('''SELECT id FROM public.listings
|
||||
ORDER BY id DESC LIMIT 1''');
|
||||
@ -500,7 +574,8 @@ void main() async {
|
||||
UPDATE businesses SET
|
||||
name = '${business.name.replaceAll("'", "''").replaceAll("\"", "\"\"")}'::text, description = '${business.description.replaceAll("'", "''").replaceAll("\"", "\"\"")}'::text, website = '${business.website!}'::text, "contactName" = '${business.contactName!.replaceAll("'", "''").replaceAll("\"", "\"\"")}'::text, "contactPhone" = '${business.contactPhone!}'::text, "contactEmail" = '${business.contactEmail!}'::text, notes = '${business.notes!.replaceAll("'", "''").replaceAll("\"", "\"\"")}'::text, "locationName" = '${business.locationName!.replaceAll("'", "''").replaceAll("\"", "\"\"")}'::text, "locationAddress" = '${business.locationAddress!.replaceAll("'", "''").replaceAll("\"", "\"\"")}'::text WHERE
|
||||
id = ${business.id};
|
||||
''');
|
||||
'''
|
||||
.replaceAll("'null'", 'NULL'));
|
||||
|
||||
var logoResponse = await http.get(
|
||||
Uri.http('logo.clearbit.com', '/${business.website}'),
|
||||
@ -546,7 +621,8 @@ void main() async {
|
||||
UPDATE listings SET
|
||||
"businessId" = ${listing.businessId}, name = '${listing.name.replaceAll("'", "''")}'::text, description = '${listing.description.replaceAll("'", "''")}'::text, type = '${listing.type.name}'::text, wage = '${listing.wage ?? 'NULL'}'::text, link = '${listing.link?.replaceAll("'", "''") ?? 'NULL'}'::text WHERE
|
||||
id = ${listing.id};
|
||||
''');
|
||||
'''
|
||||
.replaceAll("'null'", 'NULL'));
|
||||
|
||||
return Response.ok(
|
||||
listing.id.toString(),
|
||||
|
||||
63
fbla_ui/assets/MarinoDev.svg
Normal file
63
fbla_ui/assets/MarinoDev.svg
Normal file
@ -0,0 +1,63 @@
|
||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
||||
|
||||
<svg
|
||||
width="200mm"
|
||||
height="70mm"
|
||||
viewBox="0 0 200 70"
|
||||
version="1.1"
|
||||
id="svg1"
|
||||
inkscape:version="1.3.2 (091e20ef0f, 2023-11-25, custom)"
|
||||
sodipodi:docname="MDEVLogoFull.svg"
|
||||
xml:space="preserve"
|
||||
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
xmlns:svg="http://www.w3.org/2000/svg"><sodipodi:namedview
|
||||
id="namedview1"
|
||||
pagecolor="#ffffff"
|
||||
bordercolor="#000000"
|
||||
borderopacity="0.25"
|
||||
inkscape:showpageshadow="2"
|
||||
inkscape:pageopacity="0.0"
|
||||
inkscape:pagecheckerboard="0"
|
||||
inkscape:deskcolor="#d1d1d1"
|
||||
inkscape:document-units="mm"
|
||||
inkscape:zoom="0.75544332"
|
||||
inkscape:cx="47.654138"
|
||||
inkscape:cy="97.955728"
|
||||
inkscape:window-width="1920"
|
||||
inkscape:window-height="1046"
|
||||
inkscape:window-x="0"
|
||||
inkscape:window-y="0"
|
||||
inkscape:window-maximized="1"
|
||||
inkscape:current-layer="layer1" /><defs
|
||||
id="defs1"><rect
|
||||
x="164.6723"
|
||||
y="176.95522"
|
||||
width="237.7821"
|
||||
height="119.97466"
|
||||
id="rect1047" /><rect
|
||||
x="170.3439"
|
||||
y="136.68353"
|
||||
width="277.49832"
|
||||
height="196.87138"
|
||||
id="rect9139" /></defs><g
|
||||
inkscape:label="Layer 1"
|
||||
inkscape:groupmode="layer"
|
||||
id="layer1"><path
|
||||
style="font-size:48px;font-family:HeadlineNEWS;-inkscape-font-specification:HeadlineNEWS;white-space:pre;shape-padding:0.133618;fill:#000000;fill-opacity:1;stroke-width:0.608723"
|
||||
d="m 103.58306,22.62203 h -7.380754 q 0,-2.975635 0,-5.95127 0,-4.113378 -0.233384,-5.74706 l -4.113377,8.489312 -3.617439,-0.08752 -3.82165,-8.401793 q -0.262556,1.779547 -0.291729,5.74706 0,2.975635 0,5.95127 h -7.380745 l 1.10857,-21.90884035 h 7.205708 l 5.105257,9.91878235 5.134429,-9.91878235 h 7.147374 z m 22.60912,0 h -5.68875 l -0.90436,-2.683906 h -7.20575 l -0.93354,2.683906 h -5.71792 L 112.7434,5.1182952 h 6.47642 z m -7.52665,-5.54285 -2.71309,-8.4601379 -2.65475,8.4601379 z m 27.62688,5.54285 h -6.12634 q -1.72121,-3.442402 -1.80873,-3.617439 -1.40031,-2.479696 -2.45054,-3.033981 l -1.72121,-0.904359 V 22.62203 H 128.4385 V 5.1182952 h 9.71463 q 2.77144,0 4.43431,1.050224 2.01294,1.2836072 2.01294,3.8799948 0,1.808719 -1.16692,2.888116 -0.81685,0.787668 -2.77145,1.51699 1.25445,0.583458 3.23821,3.850822 1.1961,2.158794 2.39219,4.317588 z m -7.29326,-12.194269 q 0,-2.4505225 -3.034,-2.4505225 h -1.77956 V 12.93663 h 1.77956 q 1.31279,0 2.15881,-0.612631 0.87519,-0.641804 0.87519,-1.896238 z m 15.66591,12.194269 h -6.068 V 5.1182952 h 6.068 z m 22.40489,0 h -5.33867 l -7.67251,-9.743746 0.20421,9.743746 h -5.74709 V 5.1182952 h 5.36784 l 7.76003,9.6562268 -0.26256,-9.6562268 h 5.68875 z m 22.02554,-8.839386 q 0,4.230069 -2.59639,6.709764 -2.59639,2.450523 -6.85563,2.450523 -4.23007,0 -6.85563,-2.42135 -2.62556,-2.42135 -2.62556,-6.622246 0,-4.1717228 2.71308,-6.6805917 2.62556,-2.4505228 6.85563,-2.4505228 4.11338,0 6.70976,2.4213499 2.65474,2.5088688 2.65474,6.5930736 z m -5.74706,0.02917 q 0,-1.808719 -0.96271,-2.946462 -0.9627,-1.1669148 -2.74225,-1.1669148 -1.83789,0 -2.77143,1.2252608 -0.87518,1.10857 -0.87518,2.975635 0,1.837892 0.87518,2.917289 0.93354,1.196089 2.71308,1.196089 3.76331,0 3.76331,-4.200897 z"
|
||||
id="text9137"
|
||||
inkscape:label="Marino"
|
||||
aria-label="Marino" /><path
|
||||
d="m 104.7848,29.612184 q 8.86861,0 14.46983,5.4456 5.60122,5.393746 5.60122,14.210437 0,8.505515 -5.18631,13.899254 -5.13446,5.341879 -13.58816,5.341879 H 86.114135 v -38.89717 z m 6.37918,19.500449 q 0,-4.200897 -2.1264,-6.690317 -2.28198,-2.696869 -6.43104,-2.696869 h -2.956187 v 18.774366 h 3.059917 q 4.25279,0 6.43104,-2.645003 2.02267,-2.437559 2.02267,-6.742177 z m 45.19311,19.396721 H 128.71401 V 29.560317 h 27.53935 v 8.92042 H 142.2503 v 5.964233 h 12.65462 v 8.868553 H 142.2503 v 6.327278 h 14.10679 z m 43.5333,-38.949037 -15.40328,38.949037 H 173.18098 L 157.31084,29.560317 h 14.41797 l 7.62387,23.234579 q 0.15559,-0.20745 7.31267,-23.234579 z"
|
||||
id="text1045"
|
||||
style="font-size:85.3333px;font-family:HeadlineNEWS;-inkscape-font-specification:HeadlineNEWS;letter-spacing:-4.66px;white-space:pre;fill:#000000;fill-opacity:1;stroke-width:0.608723"
|
||||
inkscape:label="Dev"
|
||||
aria-label="DEV" /><path
|
||||
d="m 39.967467,0 c -1.24713,0 -2.07854,1.440059 -2.07854,1.440059 l -4.57281,7.920321 17.81611,30.858404 c 4.37185,-0.99615 6.86321,4.80385 3.14145,7.29302 -3.72178,2.48917 -8.13084,-2.02826 -5.54068,-5.68838 v 0 L 31.653287,12.240495 0.47509223,66.242704 c -0.55431396,0.96004 -0.62353496,1.80008 -0.20783896,2.52011 0.41568996,0.72005 1.17784883,1.08004 2.28640583,1.08004 h 9.1456049 l 17.816103,-30.8584 c -3.04862,-3.28806 0.72867,-8.34565 4.74522,-6.36708 4.01656,1.97857 2.30888,8.05564 -2.15596,7.64256 v 0 l -17.079704,29.58292 h 62.356384 c 0,0 1.66283,0 2.2864,-1.08004 0.62353,-1.08005 -0.20784,-2.52011 -0.20784,-2.52011 l -4.57281,-7.92032 h -35.63222 c -1.32322,4.28422 -7.59186,3.5418 -7.88666,-0.92594 -0.29484,-4.46773 5.82195,-6.02738 7.69662,-1.95418 v 0 h 34.15941 L 42.046017,1.440059 c 0,0 -0.83138,-1.440059 -2.07855,-1.440059"
|
||||
style="display:inline;fill:#2096f3;fill-opacity:1;stroke:none;stroke-width:2.37548"
|
||||
id="path395"
|
||||
inkscape:label="triangle"
|
||||
inkscape:transform-center-y="-11.666667" /></g></svg>
|
||||
|
After Width: | Height: | Size: 5.4 KiB |
BIN
fbla_ui/assets/Triangle256.png
Normal file
BIN
fbla_ui/assets/Triangle256.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 9.5 KiB |
@ -1,80 +1,50 @@
|
||||
import 'package:dart_jsonwebtoken/dart_jsonwebtoken.dart';
|
||||
import 'package:fbla_ui/api_logic.dart';
|
||||
import 'package:fbla_ui/main.dart';
|
||||
import 'package:fbla_ui/pages/businesses_overview.dart';
|
||||
import 'package:fbla_ui/pages/create_edit_business.dart';
|
||||
import 'package:fbla_ui/pages/export_data.dart';
|
||||
import 'package:fbla_ui/pages/signin_page.dart';
|
||||
import 'package:fbla_ui/shared.dart';
|
||||
import 'package:fbla_ui/pages/create_edit_listing.dart';
|
||||
import 'package:fbla_ui/pages/listings_overview.dart';
|
||||
import 'package:fbla_ui/shared/api_logic.dart';
|
||||
import 'package:fbla_ui/shared/global_vars.dart';
|
||||
import 'package:fbla_ui/shared/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:rive/rive.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
import 'package:url_launcher/url_launcher.dart';
|
||||
|
||||
class Home extends StatefulWidget {
|
||||
final void Function() themeCallback;
|
||||
final int? initialPage;
|
||||
|
||||
const Home({super.key, required this.themeCallback});
|
||||
const Home({super.key, required this.themeCallback, this.initialPage});
|
||||
|
||||
@override
|
||||
State<Home> createState() => _HomeState();
|
||||
}
|
||||
|
||||
class _HomeState extends State<Home> {
|
||||
late Future refreshBusinessDataOverviewFuture;
|
||||
bool _isPreviousData = false;
|
||||
late Map<JobType, List<Business>> overviewBusinesses;
|
||||
Set<JobType> jobTypeFilters = <JobType>{};
|
||||
Set<BusinessType> businessTypeFilters = <BusinessType>{};
|
||||
String searchQuery = '';
|
||||
Set<DataTypeJob> selectedDataTypesJob = <DataTypeJob>{};
|
||||
Set<DataTypeBusiness> selectedDataTypesBusiness = <DataTypeBusiness>{};
|
||||
late Future refreshBusinessDataOverviewJobFuture;
|
||||
late Future refreshBusinessDataOverviewBusinessFuture;
|
||||
int currentPageIndex = 0;
|
||||
late dynamic previousJobData;
|
||||
ScrollController scrollControllerBusinesses = ScrollController();
|
||||
ScrollController scrollControllerJobs = ScrollController();
|
||||
|
||||
Future<void> _setFilters(Set<JobType> filters) async {
|
||||
void _updateLoggedIn(bool updated) {
|
||||
setState(() {
|
||||
jobTypeFilters = filters;
|
||||
loggedIn = updated;
|
||||
});
|
||||
_updateOverviewBusinesses();
|
||||
}
|
||||
|
||||
Future<void> _updateOverviewBusinesses() async {
|
||||
var refreshedData =
|
||||
fetchBusinessDataOverview(typeFilters: jobTypeFilters.toList());
|
||||
await refreshedData;
|
||||
setState(() {
|
||||
refreshBusinessDataOverviewFuture = refreshedData;
|
||||
});
|
||||
}
|
||||
|
||||
Map<JobType, List<Business>> _filterBySearch(
|
||||
Map<JobType, List<Business>> businesses) {
|
||||
Map<JobType, List<Business>> filteredBusinesses = businesses;
|
||||
|
||||
for (JobType jobType in businesses.keys) {
|
||||
filteredBusinesses[jobType]!.removeWhere((tmpBusiness) => !tmpBusiness
|
||||
.name
|
||||
.replaceAll(RegExp(r'[^a-zA-Z]'), '')
|
||||
.toLowerCase()
|
||||
.contains(searchQuery
|
||||
.replaceAll(RegExp(r'[^a-zA-Z]'), '')
|
||||
.toLowerCase()
|
||||
.trim()));
|
||||
}
|
||||
filteredBusinesses.removeWhere((key, value) => value.isEmpty);
|
||||
return filteredBusinesses;
|
||||
}
|
||||
|
||||
Future<void> _setSearch(String search) async {
|
||||
setState(() {
|
||||
searchQuery = search;
|
||||
});
|
||||
_updateOverviewBusinesses();
|
||||
}
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
|
||||
refreshBusinessDataOverviewFuture = fetchBusinessDataOverview();
|
||||
currentPageIndex = widget.initialPage ?? 0;
|
||||
|
||||
initialLogin();
|
||||
refreshBusinessDataOverviewJobFuture = fetchBusinessDataOverviewJobs();
|
||||
refreshBusinessDataOverviewBusinessFuture =
|
||||
fetchBusinessDataOverviewTypes();
|
||||
}
|
||||
|
||||
Future<void> initialLogin() async {
|
||||
@ -94,319 +64,321 @@ class _HomeState extends State<Home> {
|
||||
}
|
||||
}
|
||||
|
||||
void setStateCallback() {
|
||||
Future<void> _updateOverviewBusinessesJobsCallback(
|
||||
Set<JobType>? newFilters) async {
|
||||
if (newFilters != null) {
|
||||
jobTypeFilters = Set.from(newFilters);
|
||||
}
|
||||
var refreshedData =
|
||||
fetchBusinessDataOverviewJobs(typeFilters: jobTypeFilters.toList());
|
||||
await refreshedData;
|
||||
setState(() {
|
||||
loggedIn = loggedIn;
|
||||
refreshBusinessDataOverviewJobFuture = refreshedData;
|
||||
});
|
||||
}
|
||||
|
||||
Future<void> _updateOverviewBusinessesBusinessCallback(
|
||||
Set<BusinessType>? newFilters) async {
|
||||
if (newFilters != null) {
|
||||
businessTypeFilters = Set.from(newFilters);
|
||||
}
|
||||
var refreshedData = fetchBusinessDataOverviewTypes(
|
||||
typeFilters: businessTypeFilters.toList());
|
||||
await refreshedData;
|
||||
setState(() {
|
||||
refreshBusinessDataOverviewBusinessFuture = refreshedData;
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
bool widescreen = MediaQuery.sizeOf(context).width >= 1000;
|
||||
bool widescreen = MediaQuery.sizeOf(context).width >= widescreenWidth;
|
||||
return Scaffold(
|
||||
// backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
||||
floatingActionButton: _getFAB(),
|
||||
// floatingActionButton: _getFAB(widescreen, scrollControllerBusinesses,
|
||||
// scrollControllerJobs, currentPageIndex),
|
||||
bottomNavigationBar: _getNavigationBar(widescreen),
|
||||
body: RefreshIndicator(
|
||||
edgeOffset: 120,
|
||||
edgeOffset: 145,
|
||||
onRefresh: () async {
|
||||
_updateOverviewBusinesses();
|
||||
_updateOverviewBusinessesJobsCallback(null);
|
||||
_updateOverviewBusinessesBusinessCallback(null);
|
||||
},
|
||||
child: CustomScrollView(
|
||||
slivers: [
|
||||
SliverAppBar(
|
||||
title: widescreen
|
||||
? BusinessSearchBar(
|
||||
filters: jobTypeFilters,
|
||||
setFiltersCallback: _setFilters,
|
||||
setSearchCallback: _setSearch)
|
||||
: const Text('Job Link'),
|
||||
toolbarHeight: 70,
|
||||
pinned: true,
|
||||
scrolledUnderElevation: 0,
|
||||
centerTitle: true,
|
||||
expandedHeight: widescreen ? 70 : 120,
|
||||
bottom: _getBottom(),
|
||||
leading: IconButton(
|
||||
icon: getIconFromThemeMode(themeMode),
|
||||
child: widescreen
|
||||
? Row(
|
||||
children: [
|
||||
_getNavigationRail(),
|
||||
Expanded(
|
||||
child: _ContentPane(
|
||||
themeCallback: widget.themeCallback,
|
||||
searchQuery: searchQuery,
|
||||
currentPageIndex: currentPageIndex,
|
||||
refreshBusinessDataOverviewBusinessFuture:
|
||||
refreshBusinessDataOverviewBusinessFuture,
|
||||
refreshBusinessDataOverviewJobFuture:
|
||||
refreshBusinessDataOverviewJobFuture,
|
||||
updateOverviewBusinessesBusinessCallback:
|
||||
_updateOverviewBusinessesBusinessCallback,
|
||||
updateOverviewBusinessesJobsCallback:
|
||||
_updateOverviewBusinessesJobsCallback,
|
||||
updateLoggedIn: _updateLoggedIn,
|
||||
),
|
||||
)
|
||||
],
|
||||
)
|
||||
: _ContentPane(
|
||||
themeCallback: widget.themeCallback,
|
||||
searchQuery: searchQuery,
|
||||
currentPageIndex: currentPageIndex,
|
||||
refreshBusinessDataOverviewBusinessFuture:
|
||||
refreshBusinessDataOverviewBusinessFuture,
|
||||
refreshBusinessDataOverviewJobFuture:
|
||||
refreshBusinessDataOverviewJobFuture,
|
||||
updateOverviewBusinessesBusinessCallback:
|
||||
_updateOverviewBusinessesBusinessCallback,
|
||||
updateOverviewBusinessesJobsCallback:
|
||||
_updateOverviewBusinessesJobsCallback,
|
||||
updateLoggedIn: _updateLoggedIn,
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget? _getNavigationBar(bool widescreen) {
|
||||
if (!widescreen) {
|
||||
return NavigationBar(
|
||||
selectedIndex: currentPageIndex,
|
||||
indicatorColor: Theme.of(context).colorScheme.primary.withOpacity(0.5),
|
||||
onDestinationSelected: (int index) {
|
||||
setState(() {
|
||||
currentPageIndex = index;
|
||||
});
|
||||
},
|
||||
destinations: <NavigationDestination>[
|
||||
NavigationDestination(
|
||||
icon: const Icon(Icons.business_outlined),
|
||||
selectedIcon: Icon(
|
||||
Icons.business,
|
||||
color: Theme.of(context).colorScheme.onSurface,
|
||||
),
|
||||
label: 'Businesses'),
|
||||
NavigationDestination(
|
||||
icon: const Icon(Icons.work_outline),
|
||||
selectedIcon: Icon(
|
||||
Icons.work,
|
||||
color: Theme.of(context).colorScheme.onSurface,
|
||||
),
|
||||
label: 'Job Listings'),
|
||||
// NavigationDestination(
|
||||
// icon: const Icon(Icons.description_outlined),
|
||||
// selectedIcon: Icon(
|
||||
// Icons.description,
|
||||
// color: Theme.of(context).colorScheme.onSurface,
|
||||
// ),
|
||||
// label: 'Export Data')
|
||||
],
|
||||
);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
Widget _getNavigationRail() {
|
||||
return Row(
|
||||
children: [
|
||||
NavigationRail(
|
||||
selectedIndex: currentPageIndex,
|
||||
groupAlignment: -1,
|
||||
indicatorColor:
|
||||
Theme.of(context).colorScheme.primary.withOpacity(0.5),
|
||||
trailing: Expanded(
|
||||
child: Align(
|
||||
alignment: Alignment.bottomCenter,
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16),
|
||||
child: IconButton(
|
||||
iconSize: 30,
|
||||
icon: Icon(
|
||||
getIconFromThemeMode(themeMode),
|
||||
),
|
||||
onPressed: () {
|
||||
setState(() {
|
||||
widget.themeCallback();
|
||||
});
|
||||
},
|
||||
),
|
||||
actions: [
|
||||
IconButton(
|
||||
icon: const Icon(Icons.help),
|
||||
),
|
||||
),
|
||||
),
|
||||
leading: Column(
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(top: 2.0, bottom: 8.0),
|
||||
child: Image.asset(
|
||||
'assets/Triangle256.png',
|
||||
height: 50,
|
||||
),
|
||||
),
|
||||
if (loggedIn)
|
||||
FloatingActionButton(
|
||||
child: Icon(Icons.add),
|
||||
heroTag: 'Homepage',
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
title: const Text('About'),
|
||||
backgroundColor:
|
||||
Theme.of(context).colorScheme.surface,
|
||||
content: SizedBox(
|
||||
width: 500,
|
||||
child: IntrinsicHeight(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
const Text(
|
||||
'Welcome to my FBLA 2024 Coding and Programming submission!\n\n'
|
||||
'MarinoDev Job Link aims to provide comprehensive details of businesses and community partners'
|
||||
' for Waukesha West High School\'s Career and Technical Education Department.\n\n'),
|
||||
MouseRegion(
|
||||
cursor: SystemMouseCursors.click,
|
||||
child: GestureDetector(
|
||||
child: const Column(
|
||||
crossAxisAlignment:
|
||||
CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text('Git Repo:'),
|
||||
Text(
|
||||
'https://git.marinodev.com/MarinoDev/FBLA24\n',
|
||||
style: TextStyle(
|
||||
color: Colors.blue)),
|
||||
],
|
||||
),
|
||||
onTap: () {
|
||||
launchUrl(Uri.https(
|
||||
'git.marinodev.com',
|
||||
'/MarinoDev/FBLA24'));
|
||||
},
|
||||
),
|
||||
),
|
||||
MouseRegion(
|
||||
cursor: SystemMouseCursors.click,
|
||||
child: GestureDetector(
|
||||
child: const Column(
|
||||
crossAxisAlignment:
|
||||
CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
'Please direct any questions to'),
|
||||
Text('drake@marinodev.com',
|
||||
style: TextStyle(
|
||||
color: Colors.blue)),
|
||||
],
|
||||
),
|
||||
onTap: () {
|
||||
launchUrl(Uri.parse(
|
||||
'mailto:drake@marinodev.com'));
|
||||
},
|
||||
),
|
||||
)
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('OK'),
|
||||
onPressed: () {
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
],
|
||||
);
|
||||
});
|
||||
},
|
||||
),
|
||||
IconButton(
|
||||
icon: const Icon(Icons.picture_as_pdf),
|
||||
onPressed: () async {
|
||||
if (!_isPreviousData) {
|
||||
ScaffoldMessenger.of(context).clearSnackBars();
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
const SnackBar(
|
||||
width: 300,
|
||||
behavior: SnackBarBehavior.floating,
|
||||
content: Text('There is no data!'),
|
||||
duration: Duration(seconds: 2),
|
||||
),
|
||||
);
|
||||
} else {
|
||||
selectedDataTypesBusiness = <DataTypeBusiness>{};
|
||||
if (currentPageIndex == 0) {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => ExportData(
|
||||
groupedBusinesses: overviewBusinesses)));
|
||||
builder: (context) =>
|
||||
const CreateEditBusiness()));
|
||||
} else if (currentPageIndex == 1) {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) =>
|
||||
const CreateEditJobListing()));
|
||||
}
|
||||
},
|
||||
)
|
||||
],
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(right: 8.0),
|
||||
child: IconButton(
|
||||
icon: loggedIn
|
||||
? const Icon(Icons.account_circle)
|
||||
: const Icon(Icons.login),
|
||||
onPressed: () {
|
||||
if (loggedIn) {
|
||||
var payload = JWT.decode(jwt).payload;
|
||||
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
backgroundColor:
|
||||
Theme.of(context).colorScheme.surface,
|
||||
title: Text('Hi, ${payload['username']}!'),
|
||||
content: Text(
|
||||
'You are logged in as an admin with username ${payload['username']}.'),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Cancel'),
|
||||
onPressed: () {
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Logout'),
|
||||
onPressed: () async {
|
||||
final prefs = await SharedPreferences
|
||||
.getInstance();
|
||||
prefs.setBool('rememberMe', false);
|
||||
prefs.setString('username', '');
|
||||
prefs.setString('password', '');
|
||||
|
||||
onDestinationSelected: (int index) {
|
||||
setState(() {
|
||||
loggedIn = false;
|
||||
currentPageIndex = index;
|
||||
});
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
},
|
||||
labelType: NavigationRailLabelType.all,
|
||||
destinations: <NavigationRailDestination>[
|
||||
NavigationRailDestination(
|
||||
icon: const Icon(Icons.business_outlined),
|
||||
selectedIcon: Icon(
|
||||
Icons.business,
|
||||
color: Theme.of(context).colorScheme.onSurface,
|
||||
),
|
||||
label: const Text('Businesses')),
|
||||
NavigationRailDestination(
|
||||
icon: const Icon(Icons.work_outline),
|
||||
selectedIcon: Icon(
|
||||
Icons.work,
|
||||
color: Theme.of(context).colorScheme.onSurface,
|
||||
),
|
||||
label: const Text('Job Listings')),
|
||||
// NavigationRailDestination(
|
||||
// icon: const Icon(Icons.description_outlined),
|
||||
// selectedIcon: Icon(
|
||||
// Icons.description,
|
||||
// color: Theme.of(context).colorScheme.onSurface,
|
||||
// ),
|
||||
// label: const Text('Export Data'))
|
||||
],
|
||||
),
|
||||
// children.first
|
||||
],
|
||||
);
|
||||
// }
|
||||
// return children.first;
|
||||
}
|
||||
|
||||
// Widget _contentPane() {
|
||||
// return IndexedStack(
|
||||
// index: currentPageIndex,
|
||||
// children: [
|
||||
// BusinessesOverview(
|
||||
// searchQuery: searchQuery,
|
||||
// refreshBusinessDataOverviewFuture:
|
||||
// refreshBusinessDataOverviewBusinessFuture,
|
||||
// updateBusinessesCallback: _updateOverviewBusinessesBusinessCallback,
|
||||
// themeCallback: widget.themeCallback,
|
||||
// updateLoggedIn: _updateLoggedIn,
|
||||
// ),
|
||||
// JobsOverview(
|
||||
// searchQuery: searchQuery,
|
||||
// refreshJobDataOverviewFuture: refreshBusinessDataOverviewJobFuture,
|
||||
// updateBusinessesCallback: _updateOverviewBusinessesJobsCallback,
|
||||
// themeCallback: widget.themeCallback, updateLoggedIn: _updateLoggedIn),
|
||||
// ExportData(
|
||||
// searchQuery: searchQuery,
|
||||
// refreshBusinessDataOverviewFuture:
|
||||
// refreshBusinessDataOverviewBusinessFuture,
|
||||
// updateBusinessesWithJobCallback:
|
||||
// _updateOverviewBusinessesJobsCallback,
|
||||
// themeCallback: widget.themeCallback,
|
||||
// refreshJobDataOverviewFuture: refreshBusinessDataOverviewJobFuture,
|
||||
// updateBusinessesCallback: _updateOverviewBusinessesBusinessCallback)
|
||||
// ],
|
||||
// );
|
||||
// }
|
||||
}
|
||||
|
||||
class _ContentPane extends StatelessWidget {
|
||||
final String searchQuery;
|
||||
final Future refreshBusinessDataOverviewBusinessFuture;
|
||||
final Future<void> Function(Set<BusinessType>)
|
||||
updateOverviewBusinessesBusinessCallback;
|
||||
final void Function() themeCallback;
|
||||
final Future refreshBusinessDataOverviewJobFuture;
|
||||
final Future<void> Function(Set<JobType>)
|
||||
updateOverviewBusinessesJobsCallback;
|
||||
final int currentPageIndex;
|
||||
final void Function(bool) updateLoggedIn;
|
||||
|
||||
const _ContentPane({
|
||||
required this.searchQuery,
|
||||
required this.refreshBusinessDataOverviewBusinessFuture,
|
||||
required this.updateOverviewBusinessesBusinessCallback,
|
||||
required this.themeCallback,
|
||||
required this.refreshBusinessDataOverviewJobFuture,
|
||||
required this.updateOverviewBusinessesJobsCallback,
|
||||
required this.currentPageIndex,
|
||||
required this.updateLoggedIn,
|
||||
});
|
||||
} else {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => SignInPage(
|
||||
refreshAccount: setStateCallback)));
|
||||
}
|
||||
},
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return IndexedStack(
|
||||
index: currentPageIndex,
|
||||
children: [
|
||||
BusinessesOverview(
|
||||
searchQuery: searchQuery,
|
||||
refreshBusinessDataOverviewFuture:
|
||||
refreshBusinessDataOverviewBusinessFuture,
|
||||
updateBusinessesCallback: updateOverviewBusinessesBusinessCallback,
|
||||
themeCallback: themeCallback,
|
||||
updateLoggedIn: updateLoggedIn,
|
||||
),
|
||||
JobsOverview(
|
||||
searchQuery: searchQuery,
|
||||
refreshJobDataOverviewFuture: refreshBusinessDataOverviewJobFuture,
|
||||
updateBusinessesCallback: updateOverviewBusinessesJobsCallback,
|
||||
themeCallback: themeCallback,
|
||||
updateLoggedIn: updateLoggedIn,
|
||||
),
|
||||
// ExportData(
|
||||
// searchQuery: searchQuery,
|
||||
// refreshBusinessDataOverviewFuture:
|
||||
// refreshBusinessDataOverviewBusinessFuture,
|
||||
// updateBusinessesWithJobCallback:
|
||||
// updateOverviewBusinessesJobsCallback,
|
||||
// themeCallback: themeCallback,
|
||||
// refreshJobDataOverviewFuture: refreshBusinessDataOverviewJobFuture,
|
||||
// updateBusinessesCallback: updateOverviewBusinessesBusinessCallback)
|
||||
],
|
||||
),
|
||||
FutureBuilder(
|
||||
future: refreshBusinessDataOverviewFuture,
|
||||
builder: (context, snapshot) {
|
||||
if (snapshot.connectionState == ConnectionState.done) {
|
||||
if (snapshot.hasData) {
|
||||
if (snapshot.data.runtimeType == String) {
|
||||
_isPreviousData = false;
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
child: Column(children: [
|
||||
Center(
|
||||
child: Text(snapshot.data,
|
||||
textAlign: TextAlign.center)),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: FilledButton(
|
||||
child: const Text('Retry'),
|
||||
onPressed: () {
|
||||
_updateOverviewBusinesses();
|
||||
},
|
||||
),
|
||||
),
|
||||
]),
|
||||
));
|
||||
}
|
||||
|
||||
overviewBusinesses = snapshot.data;
|
||||
_isPreviousData = true;
|
||||
|
||||
return BusinessDisplayPanel(
|
||||
groupedBusinesses:
|
||||
_filterBySearch(overviewBusinesses),
|
||||
widescreen: widescreen,
|
||||
selectable: false);
|
||||
} else if (snapshot.hasError) {
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 16.0, right: 16.0),
|
||||
child: Text(
|
||||
'Error when loading data! Error: ${snapshot.error}'),
|
||||
));
|
||||
}
|
||||
} else if (snapshot.connectionState ==
|
||||
ConnectionState.waiting) {
|
||||
if (_isPreviousData) {
|
||||
return BusinessDisplayPanel(
|
||||
groupedBusinesses:
|
||||
_filterBySearch(overviewBusinesses),
|
||||
widescreen: widescreen,
|
||||
selectable: false);
|
||||
} else {
|
||||
return SliverToBoxAdapter(
|
||||
child: Container(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
alignment: Alignment.center,
|
||||
child: const SizedBox(
|
||||
width: 75,
|
||||
height: 75,
|
||||
child: RiveAnimation.asset(
|
||||
'assets/mdev_triangle_loading.riv'),
|
||||
),
|
||||
));
|
||||
}
|
||||
}
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
'\nError: ${snapshot.error}',
|
||||
style: const TextStyle(fontSize: 18),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
),
|
||||
);
|
||||
}),
|
||||
const SliverToBoxAdapter(
|
||||
child: SizedBox(
|
||||
height: 80,
|
||||
),
|
||||
)
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget? _getFAB() {
|
||||
if (loggedIn) {
|
||||
return FloatingActionButton(
|
||||
child: const Icon(Icons.add_business),
|
||||
onPressed: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => const CreateEditBusiness()));
|
||||
},
|
||||
);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
PreferredSizeWidget? _getBottom() {
|
||||
if (MediaQuery.sizeOf(context).width <= 1000) {
|
||||
return PreferredSize(
|
||||
preferredSize: const Size.fromHeight(0),
|
||||
child: SizedBox(
|
||||
// color: Theme.of(context).colorScheme.background,
|
||||
height: 70,
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(10),
|
||||
child: BusinessSearchBar(
|
||||
filters: jobTypeFilters,
|
||||
setFiltersCallback: _setFilters,
|
||||
setSearchCallback: _setSearch),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
// class FABAnimator extends FloatingActionButtonAnimator {
|
||||
// @override
|
||||
// Offset getOffset({Offset begin, Offset end, double progress}) {
|
||||
// return end;
|
||||
// }
|
||||
//
|
||||
// @override
|
||||
// Animation<double> getRotationAnimation({required Animation<double> parent}) {
|
||||
// return Tween<double>(begin: 0.0, end: 1.0).animate(parent);
|
||||
// throw UnimplementedError();
|
||||
// }
|
||||
//
|
||||
// @override
|
||||
// Animation<double> getScaleAnimation({required Animation<double> parent}) {
|
||||
// return Tween<double>(begin: 0.0, end: 1.0).animate(parent);
|
||||
// throw UnimplementedError();
|
||||
// }
|
||||
// }
|
||||
|
||||
@ -1,10 +1,9 @@
|
||||
import 'package:fbla_ui/home.dart';
|
||||
import 'package:fbla_ui/shared/global_vars.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
ThemeMode themeMode = ThemeMode.system;
|
||||
|
||||
void main() async {
|
||||
WidgetsFlutterBinding.ensureInitialized();
|
||||
|
||||
@ -23,9 +22,9 @@ void main() async {
|
||||
}
|
||||
|
||||
class MainApp extends StatefulWidget {
|
||||
final bool? isDark;
|
||||
final int? initialPage;
|
||||
|
||||
const MainApp({super.key, this.isDark});
|
||||
const MainApp({super.key, this.initialPage});
|
||||
|
||||
@override
|
||||
State<MainApp> createState() => _MainAppState();
|
||||
@ -72,7 +71,7 @@ class _MainAppState extends State<MainApp> {
|
||||
darkTheme: ThemeData(
|
||||
colorScheme: ColorScheme.dark(
|
||||
brightness: Brightness.dark,
|
||||
primary: Colors.blue,
|
||||
primary: Colors.blue.shade700,
|
||||
onPrimary: Colors.white,
|
||||
secondary: Colors.blue.shade900,
|
||||
surface: const Color.fromARGB(255, 31, 31, 31),
|
||||
@ -86,7 +85,7 @@ class _MainAppState extends State<MainApp> {
|
||||
theme: ThemeData(
|
||||
colorScheme: ColorScheme.light(
|
||||
brightness: Brightness.light,
|
||||
primary: Colors.blue,
|
||||
primary: Colors.blue.shade700,
|
||||
onPrimary: Colors.white,
|
||||
secondary: Colors.blue.shade200,
|
||||
surface: Colors.grey.shade200,
|
||||
@ -98,7 +97,7 @@ class _MainAppState extends State<MainApp> {
|
||||
const InputDecorationTheme(border: UnderlineInputBorder()),
|
||||
useMaterial3: true,
|
||||
),
|
||||
home: Home(themeCallback: _switchTheme),
|
||||
home: Home(themeCallback: _switchTheme, initialPage: widget.initialPage),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@ -1,24 +1,20 @@
|
||||
import 'package:fbla_ui/api_logic.dart';
|
||||
import 'package:fbla_ui/main.dart';
|
||||
import 'package:fbla_ui/pages/create_edit_business.dart';
|
||||
import 'package:fbla_ui/pages/create_edit_listing.dart';
|
||||
import 'package:fbla_ui/pages/listing_detail.dart';
|
||||
import 'package:fbla_ui/pages/signin_page.dart';
|
||||
import 'package:fbla_ui/shared.dart';
|
||||
import 'package:fbla_ui/shared/api_logic.dart';
|
||||
import 'package:fbla_ui/shared/global_vars.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:rive/rive.dart';
|
||||
import 'package:url_launcher/url_launcher.dart';
|
||||
|
||||
import '../shared/utils.dart';
|
||||
|
||||
class BusinessDetail extends StatefulWidget {
|
||||
final int id;
|
||||
final String name;
|
||||
final JobType clickFromType;
|
||||
|
||||
const BusinessDetail(
|
||||
{super.key,
|
||||
required this.id,
|
||||
required this.name,
|
||||
required this.clickFromType});
|
||||
const BusinessDetail({super.key, required this.id, required this.name});
|
||||
|
||||
@override
|
||||
State<BusinessDetail> createState() => _CreateBusinessDetailState();
|
||||
@ -45,7 +41,7 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(snapshot.data.name),
|
||||
actions: _getActions(snapshot.data, widget.clickFromType),
|
||||
actions: _getActions(snapshot.data),
|
||||
),
|
||||
body: _detailBody(snapshot.data),
|
||||
);
|
||||
@ -120,12 +116,12 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
child: Column(
|
||||
children: [
|
||||
ListTile(
|
||||
title: Text(business.name,
|
||||
title: Text(business.name!,
|
||||
textAlign: TextAlign.left,
|
||||
style: const TextStyle(
|
||||
fontSize: 24, fontWeight: FontWeight.bold)),
|
||||
subtitle: Text(
|
||||
business.description,
|
||||
business.description!,
|
||||
textAlign: TextAlign.left,
|
||||
),
|
||||
leading: ClipRRect(
|
||||
@ -134,8 +130,8 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
width: 48,
|
||||
height: 48, errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return getIconFromJobType(widget.clickFromType, 48,
|
||||
Theme.of(context).colorScheme.onSurface);
|
||||
return Icon(getIconFromBusinessType(business.type!),
|
||||
size: 48);
|
||||
}),
|
||||
),
|
||||
),
|
||||
@ -143,13 +139,13 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
leading: const Icon(Icons.link),
|
||||
title: const Text('Website'),
|
||||
subtitle: Text(
|
||||
business.website
|
||||
business.website!
|
||||
.replaceAll('https://', '')
|
||||
.replaceAll('http://', '')
|
||||
.replaceAll('www.', ''),
|
||||
style: const TextStyle(color: Colors.blue)),
|
||||
onTap: () {
|
||||
launchUrl(Uri.parse(business.website));
|
||||
launchUrl(Uri.parse(business.website!));
|
||||
},
|
||||
),
|
||||
],
|
||||
@ -157,6 +153,7 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
),
|
||||
),
|
||||
// Available positions
|
||||
if (business.listings != null)
|
||||
Card(
|
||||
clipBehavior: Clip.antiAlias,
|
||||
child:
|
||||
@ -185,9 +182,8 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
),
|
||||
],
|
||||
),
|
||||
Visibility(
|
||||
visible: business.contactPhone != null,
|
||||
child: ListTile(
|
||||
if (business.contactPhone != null)
|
||||
ListTile(
|
||||
leading: const Icon(Icons.phone),
|
||||
title: Text(business.contactPhone!),
|
||||
// maybe replace ! with ?? ''. same is true for below
|
||||
@ -221,10 +217,10 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
});
|
||||
},
|
||||
),
|
||||
),
|
||||
if (business.contactEmail != null)
|
||||
ListTile(
|
||||
leading: const Icon(Icons.email),
|
||||
title: Text(business.contactEmail),
|
||||
title: Text(business.contactEmail!),
|
||||
onTap: () {
|
||||
launchUrl(Uri.parse('mailto:${business.contactEmail}'));
|
||||
},
|
||||
@ -233,8 +229,7 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
),
|
||||
),
|
||||
// Location
|
||||
Visibility(
|
||||
child: Card(
|
||||
Card(
|
||||
clipBehavior: Clip.antiAlias,
|
||||
child: ListTile(
|
||||
leading: const Icon(Icons.location_on),
|
||||
@ -246,11 +241,9 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
},
|
||||
),
|
||||
),
|
||||
),
|
||||
// Notes
|
||||
Visibility(
|
||||
visible: business.notes != null && business.notes != '',
|
||||
child: Card(
|
||||
if (business.notes != null && business.notes != '')
|
||||
Card(
|
||||
child: ListTile(
|
||||
leading: const Icon(Icons.notes),
|
||||
title: const Text(
|
||||
@ -260,12 +253,11 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
subtitle: Text(business.notes!),
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
List<Widget>? _getActions(Business business, JobType clickFromType) {
|
||||
List<Widget>? _getActions(Business business) {
|
||||
if (loggedIn) {
|
||||
return [
|
||||
IconButton(
|
||||
@ -274,7 +266,6 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
||||
Navigator.of(context).push(MaterialPageRoute(
|
||||
builder: (context) => CreateEditBusiness(
|
||||
inputBusiness: business,
|
||||
clickFromType: clickFromType,
|
||||
)));
|
||||
},
|
||||
),
|
||||
@ -354,8 +345,7 @@ class _JobListItem extends StatelessWidget {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return ListTile(
|
||||
leading: getIconFromJobType(
|
||||
jobListing.type, 24, Theme.of(context).colorScheme.onSurface),
|
||||
leading: Icon(getIconFromJobType(jobListing.type!)),
|
||||
title: Text(jobListing.name),
|
||||
subtitle: Text(
|
||||
jobListing.description,
|
||||
|
||||
581
fbla_ui/lib/pages/businesses_overview.dart
Normal file
581
fbla_ui/lib/pages/businesses_overview.dart
Normal file
@ -0,0 +1,581 @@
|
||||
import 'package:fbla_ui/pages/business_detail.dart';
|
||||
import 'package:fbla_ui/pages/create_edit_business.dart';
|
||||
import 'package:fbla_ui/shared/api_logic.dart';
|
||||
import 'package:fbla_ui/shared/export.dart';
|
||||
import 'package:fbla_ui/shared/global_vars.dart';
|
||||
import 'package:fbla_ui/shared/utils.dart';
|
||||
import 'package:fbla_ui/shared/widgets.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/rendering.dart';
|
||||
import 'package:flutter_sticky_header/flutter_sticky_header.dart';
|
||||
import 'package:rive/rive.dart';
|
||||
import 'package:sliver_tools/sliver_tools.dart';
|
||||
import 'package:url_launcher/url_launcher.dart';
|
||||
|
||||
class BusinessesOverview extends StatefulWidget {
|
||||
final String searchQuery;
|
||||
final Future refreshBusinessDataOverviewFuture;
|
||||
final Future<void> Function(Set<BusinessType>) updateBusinessesCallback;
|
||||
final void Function() themeCallback;
|
||||
final void Function(bool) updateLoggedIn;
|
||||
|
||||
const BusinessesOverview({
|
||||
super.key,
|
||||
required this.searchQuery,
|
||||
required this.refreshBusinessDataOverviewFuture,
|
||||
required this.updateBusinessesCallback,
|
||||
required this.themeCallback,
|
||||
required this.updateLoggedIn,
|
||||
});
|
||||
|
||||
@override
|
||||
State<BusinessesOverview> createState() => _BusinessesOverviewState();
|
||||
}
|
||||
|
||||
class _BusinessesOverviewState extends State<BusinessesOverview> {
|
||||
bool _isPreviousData = false;
|
||||
late Map<BusinessType, List<Business>> overviewBusinesses;
|
||||
Set<BusinessType> businessTypeFilters = <BusinessType>{};
|
||||
String searchQuery = '';
|
||||
ScrollController controller = ScrollController();
|
||||
bool _extended = true;
|
||||
double prevPixelPosition = 0;
|
||||
|
||||
Map<BusinessType, List<Business>> _filterBySearch(
|
||||
Map<BusinessType, List<Business>> businesses, String query) {
|
||||
Map<BusinessType, List<Business>> filteredBusinesses = {};
|
||||
|
||||
for (BusinessType businessType in businesses.keys) {
|
||||
filteredBusinesses[businessType] = List.from(businesses[businessType]!
|
||||
.where((element) => element.name!
|
||||
.replaceAll(RegExp(r'[^a-zA-Z]'), '')
|
||||
.toLowerCase()
|
||||
.contains(query
|
||||
.replaceAll(RegExp(r'[^a-zA-Z]'), '')
|
||||
.toLowerCase()
|
||||
.trim())));
|
||||
}
|
||||
|
||||
filteredBusinesses.removeWhere((key, value) => value.isEmpty);
|
||||
return filteredBusinesses;
|
||||
}
|
||||
|
||||
void _setSearch(String search) async {
|
||||
setState(() {
|
||||
searchQuery = search;
|
||||
});
|
||||
}
|
||||
|
||||
void _setFilters(Set<BusinessType> filters) async {
|
||||
businessTypeFilters = Set.from(filters);
|
||||
widget.updateBusinessesCallback(businessTypeFilters);
|
||||
}
|
||||
|
||||
void _scrollListener() {
|
||||
if ((prevPixelPosition - controller.position.pixels).abs() > 10) {
|
||||
setState(() {
|
||||
_extended =
|
||||
controller.position.userScrollDirection == ScrollDirection.forward;
|
||||
});
|
||||
}
|
||||
prevPixelPosition = controller.position.pixels;
|
||||
}
|
||||
|
||||
void _generatePDF() {
|
||||
List<Business> allBusinesses = [];
|
||||
for (List<Business> businessList
|
||||
in _filterBySearch(overviewBusinesses, searchQuery).values) {
|
||||
allBusinesses.addAll(businessList);
|
||||
}
|
||||
|
||||
generatePDF(
|
||||
context: context,
|
||||
documentTypeIndex: 0,
|
||||
selectedBusinesses: Set.from(allBusinesses));
|
||||
}
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
|
||||
controller.addListener(_scrollListener);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
bool widescreen = MediaQuery.sizeOf(context).width >= widescreenWidth;
|
||||
return Scaffold(
|
||||
floatingActionButton: _getFAB(widescreen),
|
||||
body: CustomScrollView(
|
||||
controller: controller,
|
||||
slivers: [
|
||||
MainSliverAppBar(
|
||||
widescreen: widescreen,
|
||||
setSearch: _setSearch,
|
||||
searchHintText: 'Search Businesses',
|
||||
themeCallback: widget.themeCallback,
|
||||
filterIconButton: _filterIconButton(
|
||||
businessTypeFilters,
|
||||
),
|
||||
updateLoggedIn: widget.updateLoggedIn,
|
||||
generatePDF: _generatePDF,
|
||||
),
|
||||
FutureBuilder(
|
||||
future: widget.refreshBusinessDataOverviewFuture,
|
||||
builder: (context, snapshot) {
|
||||
if (snapshot.connectionState == ConnectionState.done) {
|
||||
if (snapshot.hasData) {
|
||||
if (snapshot.data.runtimeType == String) {
|
||||
_isPreviousData = false;
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
child: Column(children: [
|
||||
Center(
|
||||
child: Text(snapshot.data,
|
||||
textAlign: TextAlign.center)),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: FilledButton(
|
||||
child: const Text('Retry'),
|
||||
onPressed: () {
|
||||
widget.updateBusinessesCallback(
|
||||
businessTypeFilters);
|
||||
},
|
||||
),
|
||||
),
|
||||
]),
|
||||
));
|
||||
}
|
||||
|
||||
overviewBusinesses = snapshot.data;
|
||||
_isPreviousData = true;
|
||||
|
||||
return BusinessDisplayPanel(
|
||||
groupedBusinesses:
|
||||
_filterBySearch(overviewBusinesses, searchQuery),
|
||||
widescreen: widescreen,
|
||||
);
|
||||
} else if (snapshot.hasError) {
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 16.0, right: 16.0),
|
||||
child: Text(
|
||||
'Error when loading data! Error: ${snapshot.error}'),
|
||||
));
|
||||
}
|
||||
} else if (snapshot.connectionState ==
|
||||
ConnectionState.waiting) {
|
||||
if (_isPreviousData) {
|
||||
return BusinessDisplayPanel(
|
||||
groupedBusinesses:
|
||||
_filterBySearch(overviewBusinesses, searchQuery),
|
||||
widescreen: widescreen,
|
||||
);
|
||||
} else {
|
||||
return SliverToBoxAdapter(
|
||||
child: Container(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
alignment: Alignment.center,
|
||||
child: const SizedBox(
|
||||
width: 75,
|
||||
height: 75,
|
||||
child: RiveAnimation.asset(
|
||||
'assets/mdev_triangle_loading.riv'),
|
||||
),
|
||||
));
|
||||
}
|
||||
}
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
'\nError: ${snapshot.error}',
|
||||
style: const TextStyle(fontSize: 18),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
),
|
||||
);
|
||||
}),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _filterIconButton(Set<BusinessType> filters) {
|
||||
Set<BusinessType> selectedChips = Set.from(filters);
|
||||
|
||||
return IconButton(
|
||||
icon: Icon(
|
||||
Icons.filter_list,
|
||||
color: filters.isNotEmpty
|
||||
? Theme.of(context).colorScheme.primary
|
||||
: Theme.of(context).colorScheme.onSurface,
|
||||
),
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return StatefulBuilder(
|
||||
builder: (BuildContext context, StateSetter setState) {
|
||||
void setDialogState(Set<BusinessType> newFilters) {
|
||||
setState(() {
|
||||
filters = newFilters;
|
||||
});
|
||||
}
|
||||
|
||||
List<Padding> chips = [];
|
||||
for (var type in BusinessType.values) {
|
||||
chips.add(Padding(
|
||||
padding: const EdgeInsets.all(4),
|
||||
child: FilterChip(
|
||||
showCheckmark: false,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(20)),
|
||||
label: Text(getNameFromBusinessType(type)),
|
||||
selected: selectedChips.contains(type),
|
||||
onSelected: (bool selected) {
|
||||
if (selected) {
|
||||
selectedChips.add(type);
|
||||
} else {
|
||||
selectedChips.remove(type);
|
||||
}
|
||||
setDialogState(filters);
|
||||
}),
|
||||
));
|
||||
}
|
||||
|
||||
return AlertDialog(
|
||||
title: const Text('Filter Options'),
|
||||
content: SizedBox(
|
||||
width: 400,
|
||||
child: Wrap(
|
||||
children: chips,
|
||||
),
|
||||
),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Reset'),
|
||||
onPressed: () {
|
||||
_setFilters(<BusinessType>{});
|
||||
// selectedChips = <BusinessType>{};
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
),
|
||||
TextButton(
|
||||
child: const Text('Cancel'),
|
||||
onPressed: () {
|
||||
// selectedChips = Set.from(filters);
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
),
|
||||
TextButton(
|
||||
child: const Text('Apply'),
|
||||
onPressed: () {
|
||||
_setFilters(selectedChips);
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
)
|
||||
],
|
||||
);
|
||||
});
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
Widget? _getFAB(bool widescreen) {
|
||||
if (!widescreen && loggedIn) {
|
||||
return FloatingActionButton.extended(
|
||||
extendedIconLabelSpacing: _extended ? 8.0 : 0,
|
||||
extendedPadding: const EdgeInsets.symmetric(horizontal: 16),
|
||||
icon: const Icon(Icons.add),
|
||||
label: AnimatedSize(
|
||||
curve: Easing.standard,
|
||||
duration: const Duration(milliseconds: 300),
|
||||
child: _extended ? const Text('Add Business') : Container(),
|
||||
),
|
||||
onPressed: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => const CreateEditBusiness()));
|
||||
},
|
||||
);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
class BusinessDisplayPanel extends StatefulWidget {
|
||||
final Map<BusinessType, List<Business>> groupedBusinesses;
|
||||
final bool widescreen;
|
||||
|
||||
const BusinessDisplayPanel({
|
||||
super.key,
|
||||
required this.groupedBusinesses,
|
||||
required this.widescreen,
|
||||
});
|
||||
|
||||
@override
|
||||
State<BusinessDisplayPanel> createState() => _BusinessDisplayPanelState();
|
||||
}
|
||||
|
||||
class _BusinessDisplayPanelState extends State<BusinessDisplayPanel> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
if (widget.groupedBusinesses.keys.isEmpty) {
|
||||
return const SliverToBoxAdapter(
|
||||
child: Center(
|
||||
child: Padding(
|
||||
padding: EdgeInsets.all(16.0),
|
||||
child: Text(
|
||||
'No results found!\nPlease change your search filters.',
|
||||
textAlign: TextAlign.center,
|
||||
style: TextStyle(fontSize: 18),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
List<BusinessHeader> headers = [];
|
||||
for (BusinessType businessType in widget.groupedBusinesses.keys) {
|
||||
headers.add(BusinessHeader(
|
||||
businessType: businessType,
|
||||
widescreen: widget.widescreen,
|
||||
businesses: widget.groupedBusinesses[businessType]!));
|
||||
}
|
||||
headers
|
||||
.sort((a, b) => a.businessType.index.compareTo(b.businessType.index));
|
||||
return MultiSliver(children: headers);
|
||||
}
|
||||
}
|
||||
|
||||
class BusinessHeader extends StatefulWidget {
|
||||
final BusinessType businessType;
|
||||
final List<Business> businesses;
|
||||
final bool widescreen;
|
||||
|
||||
const BusinessHeader({
|
||||
super.key,
|
||||
required this.businessType,
|
||||
required this.businesses,
|
||||
required this.widescreen,
|
||||
});
|
||||
|
||||
@override
|
||||
State<BusinessHeader> createState() => _BusinessHeaderState();
|
||||
}
|
||||
|
||||
class _BusinessHeaderState extends State<BusinessHeader> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return SliverStickyHeader(
|
||||
header: Container(
|
||||
height: 55.0,
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
padding: const EdgeInsets.symmetric(horizontal: 16.0),
|
||||
alignment: Alignment.centerLeft,
|
||||
child: _getHeaderRow(),
|
||||
),
|
||||
sliver: _getChildSliver(widget.businesses, widget.widescreen),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _getHeaderRow() {
|
||||
return Row(
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(left: 4.0, right: 12.0),
|
||||
child: Icon(
|
||||
getIconFromBusinessType(widget.businessType),
|
||||
color: Theme.of(context).colorScheme.onPrimary,
|
||||
)),
|
||||
Text(getNameFromBusinessType(widget.businessType)),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
Widget _getChildSliver(List<Business> businesses, bool widescreen) {
|
||||
if (widescreen) {
|
||||
return SliverPadding(
|
||||
padding: const EdgeInsets.all(4),
|
||||
sliver: SliverGrid(
|
||||
gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent(
|
||||
mainAxisExtent: 250.0,
|
||||
maxCrossAxisExtent: 400.0,
|
||||
mainAxisSpacing: 4.0,
|
||||
crossAxisSpacing: 4.0,
|
||||
),
|
||||
delegate: SliverChildBuilderDelegate(
|
||||
childCount: businesses.length,
|
||||
(BuildContext context, int index) {
|
||||
return _businessTile(
|
||||
businesses[index],
|
||||
widget.businessType,
|
||||
);
|
||||
},
|
||||
),
|
||||
),
|
||||
);
|
||||
} else {
|
||||
return SliverList(
|
||||
delegate: SliverChildBuilderDelegate(
|
||||
childCount: businesses.length,
|
||||
(BuildContext context, int index) {
|
||||
return _businessListItem(
|
||||
businesses[index],
|
||||
widget.businessType,
|
||||
);
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Widget _businessTile(Business business, BusinessType jobType) {
|
||||
return MouseRegion(
|
||||
cursor: SystemMouseCursors.click,
|
||||
child: GestureDetector(
|
||||
onTap: () {
|
||||
Navigator.of(context).push(MaterialPageRoute(
|
||||
builder: (context) => BusinessDetail(
|
||||
id: business.id,
|
||||
name: business.name!,
|
||||
)));
|
||||
},
|
||||
child: Card(
|
||||
clipBehavior: Clip.antiAlias,
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.center,
|
||||
children: [
|
||||
Row(
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(6.0),
|
||||
child: Image.network('$apiAddress/logos/${business.id}',
|
||||
height: 48,
|
||||
width: 48, errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return Icon(getIconFromBusinessType(business.type!),
|
||||
size: 48);
|
||||
}),
|
||||
)),
|
||||
Flexible(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
business.name!,
|
||||
style: const TextStyle(
|
||||
fontSize: 18, fontWeight: FontWeight.bold),
|
||||
maxLines: 2,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
business.description!,
|
||||
maxLines: 5,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
),
|
||||
const Spacer(),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||
children: [
|
||||
IconButton(
|
||||
icon: const Icon(Icons.link),
|
||||
onPressed: () {
|
||||
launchUrl(Uri.parse('https://${business.website}'));
|
||||
},
|
||||
),
|
||||
IconButton(
|
||||
icon: const Icon(Icons.location_on),
|
||||
onPressed: () {
|
||||
launchUrl(Uri.parse(Uri.encodeFull(
|
||||
'https://www.google.com/maps/search/?api=1&query=${business.locationName}')));
|
||||
},
|
||||
),
|
||||
if (business.contactPhone != null)
|
||||
IconButton(
|
||||
icon: const Icon(Icons.phone),
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
backgroundColor:
|
||||
Theme.of(context).colorScheme.surface,
|
||||
title:
|
||||
Text('Contact ${business.contactName}'),
|
||||
content: Text(
|
||||
'Would you like to call or text ${business.contactName}?'),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Text'),
|
||||
onPressed: () {
|
||||
launchUrl(Uri.parse(
|
||||
'sms:${business.contactPhone}'));
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Call'),
|
||||
onPressed: () async {
|
||||
launchUrl(Uri.parse(
|
||||
'tel:${business.contactPhone}'));
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
],
|
||||
);
|
||||
});
|
||||
},
|
||||
),
|
||||
if (business.contactEmail != null)
|
||||
IconButton(
|
||||
icon: const Icon(Icons.email),
|
||||
onPressed: () {
|
||||
launchUrl(
|
||||
Uri.parse('mailto:${business.contactEmail}'));
|
||||
},
|
||||
),
|
||||
],
|
||||
)),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _businessListItem(Business business, BusinessType? jobType) {
|
||||
return Card(
|
||||
child: ListTile(
|
||||
leading: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(3.0),
|
||||
child: Image.network('$apiAddress/logos/${business.id}',
|
||||
height: 24, width: 24, errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return Icon(getIconFromBusinessType(business.type!));
|
||||
})),
|
||||
title: Text(business.name!),
|
||||
subtitle: Text(business.description!,
|
||||
maxLines: 2, overflow: TextOverflow.ellipsis),
|
||||
onTap: () {
|
||||
Navigator.of(context).push(MaterialPageRoute(
|
||||
builder: (context) => BusinessDetail(
|
||||
id: business.id,
|
||||
name: business.name!,
|
||||
)));
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
@ -1,14 +1,13 @@
|
||||
import 'package:fbla_ui/api_logic.dart';
|
||||
import 'package:fbla_ui/main.dart';
|
||||
import 'package:fbla_ui/shared.dart';
|
||||
import 'package:fbla_ui/shared/api_logic.dart';
|
||||
import 'package:fbla_ui/shared/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
|
||||
class CreateEditBusiness extends StatefulWidget {
|
||||
final Business? inputBusiness;
|
||||
final JobType? clickFromType;
|
||||
|
||||
const CreateEditBusiness({super.key, this.inputBusiness, this.clickFromType});
|
||||
const CreateEditBusiness({super.key, this.inputBusiness});
|
||||
|
||||
@override
|
||||
State<CreateEditBusiness> createState() => _CreateEditBusinessState();
|
||||
@ -25,19 +24,23 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
late TextEditingController _locationNameController;
|
||||
late TextEditingController _locationAddressController;
|
||||
|
||||
// late TextEditingController _businessTypeController;
|
||||
|
||||
Business business = Business(
|
||||
id: 0,
|
||||
name: 'Business',
|
||||
description: 'Add details about the business below.',
|
||||
type: null,
|
||||
website: '',
|
||||
contactName: '',
|
||||
contactEmail: '',
|
||||
contactPhone: '',
|
||||
notes: '',
|
||||
contactName: null,
|
||||
contactEmail: null,
|
||||
contactPhone: null,
|
||||
notes: null,
|
||||
locationName: '',
|
||||
locationAddress: '',
|
||||
locationAddress: null,
|
||||
);
|
||||
bool _isLoading = false;
|
||||
String? dropDownErrorText;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
@ -47,11 +50,16 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
_nameController = TextEditingController(text: business.name);
|
||||
_descriptionController =
|
||||
TextEditingController(text: business.description);
|
||||
business.type = widget.inputBusiness?.type;
|
||||
} else {
|
||||
_nameController = TextEditingController();
|
||||
_descriptionController = TextEditingController();
|
||||
}
|
||||
_websiteController = TextEditingController(text: business.website);
|
||||
_websiteController = TextEditingController(
|
||||
text: business.website!
|
||||
.replaceAll('https://', '')
|
||||
.replaceAll('http://', '')
|
||||
.replaceAll('www.', ''));
|
||||
_contactNameController = TextEditingController(text: business.contactName);
|
||||
_contactPhoneController =
|
||||
TextEditingController(text: business.contactPhone);
|
||||
@ -65,7 +73,6 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
}
|
||||
|
||||
final formKey = GlobalKey<FormState>();
|
||||
final TextEditingController businessTypeController = TextEditingController();
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
@ -91,15 +98,21 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
)
|
||||
: const Icon(Icons.save),
|
||||
onPressed: () async {
|
||||
if (business.type == null) {
|
||||
setState(() {
|
||||
dropDownErrorText = 'Business type is required';
|
||||
});
|
||||
formKey.currentState!.validate();
|
||||
} else {
|
||||
setState(() {
|
||||
dropDownErrorText = null;
|
||||
});
|
||||
if (formKey.currentState!.validate()) {
|
||||
formKey.currentState?.save();
|
||||
setState(() {
|
||||
_isLoading = true;
|
||||
});
|
||||
String? result;
|
||||
// if (business.contactName == '') {
|
||||
// business.contactName = 'Contact ${business.name}';
|
||||
// }
|
||||
if (widget.inputBusiness != null) {
|
||||
result = await editBusiness(business);
|
||||
} else {
|
||||
@ -130,22 +143,23 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
},
|
||||
),
|
||||
body: ListView(
|
||||
children: [
|
||||
Center(
|
||||
child: SizedBox(
|
||||
width: 1000,
|
||||
width: 800,
|
||||
child: Column(
|
||||
children: [
|
||||
ListTile(
|
||||
title: Text(business.name,
|
||||
title: Text(business.name!,
|
||||
textAlign: TextAlign.left,
|
||||
style: const TextStyle(
|
||||
fontSize: 24, fontWeight: FontWeight.bold)),
|
||||
subtitle: Text(
|
||||
business.description,
|
||||
business.description!,
|
||||
textAlign: TextAlign.left,
|
||||
),
|
||||
leading: ClipRRect(
|
||||
@ -156,10 +170,11 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
'https://logo.clearbit.com/${business.website}',
|
||||
errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return getIconFromJobType(
|
||||
widget.clickFromType ?? JobType.other,
|
||||
48,
|
||||
Theme.of(context).colorScheme.onSurface);
|
||||
return Icon(
|
||||
getIconFromBusinessType(business.type != null
|
||||
? business.type!
|
||||
: BusinessType.other),
|
||||
size: 48);
|
||||
}),
|
||||
),
|
||||
),
|
||||
@ -186,44 +201,13 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
labelText: 'Business Name (required)',
|
||||
),
|
||||
validator: (value) {
|
||||
if (value != null && value.isEmpty) {
|
||||
if (value != null && value.trim().isEmpty) {
|
||||
return 'Name is required';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0, bottom: 8.0),
|
||||
child: TextFormField(
|
||||
controller: _websiteController,
|
||||
autovalidateMode:
|
||||
AutovalidateMode.onUserInteraction,
|
||||
keyboardType: TextInputType.url,
|
||||
onChanged: (inputUrl) {
|
||||
business.website = Uri.encodeFull(inputUrl);
|
||||
if (!business.website.contains('http://') &&
|
||||
!business.website
|
||||
.contains('https://')) {
|
||||
business.website =
|
||||
'https://${business.website}';
|
||||
}
|
||||
},
|
||||
onTapOutside: (PointerDownEvent event) {
|
||||
FocusScope.of(context).unfocus();
|
||||
},
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Website (required)',
|
||||
),
|
||||
validator: (value) {
|
||||
if (value != null && value.isEmpty) {
|
||||
return 'Website is required';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0),
|
||||
@ -246,13 +230,82 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
'Business Description (required)',
|
||||
),
|
||||
validator: (value) {
|
||||
if (value != null && value.isEmpty) {
|
||||
if (value != null && value.trim().isEmpty) {
|
||||
return 'Description is required';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0, bottom: 16.0),
|
||||
child: TextFormField(
|
||||
controller: _websiteController,
|
||||
autovalidateMode:
|
||||
AutovalidateMode.onUserInteraction,
|
||||
keyboardType: TextInputType.url,
|
||||
onChanged: (inputUrl) {
|
||||
business.website = Uri.encodeFull(inputUrl);
|
||||
if (!business.website!
|
||||
.contains('http://') &&
|
||||
!business.website!
|
||||
.contains('https://')) {
|
||||
business.website =
|
||||
'https://${business.website}';
|
||||
}
|
||||
},
|
||||
onTapOutside: (PointerDownEvent event) {
|
||||
FocusScope.of(context).unfocus();
|
||||
},
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Website (required)',
|
||||
),
|
||||
validator: (value) {
|
||||
if (value != null &&
|
||||
!RegExp(r'(?:[a-zA-Z0-9](?:[a-zA-Z0-9\-]{0,61}[a-zA-Z0-9])?\.)+[a-zA-Z]{2,}(?:\/[^\/\s]*)*')
|
||||
.hasMatch(value)) {
|
||||
return 'Enter a valid Website';
|
||||
}
|
||||
if (value != null && value.trim().isEmpty) {
|
||||
return 'Website is required';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0, bottom: 8.0),
|
||||
child: Row(
|
||||
mainAxisAlignment:
|
||||
MainAxisAlignment.spaceBetween,
|
||||
children: [
|
||||
const Text('Type of Business',
|
||||
style: TextStyle(fontSize: 16)),
|
||||
DropdownMenu<BusinessType>(
|
||||
initialSelection: business.type,
|
||||
label: const Text('Business Type'),
|
||||
errorText: dropDownErrorText,
|
||||
dropdownMenuEntries: [
|
||||
for (BusinessType type
|
||||
in BusinessType.values)
|
||||
DropdownMenuEntry(
|
||||
value: type,
|
||||
label: getNameFromBusinessType(
|
||||
type)),
|
||||
],
|
||||
onSelected: (inputType) {
|
||||
setState(() {
|
||||
business.type = inputType!;
|
||||
dropDownErrorText = null;
|
||||
});
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
|
||||
// Padding(
|
||||
// padding: const EdgeInsets.only(
|
||||
// left: 8.0, right: 8.0, bottom: 16.0),
|
||||
@ -325,43 +378,6 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
// ],
|
||||
// ),
|
||||
// ),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0, bottom: 8.0),
|
||||
child: TextFormField(
|
||||
controller: _locationNameController,
|
||||
onChanged: (inputName) {
|
||||
setState(() {
|
||||
business.locationName = inputName;
|
||||
});
|
||||
},
|
||||
onTapOutside: (PointerDownEvent event) {
|
||||
FocusScope.of(context).unfocus();
|
||||
},
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Location Name',
|
||||
),
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0, bottom: 16.0),
|
||||
child: TextFormField(
|
||||
controller: _locationAddressController,
|
||||
onChanged: (inputAddr) {
|
||||
setState(() {
|
||||
business.locationAddress = inputAddr;
|
||||
});
|
||||
},
|
||||
onTapOutside: (PointerDownEvent event) {
|
||||
FocusScope.of(context).unfocus();
|
||||
},
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Location Address',
|
||||
),
|
||||
),
|
||||
),
|
||||
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0, bottom: 8.0),
|
||||
@ -374,8 +390,17 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
FocusScope.of(context).unfocus();
|
||||
},
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Contact Information Name',
|
||||
labelText:
|
||||
'Contact Information Name (required)',
|
||||
),
|
||||
autovalidateMode:
|
||||
AutovalidateMode.onUserInteraction,
|
||||
validator: (value) {
|
||||
if (value == null || value.trim().isEmpty) {
|
||||
return 'Contact name is required';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
@ -385,15 +410,31 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
controller: _contactPhoneController,
|
||||
inputFormatters: [PhoneFormatter()],
|
||||
keyboardType: TextInputType.phone,
|
||||
onSaved: (inputText) {
|
||||
business.contactPhone = inputText!;
|
||||
autovalidateMode:
|
||||
AutovalidateMode.onUserInteraction,
|
||||
onChanged: (inputText) {
|
||||
if (inputText.trim().isEmpty) {
|
||||
business.contactPhone = null;
|
||||
} else {
|
||||
business.contactPhone = inputText.trim();
|
||||
}
|
||||
},
|
||||
onTapOutside: (PointerDownEvent event) {
|
||||
FocusScope.of(context).unfocus();
|
||||
},
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Contact Phone # (optional)',
|
||||
labelText: 'Contact Phone #',
|
||||
),
|
||||
validator: (value) {
|
||||
if (business.contactEmail == null &&
|
||||
(value == null || value.isEmpty)) {
|
||||
return 'At least one contact method is required';
|
||||
}
|
||||
if (value != null && value.length != 14) {
|
||||
return 'Enter a valid phone number';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
@ -402,9 +443,15 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
child: TextFormField(
|
||||
controller: _contactEmailController,
|
||||
keyboardType: TextInputType.emailAddress,
|
||||
onSaved: (inputText) {
|
||||
business.contactEmail = inputText!;
|
||||
onChanged: (inputText) {
|
||||
if (inputText.trim().isEmpty) {
|
||||
business.contactEmail = null;
|
||||
} else {
|
||||
business.contactEmail = inputText.trim();
|
||||
}
|
||||
},
|
||||
autovalidateMode:
|
||||
AutovalidateMode.onUserInteraction,
|
||||
onTapOutside: (PointerDownEvent event) {
|
||||
FocusScope.of(context).unfocus();
|
||||
},
|
||||
@ -412,10 +459,16 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
labelText: 'Contact Email',
|
||||
),
|
||||
validator: (value) {
|
||||
value = value?.trim();
|
||||
if (value != null && value.isEmpty) {
|
||||
value = null;
|
||||
}
|
||||
if (value == null &&
|
||||
business.contactPhone == null) {
|
||||
return 'At least one contact method is required';
|
||||
}
|
||||
if (value != null) {
|
||||
if (value.isEmpty) {
|
||||
return null;
|
||||
} else if (!RegExp(
|
||||
if (!RegExp(
|
||||
r'^(([^<>()[\]\\.,;:\s@"]+(\.[^<>()[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$')
|
||||
.hasMatch(value)) {
|
||||
return 'Enter a valid Email';
|
||||
@ -427,6 +480,58 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
},
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0, bottom: 8.0),
|
||||
child: TextFormField(
|
||||
controller: _locationNameController,
|
||||
onChanged: (inputName) {
|
||||
setState(() {
|
||||
business.locationName = inputName.trim();
|
||||
});
|
||||
},
|
||||
autovalidateMode:
|
||||
AutovalidateMode.onUserInteraction,
|
||||
onTapOutside: (PointerDownEvent event) {
|
||||
FocusScope.of(context).unfocus();
|
||||
},
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Location Name (required)',
|
||||
),
|
||||
validator: (value) {
|
||||
if (value != null && value.trim().isEmpty) {
|
||||
return 'Location name is required';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0, bottom: 16.0),
|
||||
child: TextFormField(
|
||||
controller: _locationAddressController,
|
||||
onChanged: (inputAddr) {
|
||||
setState(() {
|
||||
business.locationAddress = inputAddr;
|
||||
});
|
||||
},
|
||||
autovalidateMode:
|
||||
AutovalidateMode.onUserInteraction,
|
||||
onTapOutside: (PointerDownEvent event) {
|
||||
FocusScope.of(context).unfocus();
|
||||
},
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Location Address (required)',
|
||||
),
|
||||
validator: (value) {
|
||||
if (value != null && value.trim().isEmpty) {
|
||||
return 'Location Address is required';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 8.0, right: 8.0, bottom: 8.0),
|
||||
@ -435,7 +540,12 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
||||
maxLength: 300,
|
||||
maxLines: null,
|
||||
onSaved: (inputText) {
|
||||
business.notes = inputText!;
|
||||
if (inputText == null ||
|
||||
inputText.trim().isEmpty) {
|
||||
business.notes = null;
|
||||
} else {
|
||||
business.notes = inputText.trim();
|
||||
}
|
||||
},
|
||||
onTapOutside: (PointerDownEvent event) {
|
||||
FocusScope.of(context).unfocus();
|
||||
|
||||
@ -1,15 +1,15 @@
|
||||
import 'package:fbla_ui/api_logic.dart';
|
||||
import 'package:fbla_ui/main.dart';
|
||||
import 'package:fbla_ui/shared.dart';
|
||||
import 'package:fbla_ui/shared/api_logic.dart';
|
||||
import 'package:fbla_ui/shared/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:rive/rive.dart';
|
||||
|
||||
class CreateEditJobListing extends StatefulWidget {
|
||||
final JobListing? inputJobListing;
|
||||
final Business inputBusiness;
|
||||
final Business? inputBusiness;
|
||||
|
||||
const CreateEditJobListing(
|
||||
{super.key, this.inputJobListing, required this.inputBusiness});
|
||||
{super.key, this.inputJobListing, this.inputBusiness});
|
||||
|
||||
@override
|
||||
State<CreateEditJobListing> createState() => _CreateEditJobListingState();
|
||||
@ -22,14 +22,15 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
late TextEditingController _wageController;
|
||||
late TextEditingController _linkController;
|
||||
List nameMapping = [];
|
||||
String? businessErrorText;
|
||||
String? typeDropdownErrorText;
|
||||
String? businessDropdownErrorText;
|
||||
|
||||
JobListing listing = JobListing(
|
||||
id: null,
|
||||
businessId: null,
|
||||
name: 'Job Listing',
|
||||
description: 'Add details about the business below.',
|
||||
type: JobType.other,
|
||||
type: null,
|
||||
wage: null,
|
||||
link: null);
|
||||
bool _isLoading = false;
|
||||
@ -46,17 +47,21 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
_descriptionController = TextEditingController();
|
||||
}
|
||||
_wageController = TextEditingController(text: listing.wage);
|
||||
_linkController = TextEditingController(text: listing.link);
|
||||
_linkController = TextEditingController(
|
||||
text: listing.link
|
||||
?.replaceAll('https://', '')
|
||||
.replaceAll('http://', '')
|
||||
.replaceAll('www.', ''));
|
||||
getBusinessNameMapping = fetchBusinessNames();
|
||||
}
|
||||
|
||||
final formKey = GlobalKey<FormState>();
|
||||
final TextEditingController jobTypeController = TextEditingController();
|
||||
final TextEditingController businessController = TextEditingController();
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
listing.businessId = widget.inputBusiness.id;
|
||||
if (widget.inputBusiness != null) {
|
||||
listing.businessId = widget.inputBusiness!.id;
|
||||
}
|
||||
return PopScope(
|
||||
canPop: !_isLoading,
|
||||
onPopInvoked: _handlePop,
|
||||
@ -79,6 +84,24 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
)
|
||||
: const Icon(Icons.save),
|
||||
onPressed: () async {
|
||||
if (listing.type == null || listing.businessId == null) {
|
||||
if (listing.type == null) {
|
||||
setState(() {
|
||||
typeDropdownErrorText = 'Job type is required';
|
||||
});
|
||||
formKey.currentState!.validate();
|
||||
}
|
||||
if (listing.businessId == null) {
|
||||
setState(() {
|
||||
businessDropdownErrorText = 'Business is required';
|
||||
});
|
||||
formKey.currentState!.validate();
|
||||
}
|
||||
} else {
|
||||
setState(() {
|
||||
typeDropdownErrorText = null;
|
||||
businessDropdownErrorText = null;
|
||||
});
|
||||
if (formKey.currentState!.validate()) {
|
||||
formKey.currentState?.save();
|
||||
setState(() {
|
||||
@ -102,7 +125,9 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
Navigator.pushReplacement(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => const MainApp()));
|
||||
builder: (context) => const MainApp(
|
||||
initialPage: 1,
|
||||
)));
|
||||
}
|
||||
} else {
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
@ -115,8 +140,8 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
),
|
||||
);
|
||||
}
|
||||
},
|
||||
),
|
||||
}
|
||||
}),
|
||||
body: FutureBuilder(
|
||||
future: getBusinessNameMapping,
|
||||
builder: (context, snapshot) {
|
||||
@ -152,7 +177,7 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
children: [
|
||||
Center(
|
||||
child: SizedBox(
|
||||
width: 1000,
|
||||
width: 800,
|
||||
child: Column(
|
||||
children: [
|
||||
ListTile(
|
||||
@ -176,12 +201,11 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
errorBuilder: (BuildContext context,
|
||||
Object exception,
|
||||
StackTrace? stackTrace) {
|
||||
return getIconFromJobType(
|
||||
listing.type,
|
||||
48,
|
||||
Theme.of(context)
|
||||
.colorScheme
|
||||
.onSurface);
|
||||
return Icon(
|
||||
getIconFromJobType(
|
||||
listing.type ?? JobType.other,
|
||||
),
|
||||
size: 48);
|
||||
}),
|
||||
),
|
||||
),
|
||||
@ -204,8 +228,9 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
TextStyle(fontSize: 16)),
|
||||
DropdownMenu<JobType>(
|
||||
initialSelection: listing.type,
|
||||
controller: jobTypeController,
|
||||
label: const Text('Job Type'),
|
||||
errorText:
|
||||
typeDropdownErrorText,
|
||||
dropdownMenuEntries: [
|
||||
for (JobType type
|
||||
in JobType.values)
|
||||
@ -218,6 +243,8 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
onSelected: (inputType) {
|
||||
setState(() {
|
||||
listing.type = inputType!;
|
||||
typeDropdownErrorText =
|
||||
null;
|
||||
});
|
||||
},
|
||||
),
|
||||
@ -239,9 +266,10 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
style:
|
||||
TextStyle(fontSize: 16)),
|
||||
DropdownMenu<int>(
|
||||
errorText:
|
||||
businessDropdownErrorText,
|
||||
initialSelection:
|
||||
widget.inputBusiness.id,
|
||||
controller: businessController,
|
||||
widget.inputBusiness?.id,
|
||||
label: const Text('Business'),
|
||||
dropdownMenuEntries: [
|
||||
for (Map<String, dynamic> map
|
||||
@ -254,6 +282,8 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
setState(() {
|
||||
listing.businessId =
|
||||
inputType!;
|
||||
businessDropdownErrorText =
|
||||
null;
|
||||
});
|
||||
},
|
||||
),
|
||||
@ -353,8 +383,7 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
.onUserInteraction,
|
||||
keyboardType: TextInputType.url,
|
||||
onChanged: (inputUrl) {
|
||||
if (listing.link != null &&
|
||||
listing.link != '') {
|
||||
if (inputUrl != '') {
|
||||
listing.link =
|
||||
Uri.encodeFull(inputUrl);
|
||||
if (!listing.link!
|
||||
@ -365,6 +394,16 @@ class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||
'https://${listing.link}';
|
||||
}
|
||||
}
|
||||
listing.link = null;
|
||||
},
|
||||
validator: (value) {
|
||||
if (value != null &&
|
||||
value.isNotEmpty &&
|
||||
!RegExp(r'(?:[a-zA-Z0-9](?:[a-zA-Z0-9\-]{0,61}[a-zA-Z0-9])?\.)+[a-zA-Z]{2,}(?:\/[^\/\s]*)*')
|
||||
.hasMatch(value)) {
|
||||
return 'Enter a valid Website';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
onTapOutside:
|
||||
(PointerDownEvent event) {
|
||||
|
||||
@ -1,850 +0,0 @@
|
||||
// import 'dart:html' as html;
|
||||
import 'dart:convert';
|
||||
import 'dart:io';
|
||||
|
||||
import 'package:fbla_ui/api_logic.dart';
|
||||
import 'package:fbla_ui/shared.dart';
|
||||
import 'package:flutter/foundation.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:open_filex/open_filex.dart';
|
||||
import 'package:path_provider/path_provider.dart';
|
||||
import 'package:pdf/pdf.dart';
|
||||
import 'package:pdf/widgets.dart' as pw;
|
||||
import 'package:printing/printing.dart';
|
||||
import 'package:rive/rive.dart';
|
||||
|
||||
class ExportData extends StatefulWidget {
|
||||
final Map<JobType, List<Business>> groupedBusinesses;
|
||||
|
||||
const ExportData({super.key, required this.groupedBusinesses});
|
||||
|
||||
@override
|
||||
State<ExportData> createState() => _ExportDataState();
|
||||
}
|
||||
|
||||
class _ExportDataState extends State<ExportData> {
|
||||
String documentType = 'Business';
|
||||
late Future refreshBusinessDataFuture;
|
||||
bool _isPreviousData = false;
|
||||
late Map<JobType, List<Business>> overviewBusinesses;
|
||||
Set<JobType> jobTypeFilters = <JobType>{};
|
||||
String searchQuery = '';
|
||||
Set<DataTypeJob> selectedDataTypesJob = <DataTypeJob>{};
|
||||
Set<DataTypeBusiness> selectedDataTypesBusiness = <DataTypeBusiness>{};
|
||||
|
||||
Future<void> _setFilters(Set<JobType> filters) async {
|
||||
setState(() {
|
||||
jobTypeFilters = filters;
|
||||
});
|
||||
_updateOverviewBusinesses();
|
||||
}
|
||||
|
||||
Future<void> _updateOverviewBusinesses() async {
|
||||
var refreshedData =
|
||||
fetchBusinessDataOverview(typeFilters: jobTypeFilters.toList());
|
||||
await refreshedData;
|
||||
setState(() {
|
||||
refreshBusinessDataFuture = refreshedData;
|
||||
});
|
||||
}
|
||||
|
||||
Future<void> _setSearch(String search) async {
|
||||
setState(() {
|
||||
searchQuery = search;
|
||||
});
|
||||
_updateOverviewBusinesses();
|
||||
}
|
||||
|
||||
Map<JobType, List<Business>> _filterBySearch(
|
||||
Map<JobType, List<Business>> businesses) {
|
||||
Map<JobType, List<Business>> filteredBusinesses = businesses;
|
||||
|
||||
for (JobType jobType in businesses.keys) {
|
||||
filteredBusinesses[jobType]!.removeWhere((tmpBusiness) => !tmpBusiness
|
||||
.name
|
||||
.replaceAll(RegExp(r'[^a-zA-Z]'), '')
|
||||
.toLowerCase()
|
||||
.contains(searchQuery
|
||||
.replaceAll(RegExp(r'[^a-zA-Z]'), '')
|
||||
.toLowerCase()
|
||||
.trim()));
|
||||
}
|
||||
filteredBusinesses.removeWhere((key, value) => value.isEmpty);
|
||||
return filteredBusinesses;
|
||||
}
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
|
||||
refreshBusinessDataFuture = fetchBusinessDataOverview();
|
||||
selectedBusinesses = <Business>{};
|
||||
}
|
||||
|
||||
void _setStateCallbackReset() {
|
||||
setState(() {
|
||||
selectedDataTypesBusiness = <DataTypeBusiness>{};
|
||||
selectedDataTypesJob = <DataTypeJob>{};
|
||||
documentType = 'Business';
|
||||
});
|
||||
}
|
||||
|
||||
void _setStateCallbackApply(String docType, Set<DataTypeJob> dataFiltersJob,
|
||||
Set<DataTypeBusiness> dataFiltersBusiness) {
|
||||
setState(() {
|
||||
selectedDataTypesBusiness = dataFiltersBusiness;
|
||||
selectedDataTypesJob = dataFiltersJob;
|
||||
documentType = docType;
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
bool widescreen = MediaQuery.sizeOf(context).width >= 1000;
|
||||
return Scaffold(
|
||||
floatingActionButton: _FAB(
|
||||
groupedBusinesses: widget.groupedBusinesses,
|
||||
documentType: documentType,
|
||||
selectedDataTypesBusiness: selectedDataTypesBusiness,
|
||||
selectedDataTypesJob: selectedDataTypesJob,
|
||||
),
|
||||
body: CustomScrollView(
|
||||
slivers: [
|
||||
SliverAppBar(
|
||||
forceMaterialTransparency: false,
|
||||
title: const Text('Export Data'),
|
||||
toolbarHeight: 70,
|
||||
pinned: true,
|
||||
centerTitle: true,
|
||||
expandedHeight: 120,
|
||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
||||
actions: [
|
||||
IconButton(
|
||||
icon: const Icon(Icons.settings),
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
Set<DataTypeBusiness> dataFiltersBusinessTmp =
|
||||
Set<DataTypeBusiness>.from(
|
||||
selectedDataTypesBusiness);
|
||||
Set<DataTypeJob> dataFiltersJobTmp =
|
||||
Set<DataTypeJob>.from(selectedDataTypesJob);
|
||||
String docTypeTmp = documentType;
|
||||
return StatefulBuilder(builder: (context, setState) {
|
||||
void segmentedCallback(String docType) {
|
||||
setState(() {
|
||||
docTypeTmp = docType;
|
||||
});
|
||||
}
|
||||
|
||||
void chipsCallback(
|
||||
{Set<DataTypeJob>? selectedDataTypesJob,
|
||||
Set<DataTypeBusiness>?
|
||||
selectedDataTypesBusiness}) {
|
||||
if (selectedDataTypesJob != null) {
|
||||
dataFiltersJobTmp = selectedDataTypesJob;
|
||||
}
|
||||
if (selectedDataTypesBusiness != null) {
|
||||
dataFiltersBusinessTmp =
|
||||
selectedDataTypesBusiness;
|
||||
}
|
||||
}
|
||||
|
||||
return AlertDialog(
|
||||
// DO NOT MOVE TO SEPARATE WIDGET, setState is needed in main tree
|
||||
backgroundColor:
|
||||
Theme.of(context).colorScheme.surface,
|
||||
title: const Text('Export Settings'),
|
||||
content: SizedBox(
|
||||
width: 450,
|
||||
child: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
const Text('Document Type:'),
|
||||
_SegmentedButton(
|
||||
callback: segmentedCallback,
|
||||
docType: docTypeTmp,
|
||||
),
|
||||
const Text(
|
||||
'Data Columns you would like to show on the datasheet:'),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: _FilterDataTypeChips(
|
||||
docTypeTmp,
|
||||
dataFiltersJobTmp,
|
||||
dataFiltersBusinessTmp,
|
||||
chipsCallback),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Reset'),
|
||||
onPressed: () {
|
||||
_setStateCallbackReset();
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Cancel'),
|
||||
onPressed: () {
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Apply'),
|
||||
onPressed: () {
|
||||
_setStateCallbackApply(
|
||||
docTypeTmp,
|
||||
dataFiltersJobTmp,
|
||||
dataFiltersBusinessTmp);
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
],
|
||||
);
|
||||
});
|
||||
});
|
||||
},
|
||||
),
|
||||
],
|
||||
bottom: PreferredSize(
|
||||
preferredSize: const Size.fromHeight(0),
|
||||
child: SizedBox(
|
||||
height: 70,
|
||||
width: 1000,
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(10),
|
||||
child: BusinessSearchBar(
|
||||
filters: jobTypeFilters,
|
||||
setFiltersCallback: _setFilters,
|
||||
setSearchCallback: _setSearch),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
FutureBuilder(
|
||||
future: refreshBusinessDataFuture,
|
||||
builder: (context, snapshot) {
|
||||
if (snapshot.connectionState == ConnectionState.done) {
|
||||
if (snapshot.hasData) {
|
||||
if (snapshot.data.runtimeType == String) {
|
||||
_isPreviousData = false;
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
child: Column(children: [
|
||||
Center(
|
||||
child: Text(snapshot.data,
|
||||
textAlign: TextAlign.center)),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: FilledButton(
|
||||
child: const Text('Retry'),
|
||||
onPressed: () {
|
||||
_updateOverviewBusinesses();
|
||||
},
|
||||
),
|
||||
),
|
||||
]),
|
||||
));
|
||||
}
|
||||
|
||||
overviewBusinesses = snapshot.data;
|
||||
_isPreviousData = true;
|
||||
|
||||
return BusinessDisplayPanel(
|
||||
groupedBusinesses: _filterBySearch(overviewBusinesses),
|
||||
widescreen: widescreen,
|
||||
selectable: true);
|
||||
} else if (snapshot.hasError) {
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 16.0, right: 16.0),
|
||||
child: Text(
|
||||
'Error when loading data! Error: ${snapshot.error}'),
|
||||
));
|
||||
}
|
||||
} else if (snapshot.connectionState ==
|
||||
ConnectionState.waiting) {
|
||||
if (_isPreviousData) {
|
||||
return BusinessDisplayPanel(
|
||||
groupedBusinesses: _filterBySearch(overviewBusinesses),
|
||||
widescreen: widescreen,
|
||||
selectable: true);
|
||||
} else {
|
||||
return SliverToBoxAdapter(
|
||||
child: Container(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
alignment: Alignment.center,
|
||||
child: const SizedBox(
|
||||
width: 75,
|
||||
height: 75,
|
||||
child: RiveAnimation.asset(
|
||||
'assets/mdev_triangle_loading.riv'),
|
||||
),
|
||||
));
|
||||
}
|
||||
}
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
'\nError: ${snapshot.error}',
|
||||
style: const TextStyle(fontSize: 18),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
),
|
||||
);
|
||||
}),
|
||||
const SliverToBoxAdapter(
|
||||
child: SizedBox(
|
||||
height: 100,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class _SegmentedButton extends StatefulWidget {
|
||||
final void Function(String) callback;
|
||||
final String docType;
|
||||
|
||||
const _SegmentedButton({required this.callback, required this.docType});
|
||||
|
||||
@override
|
||||
State<_SegmentedButton> createState() => _SegmentedButtonState();
|
||||
}
|
||||
|
||||
class _SegmentedButtonState extends State<_SegmentedButton> {
|
||||
Set<String> _selected = {};
|
||||
|
||||
void updateSelected(Set<String> newSelection) {
|
||||
setState(() {
|
||||
_selected = newSelection;
|
||||
});
|
||||
widget.callback(newSelection.first);
|
||||
}
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_selected = {widget.docType};
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: SegmentedButton(
|
||||
segments: const <ButtonSegment<String>>[
|
||||
ButtonSegment<String>(
|
||||
value: 'Business',
|
||||
label: Text('Businesses'),
|
||||
icon: Icon(Icons.business)),
|
||||
ButtonSegment<String>(
|
||||
value: 'Job Listing',
|
||||
label: Text('Job Listings'),
|
||||
icon: Icon(Icons.work))
|
||||
],
|
||||
selected: _selected,
|
||||
onSelectionChanged: updateSelected,
|
||||
style: SegmentedButton.styleFrom(
|
||||
side: BorderSide(color: Theme.of(context).colorScheme.secondary),
|
||||
)),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class _FAB extends StatefulWidget {
|
||||
final String documentType;
|
||||
final Map<JobType, List<Business>> groupedBusinesses;
|
||||
final Set<DataTypeJob> selectedDataTypesJob;
|
||||
final Set<DataTypeBusiness> selectedDataTypesBusiness;
|
||||
|
||||
const _FAB(
|
||||
{required this.groupedBusinesses,
|
||||
required this.documentType,
|
||||
required this.selectedDataTypesJob,
|
||||
required this.selectedDataTypesBusiness});
|
||||
|
||||
@override
|
||||
State<_FAB> createState() => _FABState();
|
||||
}
|
||||
|
||||
class _FABState extends State<_FAB> {
|
||||
List<Business> allBusinesses = [];
|
||||
bool _isLoading = false;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
|
||||
for (JobType type in widget.groupedBusinesses.keys) {
|
||||
allBusinesses.addAll(widget.groupedBusinesses[type]!);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return FloatingActionButton(
|
||||
child: _isLoading
|
||||
? const Padding(
|
||||
padding: EdgeInsets.all(16.0),
|
||||
child: CircularProgressIndicator(
|
||||
color: Colors.white,
|
||||
strokeWidth: 3.0,
|
||||
),
|
||||
)
|
||||
: const Icon(Icons.save_alt),
|
||||
onPressed: () async {
|
||||
setState(() {
|
||||
_isLoading = true;
|
||||
});
|
||||
|
||||
Set<Business> generateBusinesses = <Business>{};
|
||||
if (selectedBusinesses.isEmpty) {
|
||||
generateBusinesses = Set<Business>.from(allBusinesses);
|
||||
} else {
|
||||
generateBusinesses = selectedBusinesses;
|
||||
}
|
||||
|
||||
await _generatePDF(context, widget.documentType, generateBusinesses,
|
||||
widget.selectedDataTypesBusiness, widget.selectedDataTypesJob);
|
||||
|
||||
setState(() {
|
||||
_isLoading = false;
|
||||
});
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
class _FilterDataTypeChips extends StatefulWidget {
|
||||
final String documentType;
|
||||
final Set<DataTypeJob> selectedDataTypesJob;
|
||||
final Set<DataTypeBusiness> selectedDataTypesBusiness;
|
||||
final void Function(
|
||||
{Set<DataTypeJob>? selectedDataTypesJob,
|
||||
Set<DataTypeBusiness>? selectedDataTypesBusiness}) updateCallback;
|
||||
|
||||
const _FilterDataTypeChips(this.documentType, this.selectedDataTypesJob,
|
||||
this.selectedDataTypesBusiness, this.updateCallback);
|
||||
|
||||
@override
|
||||
State<_FilterDataTypeChips> createState() => _FilterDataTypeChipsState();
|
||||
}
|
||||
|
||||
class _FilterDataTypeChipsState extends State<_FilterDataTypeChips> {
|
||||
List<Padding> filterDataTypeChips() {
|
||||
List<Padding> chips = [];
|
||||
|
||||
if (widget.documentType == 'Business') {
|
||||
for (var type in DataTypeBusiness.values) {
|
||||
chips.add(Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 3.0, right: 3.0, bottom: 3.0, top: 3.0),
|
||||
child: FilterChip(
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(20),
|
||||
side: BorderSide(
|
||||
color: Theme.of(context).colorScheme.secondary)),
|
||||
label: Text(dataTypeFriendlyBusiness[type]!),
|
||||
showCheckmark: false,
|
||||
selected: widget.selectedDataTypesBusiness.contains(type),
|
||||
onSelected: (bool selected) {
|
||||
setState(() {
|
||||
if (selected) {
|
||||
widget.selectedDataTypesBusiness.add(type);
|
||||
} else {
|
||||
widget.selectedDataTypesBusiness.remove(type);
|
||||
}
|
||||
});
|
||||
widget.updateCallback(
|
||||
selectedDataTypesBusiness:
|
||||
widget.selectedDataTypesBusiness);
|
||||
}),
|
||||
));
|
||||
}
|
||||
} else if (widget.documentType == 'Job Listing') {
|
||||
for (var type in DataTypeJob.values) {
|
||||
chips.add(Padding(
|
||||
padding: const EdgeInsets.only(
|
||||
left: 3.0, right: 3.0, bottom: 3.0, top: 3.0),
|
||||
child: FilterChip(
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(20),
|
||||
side: BorderSide(
|
||||
color: Theme.of(context).colorScheme.secondary)),
|
||||
label: Text(dataTypeFriendlyJob[type]!),
|
||||
showCheckmark: false,
|
||||
selected: widget.selectedDataTypesJob.contains(type),
|
||||
onSelected: (bool selected) {
|
||||
setState(() {
|
||||
if (selected) {
|
||||
widget.selectedDataTypesJob.add(type);
|
||||
} else {
|
||||
widget.selectedDataTypesJob.remove(type);
|
||||
}
|
||||
});
|
||||
widget.updateCallback(
|
||||
selectedDataTypesJob: widget.selectedDataTypesJob);
|
||||
}),
|
||||
));
|
||||
}
|
||||
}
|
||||
return chips;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Wrap(
|
||||
children: filterDataTypeChips(),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> _generatePDF(
|
||||
BuildContext context,
|
||||
String documentType,
|
||||
Set<Business>? selectedBusinesses,
|
||||
Set<DataTypeBusiness>? dataTypesBusinessInput,
|
||||
Set<DataTypeJob>? dataTypesJobInput) async {
|
||||
Set<DataTypeBusiness> dataTypesBusiness = {};
|
||||
Set<DataTypeJob> dataTypesJob = {};
|
||||
List<pw.Widget> headerColumns = [];
|
||||
List<pw.TableRow> tableRows = [];
|
||||
List<Business> businesses = await fetchBusinesses(
|
||||
selectedBusinesses!.map((business) => business.id).toList());
|
||||
|
||||
if (documentType == 'Business') {
|
||||
dataTypesBusiness = Set.from(dataTypesBusinessInput!);
|
||||
if (dataTypesBusiness.isEmpty) {
|
||||
dataTypesBusiness.addAll(DataTypeBusiness.values);
|
||||
}
|
||||
dataTypesBusiness = sortDataTypesBusiness(dataTypesBusiness);
|
||||
|
||||
for (Business business in businesses) {
|
||||
List<pw.Widget> businessRow = [];
|
||||
if (dataTypesBusiness.contains(DataTypeBusiness.logo)) {
|
||||
var apiLogo = await getLogo(business.id);
|
||||
if (apiLogo.runtimeType != String) {
|
||||
businessRow.add(pw.Padding(
|
||||
child: pw.ClipRRect(
|
||||
child:
|
||||
pw.Image(pw.MemoryImage(apiLogo), height: 24, width: 24),
|
||||
horizontalRadius: 4,
|
||||
verticalRadius: 4),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
} else {
|
||||
businessRow.add(pw.Padding(
|
||||
child: pw.Icon(const pw.IconData(0xe0af), size: 24),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
}
|
||||
}
|
||||
for (DataTypeBusiness dataType in dataTypesBusiness) {
|
||||
if (dataType != DataTypeBusiness.logo) {
|
||||
businessRow.add(pw.Padding(
|
||||
child: pw.Text(businessValueFromDataType(business, dataType)),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
}
|
||||
}
|
||||
tableRows.add(pw.TableRow(children: businessRow));
|
||||
}
|
||||
|
||||
for (var filter in dataTypesBusiness) {
|
||||
headerColumns.add(pw.Padding(
|
||||
child: pw.Text(dataTypeFriendlyBusiness[filter]!,
|
||||
style: const pw.TextStyle(fontSize: 10)),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
}
|
||||
} else if (documentType == 'Job Listing') {
|
||||
dataTypesJob = Set.from(dataTypesJobInput!);
|
||||
if (dataTypesJob.isEmpty) {
|
||||
dataTypesJob.addAll(DataTypeJob.values);
|
||||
}
|
||||
List<DataTypeJob> dataTypesJobList =
|
||||
sortDataTypesJob(dataTypesJob).toList();
|
||||
|
||||
List<Map<String, dynamic>> nameMapping = await fetchBusinessNames();
|
||||
|
||||
for (Business business in businesses) {
|
||||
for (JobListing job in business.listings!) {
|
||||
List<pw.Widget> jobRow = [];
|
||||
for (DataTypeJob dataType in dataTypesJobList) {
|
||||
jobRow.add(pw.Padding(
|
||||
child: pw.Text(jobValueFromDataType(job, dataType, nameMapping)),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
}
|
||||
tableRows.add(pw.TableRow(children: jobRow));
|
||||
}
|
||||
}
|
||||
|
||||
for (var filter in dataTypesJobList) {
|
||||
headerColumns.add(pw.Padding(
|
||||
child: pw.Text(dataTypeFriendlyJob[filter]!,
|
||||
style: const pw.TextStyle(fontSize: 10)),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
}
|
||||
} else {
|
||||
ScaffoldMessenger.of(context).showSnackBar(const SnackBar(
|
||||
content: Text(
|
||||
'Could not identify document type! Please select a type in the generation settings.')));
|
||||
return;
|
||||
}
|
||||
|
||||
// Final Generation
|
||||
DateTime dateTime = DateTime.now();
|
||||
String minute = '00';
|
||||
if (dateTime.minute.toString().length < 2) {
|
||||
minute = '0${dateTime.minute}';
|
||||
} else {
|
||||
minute = dateTime.minute.toString();
|
||||
}
|
||||
|
||||
String time = dateTime.hour <= 12
|
||||
? '${dateTime.hour}:${minute}AM'
|
||||
: '${dateTime.hour - 12}:${minute}PM';
|
||||
String fileName =
|
||||
'$documentType Data - ${dateTime.month}-${dateTime.day}-${dateTime.year} $time.pdf';
|
||||
|
||||
final pdf = pw.Document();
|
||||
var svgBytes = await marinoDevLogo();
|
||||
|
||||
var themeIcon = pw.ThemeData.withFont(
|
||||
base: await PdfGoogleFonts.notoSansDisplayMedium(),
|
||||
icons: await PdfGoogleFonts.materialIcons());
|
||||
|
||||
var finalTheme = themeIcon.copyWith(
|
||||
defaultTextStyle: const pw.TextStyle(fontSize: 9),
|
||||
);
|
||||
|
||||
pdf.addPage(pw.MultiPage(
|
||||
theme: finalTheme,
|
||||
pageFormat: PdfPageFormat.letter,
|
||||
orientation: pw.PageOrientation.landscape,
|
||||
margin: const pw.EdgeInsets.all(24),
|
||||
build: (pw.Context context) {
|
||||
return [
|
||||
pw.Row(
|
||||
mainAxisAlignment: pw.MainAxisAlignment.spaceBetween,
|
||||
children: [
|
||||
pw.SvgImage(svg: utf8.decode(svgBytes), height: 40),
|
||||
pw.Padding(
|
||||
padding: const pw.EdgeInsets.all(8.0),
|
||||
child: pw.Text('$documentType Datasheet',
|
||||
style: pw.TextStyle(
|
||||
fontSize: 32, fontWeight: pw.FontWeight.bold)),
|
||||
),
|
||||
pw.Text(
|
||||
'Generated on ${dateTime.month}/${dateTime.day}/${dateTime.year} at $time',
|
||||
style: const pw.TextStyle(fontSize: 12),
|
||||
textAlign: pw.TextAlign.right),
|
||||
//
|
||||
]),
|
||||
pw.Table(
|
||||
columnWidths: documentType == 'Business'
|
||||
? _businessColumnSizes(dataTypesBusiness)
|
||||
: _jobColumnSizes(dataTypesJob),
|
||||
border: const pw.TableBorder(
|
||||
bottom: pw.BorderSide(),
|
||||
left: pw.BorderSide(),
|
||||
right: pw.BorderSide(),
|
||||
top: pw.BorderSide(),
|
||||
horizontalInside: pw.BorderSide(),
|
||||
verticalInside: pw.BorderSide()),
|
||||
children: [
|
||||
pw.TableRow(
|
||||
decoration: const pw.BoxDecoration(color: PdfColors.blue400),
|
||||
children: headerColumns,
|
||||
repeat: true,
|
||||
),
|
||||
...tableRows,
|
||||
])
|
||||
];
|
||||
}));
|
||||
|
||||
Uint8List pdfBytes = await pdf.save();
|
||||
|
||||
if (kIsWeb) {
|
||||
// List<int> fileInts = List.from(pdfBytes);
|
||||
// html.AnchorElement(
|
||||
// href:
|
||||
// 'data:application/octet-stream;charset=utf-16le;base64,${base64.encode(fileInts)}')
|
||||
// ..setAttribute('download', fileName)
|
||||
// ..click();
|
||||
|
||||
await Printing.sharePdf(
|
||||
bytes: await pdf.save(),
|
||||
filename: fileName,
|
||||
);
|
||||
} else {
|
||||
var dir = await getTemporaryDirectory();
|
||||
var tempDir = dir.path;
|
||||
|
||||
File pdfFile = File('$tempDir/$fileName');
|
||||
pdfFile.writeAsBytesSync(pdfBytes);
|
||||
|
||||
OpenFilex.open(pdfFile.path);
|
||||
}
|
||||
}
|
||||
|
||||
Map<int, pw.TableColumnWidth> _businessColumnSizes(
|
||||
Set<DataTypeBusiness> dataTypes) {
|
||||
double space = 744.0;
|
||||
Map<int, pw.TableColumnWidth> map = {};
|
||||
|
||||
if (dataTypes.contains(DataTypeBusiness.logo)) {
|
||||
space -= 28;
|
||||
map.addAll({
|
||||
dataTypePriorityBusiness[DataTypeBusiness.logo]!:
|
||||
const pw.FixedColumnWidth(28)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.contactName)) {
|
||||
space -= 72;
|
||||
map.addAll({
|
||||
dataTypePriorityBusiness[DataTypeBusiness.contactName]!:
|
||||
const pw.FixedColumnWidth(72)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.contactPhone)) {
|
||||
space -= 76;
|
||||
map.addAll({
|
||||
dataTypePriorityBusiness[DataTypeBusiness.contactPhone]!:
|
||||
const pw.FixedColumnWidth(76)
|
||||
});
|
||||
}
|
||||
double leftNum = 0;
|
||||
if (dataTypes.contains(DataTypeBusiness.name)) {
|
||||
leftNum += 1;
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.website)) {
|
||||
leftNum += 1;
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.contactEmail)) {
|
||||
leftNum += 1;
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.notes)) {
|
||||
leftNum += 2;
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.description)) {
|
||||
leftNum += 3;
|
||||
}
|
||||
leftNum = space / leftNum;
|
||||
if (dataTypes.contains(DataTypeBusiness.name)) {
|
||||
map.addAll({
|
||||
dataTypePriorityBusiness[DataTypeBusiness.name]!:
|
||||
pw.FixedColumnWidth(leftNum)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.website)) {
|
||||
map.addAll({
|
||||
dataTypePriorityBusiness[DataTypeBusiness.website]!:
|
||||
pw.FixedColumnWidth(leftNum)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.contactEmail)) {
|
||||
map.addAll({
|
||||
dataTypePriorityBusiness[DataTypeBusiness.contactEmail]!:
|
||||
pw.FixedColumnWidth(leftNum)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.notes)) {
|
||||
map.addAll({
|
||||
dataTypePriorityBusiness[DataTypeBusiness.notes]!:
|
||||
pw.FixedColumnWidth(leftNum * 2)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.description)) {
|
||||
map.addAll({
|
||||
dataTypePriorityBusiness[DataTypeBusiness.description]!:
|
||||
pw.FixedColumnWidth(leftNum * 3)
|
||||
});
|
||||
}
|
||||
|
||||
return map;
|
||||
}
|
||||
|
||||
Map<int, pw.TableColumnWidth> _jobColumnSizes(Set<DataTypeJob> dataTypes) {
|
||||
Map<int, pw.TableColumnWidth> map = {};
|
||||
List<DataTypeJob> sortedDataTypes = sortDataTypesJob(dataTypes).toList();
|
||||
|
||||
if (dataTypes.contains(DataTypeJob.businessName)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.businessName)
|
||||
.first): const pw.FractionColumnWidth(0.2)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeJob.name)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.name)
|
||||
.first): const pw.FractionColumnWidth(0.2)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeJob.description)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.description)
|
||||
.first): const pw.FractionColumnWidth(0.4)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeJob.wage)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.wage)
|
||||
.first): const pw.FractionColumnWidth(0.15)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeJob.link)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.link)
|
||||
.first): const pw.FractionColumnWidth(0.2)
|
||||
});
|
||||
}
|
||||
|
||||
return map;
|
||||
}
|
||||
|
||||
dynamic businessValueFromDataType(
|
||||
Business business, DataTypeBusiness dataType) {
|
||||
switch (dataType) {
|
||||
case DataTypeBusiness.name:
|
||||
return business.name;
|
||||
case DataTypeBusiness.description:
|
||||
return business.description;
|
||||
case DataTypeBusiness.website:
|
||||
return business.website;
|
||||
case DataTypeBusiness.contactName:
|
||||
return business.contactName;
|
||||
case DataTypeBusiness.contactEmail:
|
||||
return business.contactEmail;
|
||||
case DataTypeBusiness.contactPhone:
|
||||
return business.contactPhone;
|
||||
case DataTypeBusiness.notes:
|
||||
return business.notes;
|
||||
case DataTypeBusiness.logo:
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
dynamic jobValueFromDataType(JobListing job, DataTypeJob dataType,
|
||||
List<Map<String, dynamic>> nameMapping) {
|
||||
switch (dataType) {
|
||||
case DataTypeJob.name:
|
||||
return job.name;
|
||||
case DataTypeJob.description:
|
||||
return job.description;
|
||||
case DataTypeJob.wage:
|
||||
return job.wage;
|
||||
case DataTypeJob.link:
|
||||
return job.link;
|
||||
case DataTypeJob.businessName:
|
||||
return nameMapping
|
||||
.where((element) => element['id'] == job.businessId)
|
||||
.first['name'];
|
||||
}
|
||||
}
|
||||
@ -1,8 +1,8 @@
|
||||
import 'package:fbla_ui/api_logic.dart';
|
||||
import 'package:fbla_ui/main.dart';
|
||||
import 'package:fbla_ui/pages/create_edit_listing.dart';
|
||||
import 'package:fbla_ui/pages/signin_page.dart';
|
||||
import 'package:fbla_ui/shared.dart';
|
||||
import 'package:fbla_ui/shared/api_logic.dart';
|
||||
import 'package:fbla_ui/shared/global_vars.dart';
|
||||
import 'package:fbla_ui/shared/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:url_launcher/url_launcher.dart';
|
||||
|
||||
@ -39,30 +39,43 @@ class _CreateBusinessDetailState extends State<JobListingDetail> {
|
||||
clipBehavior: Clip.antiAlias,
|
||||
child: Column(
|
||||
children: [
|
||||
ListTile(
|
||||
title: Text(listing.name,
|
||||
textAlign: TextAlign.left,
|
||||
style: const TextStyle(
|
||||
fontSize: 24, fontWeight: FontWeight.bold)),
|
||||
subtitle: Text(
|
||||
listing.description,
|
||||
textAlign: TextAlign.left,
|
||||
),
|
||||
leading: ClipRRect(
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
child: Row(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(right: 16.0),
|
||||
child: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(6.0),
|
||||
child: Image.network(
|
||||
'$apiAddress/logos/${listing.businessId}',
|
||||
width: 48,
|
||||
height: 48, errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return getIconFromJobType(listing.type, 48,
|
||||
Theme.of(context).colorScheme.onSurface);
|
||||
return Icon(getIconFromJobType(listing.type!),
|
||||
size: 48);
|
||||
}),
|
||||
),
|
||||
),
|
||||
Visibility(
|
||||
visible: listing.link != null && listing.link != '',
|
||||
child: ListTile(
|
||||
Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(listing.name,
|
||||
style: const TextStyle(
|
||||
fontSize: 24, fontWeight: FontWeight.bold)),
|
||||
Text(widget.fromBusiness.name!,
|
||||
style: const TextStyle(fontSize: 16)),
|
||||
Text(
|
||||
listing.description,
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
if (listing.link != null && listing.link != '')
|
||||
ListTile(
|
||||
leading: const Icon(Icons.link),
|
||||
title: const Text('More Information'),
|
||||
subtitle: Text(
|
||||
@ -75,7 +88,6 @@ class _CreateBusinessDetailState extends State<JobListingDetail> {
|
||||
launchUrl(Uri.parse(listing.link!));
|
||||
},
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
@ -108,9 +120,8 @@ class _CreateBusinessDetailState extends State<JobListingDetail> {
|
||||
),
|
||||
],
|
||||
),
|
||||
Visibility(
|
||||
visible: widget.fromBusiness.contactPhone != null,
|
||||
child: ListTile(
|
||||
if (widget.fromBusiness.contactPhone != null)
|
||||
ListTile(
|
||||
leading: const Icon(Icons.phone),
|
||||
title: Text(widget.fromBusiness.contactPhone!),
|
||||
// maybe replace ! with ?? ''. same is true for below
|
||||
@ -145,13 +156,13 @@ class _CreateBusinessDetailState extends State<JobListingDetail> {
|
||||
});
|
||||
},
|
||||
),
|
||||
),
|
||||
if (widget.fromBusiness.contactEmail != null)
|
||||
ListTile(
|
||||
leading: const Icon(Icons.email),
|
||||
title: Text(widget.fromBusiness.contactEmail),
|
||||
title: Text(widget.fromBusiness.contactEmail!),
|
||||
onTap: () {
|
||||
launchUrl(
|
||||
Uri.parse('mailto:${widget.fromBusiness.contactEmail}'));
|
||||
launchUrl(Uri.parse(
|
||||
'mailto:${widget.fromBusiness.contactEmail}'));
|
||||
},
|
||||
),
|
||||
],
|
||||
|
||||
582
fbla_ui/lib/pages/listings_overview.dart
Normal file
582
fbla_ui/lib/pages/listings_overview.dart
Normal file
@ -0,0 +1,582 @@
|
||||
import 'package:fbla_ui/pages/create_edit_listing.dart';
|
||||
import 'package:fbla_ui/pages/listing_detail.dart';
|
||||
import 'package:fbla_ui/shared/api_logic.dart';
|
||||
import 'package:fbla_ui/shared/export.dart';
|
||||
import 'package:fbla_ui/shared/global_vars.dart';
|
||||
import 'package:fbla_ui/shared/utils.dart';
|
||||
import 'package:fbla_ui/shared/widgets.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/rendering.dart';
|
||||
import 'package:flutter_sticky_header/flutter_sticky_header.dart';
|
||||
import 'package:rive/rive.dart';
|
||||
import 'package:sliver_tools/sliver_tools.dart';
|
||||
import 'package:url_launcher/url_launcher.dart';
|
||||
|
||||
class JobsOverview extends StatefulWidget {
|
||||
final String searchQuery;
|
||||
final Future refreshJobDataOverviewFuture;
|
||||
final Future<void> Function(Set<JobType>) updateBusinessesCallback;
|
||||
final void Function() themeCallback;
|
||||
final void Function(bool) updateLoggedIn;
|
||||
|
||||
const JobsOverview({
|
||||
super.key,
|
||||
required this.searchQuery,
|
||||
required this.refreshJobDataOverviewFuture,
|
||||
required this.updateBusinessesCallback,
|
||||
required this.themeCallback,
|
||||
required this.updateLoggedIn,
|
||||
});
|
||||
|
||||
@override
|
||||
State<JobsOverview> createState() => _JobsOverviewState();
|
||||
}
|
||||
|
||||
class _JobsOverviewState extends State<JobsOverview> {
|
||||
bool _isPreviousData = false;
|
||||
late Map<JobType, List<Business>> overviewBusinesses;
|
||||
Set<JobType> jobTypeFilters = <JobType>{};
|
||||
String searchQuery = '';
|
||||
ScrollController controller = ScrollController();
|
||||
bool _extended = true;
|
||||
double prevPixelPosition = 0;
|
||||
|
||||
Map<JobType, List<Business>> _filterBySearch(
|
||||
Map<JobType, List<Business>> businesses, String query) {
|
||||
Map<JobType, List<Business>> filteredBusinesses = {};
|
||||
|
||||
for (JobType jobType in businesses.keys) {
|
||||
filteredBusinesses[jobType] = List.from(businesses[jobType]!.where(
|
||||
(element) => element.listings![0].name
|
||||
.replaceAll(RegExp(r'[^a-zA-Z]'), '')
|
||||
.toLowerCase()
|
||||
.contains(query
|
||||
.replaceAll(RegExp(r'[^a-zA-Z]'), '')
|
||||
.toLowerCase()
|
||||
.trim())));
|
||||
}
|
||||
|
||||
filteredBusinesses.removeWhere((key, value) => value.isEmpty);
|
||||
return filteredBusinesses;
|
||||
}
|
||||
|
||||
void _setSearch(String search) async {
|
||||
setState(() {
|
||||
searchQuery = search;
|
||||
});
|
||||
}
|
||||
|
||||
void _setFilters(Set<JobType> filters) async {
|
||||
jobTypeFilters = Set.from(filters);
|
||||
widget.updateBusinessesCallback(jobTypeFilters);
|
||||
}
|
||||
|
||||
void _scrollListener() {
|
||||
if ((prevPixelPosition - controller.position.pixels).abs() > 10) {
|
||||
setState(() {
|
||||
_extended =
|
||||
controller.position.userScrollDirection == ScrollDirection.forward;
|
||||
});
|
||||
}
|
||||
prevPixelPosition = controller.position.pixels;
|
||||
}
|
||||
|
||||
void _generatePDF() {
|
||||
List<Business> allJobs = [];
|
||||
for (List<Business> businesses
|
||||
in _filterBySearch(overviewBusinesses, searchQuery).values) {
|
||||
allJobs.addAll(businesses);
|
||||
}
|
||||
|
||||
generatePDF(
|
||||
context: context,
|
||||
documentTypeIndex: 1,
|
||||
selectedJobs: Set.from(allJobs));
|
||||
}
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
|
||||
controller.addListener(_scrollListener);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
bool widescreen = MediaQuery.sizeOf(context).width >= widescreenWidth;
|
||||
return Scaffold(
|
||||
floatingActionButton: _getFAB(widescreen),
|
||||
body: CustomScrollView(
|
||||
controller: controller,
|
||||
slivers: [
|
||||
MainSliverAppBar(
|
||||
widescreen: widescreen,
|
||||
setSearch: _setSearch,
|
||||
searchHintText: 'Search Job Listings',
|
||||
themeCallback: widget.themeCallback,
|
||||
filterIconButton: _filterIconButton(
|
||||
jobTypeFilters,
|
||||
),
|
||||
updateLoggedIn: widget.updateLoggedIn,
|
||||
generatePDF: _generatePDF,
|
||||
),
|
||||
FutureBuilder(
|
||||
future: widget.refreshJobDataOverviewFuture,
|
||||
builder: (context, snapshot) {
|
||||
if (snapshot.connectionState == ConnectionState.done) {
|
||||
if (snapshot.hasData) {
|
||||
if (snapshot.data.runtimeType == String) {
|
||||
_isPreviousData = false;
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
child: Column(children: [
|
||||
Center(
|
||||
child: Text(snapshot.data,
|
||||
textAlign: TextAlign.center)),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: FilledButton(
|
||||
child: const Text('Retry'),
|
||||
onPressed: () {
|
||||
widget.updateBusinessesCallback(jobTypeFilters);
|
||||
},
|
||||
),
|
||||
),
|
||||
]),
|
||||
));
|
||||
}
|
||||
|
||||
overviewBusinesses = snapshot.data;
|
||||
_isPreviousData = true;
|
||||
|
||||
return JobDisplayPanel(
|
||||
jobGroupedBusinesses:
|
||||
_filterBySearch(overviewBusinesses, searchQuery),
|
||||
widescreen: widescreen,
|
||||
);
|
||||
} else if (snapshot.hasError) {
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 16.0, right: 16.0),
|
||||
child: Text(
|
||||
'Error when loading data! Error: ${snapshot.error}'),
|
||||
));
|
||||
}
|
||||
} else if (snapshot.connectionState ==
|
||||
ConnectionState.waiting) {
|
||||
if (_isPreviousData) {
|
||||
return JobDisplayPanel(
|
||||
jobGroupedBusinesses:
|
||||
_filterBySearch(overviewBusinesses, searchQuery),
|
||||
widescreen: widescreen,
|
||||
);
|
||||
} else {
|
||||
return SliverToBoxAdapter(
|
||||
child: Container(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
alignment: Alignment.center,
|
||||
child: const SizedBox(
|
||||
width: 75,
|
||||
height: 75,
|
||||
child: RiveAnimation.asset(
|
||||
'assets/mdev_triangle_loading.riv'),
|
||||
),
|
||||
));
|
||||
}
|
||||
}
|
||||
return SliverToBoxAdapter(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
'\nError: ${snapshot.error}',
|
||||
style: const TextStyle(fontSize: 18),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
),
|
||||
);
|
||||
}),
|
||||
],
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _filterIconButton(Set<JobType> filters) {
|
||||
Set<JobType> selectedChips = Set.from(filters);
|
||||
|
||||
return IconButton(
|
||||
icon: Icon(
|
||||
Icons.filter_list,
|
||||
color: filters.isNotEmpty
|
||||
? Theme.of(context).colorScheme.primary
|
||||
: Theme.of(context).colorScheme.onSurface,
|
||||
),
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return StatefulBuilder(
|
||||
builder: (BuildContext context, StateSetter setState) {
|
||||
void setDialogState(Set<JobType> newFilters) {
|
||||
setState(() {
|
||||
filters = newFilters;
|
||||
});
|
||||
}
|
||||
|
||||
List<Padding> chips = [];
|
||||
for (var type in JobType.values) {
|
||||
chips.add(Padding(
|
||||
padding: const EdgeInsets.all(4),
|
||||
child: FilterChip(
|
||||
showCheckmark: false,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(20)),
|
||||
label: Text(getNameFromJobType(type)),
|
||||
selected: selectedChips.contains(type),
|
||||
onSelected: (bool selected) {
|
||||
if (selected) {
|
||||
selectedChips.add(type);
|
||||
} else {
|
||||
selectedChips.remove(type);
|
||||
}
|
||||
setDialogState(filters);
|
||||
}),
|
||||
));
|
||||
}
|
||||
|
||||
return AlertDialog(
|
||||
title: const Text('Filter Options'),
|
||||
content: SizedBox(
|
||||
width: 400,
|
||||
child: Wrap(
|
||||
children: chips,
|
||||
),
|
||||
),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Reset'),
|
||||
onPressed: () {
|
||||
_setFilters(<JobType>{});
|
||||
// selectedChips = <BusinessType>{};
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
),
|
||||
TextButton(
|
||||
child: const Text('Cancel'),
|
||||
onPressed: () {
|
||||
// selectedChips = Set.from(filters);
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
),
|
||||
TextButton(
|
||||
child: const Text('Apply'),
|
||||
onPressed: () {
|
||||
_setFilters(selectedChips);
|
||||
Navigator.of(context).pop();
|
||||
},
|
||||
)
|
||||
],
|
||||
);
|
||||
});
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
Widget? _getFAB(bool widescreen) {
|
||||
if (!widescreen && loggedIn) {
|
||||
return FloatingActionButton.extended(
|
||||
extendedIconLabelSpacing: _extended ? 8.0 : 0,
|
||||
extendedPadding: const EdgeInsets.symmetric(horizontal: 16),
|
||||
icon: const Icon(Icons.add),
|
||||
label: AnimatedSize(
|
||||
curve: Easing.standard,
|
||||
duration: const Duration(milliseconds: 300),
|
||||
child: _extended ? const Text('Add Job Listing') : Container(),
|
||||
),
|
||||
onPressed: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => const CreateEditJobListing()));
|
||||
},
|
||||
);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
class JobDisplayPanel extends StatefulWidget {
|
||||
final Map<JobType, List<Business>> jobGroupedBusinesses;
|
||||
final bool widescreen;
|
||||
|
||||
const JobDisplayPanel({
|
||||
super.key,
|
||||
required this.jobGroupedBusinesses,
|
||||
required this.widescreen,
|
||||
});
|
||||
|
||||
@override
|
||||
State<JobDisplayPanel> createState() => _JobDisplayPanelState();
|
||||
}
|
||||
|
||||
class _JobDisplayPanelState extends State<JobDisplayPanel> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
if (widget.jobGroupedBusinesses.keys.isEmpty) {
|
||||
return const SliverToBoxAdapter(
|
||||
child: Center(
|
||||
child: Padding(
|
||||
padding: EdgeInsets.all(16.0),
|
||||
child: Text(
|
||||
'No results found!\nPlease change your search filters.',
|
||||
textAlign: TextAlign.center,
|
||||
style: TextStyle(fontSize: 18),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
List<BusinessHeader> headers = [];
|
||||
for (JobType jobType in widget.jobGroupedBusinesses.keys) {
|
||||
headers.add(BusinessHeader(
|
||||
jobType: jobType,
|
||||
widescreen: widget.widescreen,
|
||||
businesses: widget.jobGroupedBusinesses[jobType]!));
|
||||
}
|
||||
headers.sort((a, b) => a.jobType.index.compareTo(b.jobType.index));
|
||||
return MultiSliver(children: headers);
|
||||
}
|
||||
}
|
||||
|
||||
class BusinessHeader extends StatefulWidget {
|
||||
final JobType jobType;
|
||||
final List<Business> businesses;
|
||||
final bool widescreen;
|
||||
|
||||
const BusinessHeader({
|
||||
super.key,
|
||||
required this.jobType,
|
||||
required this.businesses,
|
||||
required this.widescreen,
|
||||
});
|
||||
|
||||
@override
|
||||
State<BusinessHeader> createState() => _BusinessHeaderState();
|
||||
}
|
||||
|
||||
class _BusinessHeaderState extends State<BusinessHeader> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return SliverStickyHeader(
|
||||
header: Container(
|
||||
height: 55.0,
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
padding: const EdgeInsets.symmetric(horizontal: 16.0),
|
||||
alignment: Alignment.centerLeft,
|
||||
child: _getHeaderRow(),
|
||||
),
|
||||
sliver: _getChildSliver(widget.businesses, widget.widescreen),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _getHeaderRow() {
|
||||
return Row(
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(left: 4.0, right: 12.0),
|
||||
child: Icon(
|
||||
getIconFromJobType(widget.jobType),
|
||||
color: Theme.of(context).colorScheme.onPrimary,
|
||||
)),
|
||||
Text(getNameFromJobType(widget.jobType)),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
Widget _getChildSliver(List<Business> businesses, bool widescreen) {
|
||||
if (widescreen) {
|
||||
return SliverPadding(
|
||||
padding: const EdgeInsets.all(4),
|
||||
sliver: SliverGrid(
|
||||
gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent(
|
||||
mainAxisExtent: 250.0,
|
||||
maxCrossAxisExtent: 400.0,
|
||||
mainAxisSpacing: 4.0,
|
||||
crossAxisSpacing: 4.0,
|
||||
),
|
||||
delegate: SliverChildBuilderDelegate(
|
||||
childCount: businesses.length,
|
||||
(BuildContext context, int index) {
|
||||
return _businessTile(
|
||||
businesses[index],
|
||||
widget.jobType,
|
||||
);
|
||||
},
|
||||
),
|
||||
),
|
||||
);
|
||||
} else {
|
||||
return SliverList(
|
||||
delegate: SliverChildBuilderDelegate(
|
||||
childCount: businesses.length,
|
||||
(BuildContext context, int index) {
|
||||
return _businessListItem(
|
||||
businesses[index],
|
||||
widget.jobType,
|
||||
);
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Widget _businessTile(Business business, JobType jobType) {
|
||||
return MouseRegion(
|
||||
cursor: SystemMouseCursors.click,
|
||||
child: GestureDetector(
|
||||
onTap: () {
|
||||
Navigator.of(context).push(MaterialPageRoute(
|
||||
builder: (context) => JobListingDetail(
|
||||
listing: business.listings![0],
|
||||
fromBusiness: business,
|
||||
)));
|
||||
},
|
||||
child: Card(
|
||||
clipBehavior: Clip.antiAlias,
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.center,
|
||||
children: [
|
||||
Row(
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(6.0),
|
||||
child: Image.network('$apiAddress/logos/${business.id}',
|
||||
height: 48,
|
||||
width: 48, errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return Icon(getIconFromBusinessType(business.type!),
|
||||
size: 48);
|
||||
}),
|
||||
)),
|
||||
Flexible(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
business.listings![0].name,
|
||||
style: const TextStyle(
|
||||
fontSize: 18, fontWeight: FontWeight.bold),
|
||||
maxLines: 2,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
business.listings![0].description,
|
||||
maxLines: 5,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
),
|
||||
const Spacer(),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||
children: [
|
||||
if (business.listings![0].link != null &&
|
||||
business.listings![0].link!.isNotEmpty)
|
||||
IconButton(
|
||||
icon: const Icon(Icons.link),
|
||||
onPressed: () {
|
||||
launchUrl(Uri.parse(
|
||||
'https://${business.listings![0].link!}'));
|
||||
},
|
||||
),
|
||||
IconButton(
|
||||
icon: const Icon(Icons.location_on),
|
||||
onPressed: () {
|
||||
launchUrl(Uri.parse(Uri.encodeFull(
|
||||
'https://www.google.com/maps/search/?api=1&query=${business.locationName}')));
|
||||
},
|
||||
),
|
||||
if (business.contactPhone != null)
|
||||
IconButton(
|
||||
icon: const Icon(Icons.phone),
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
backgroundColor:
|
||||
Theme.of(context).colorScheme.surface,
|
||||
title:
|
||||
Text('Contact ${business.contactName}'),
|
||||
content: Text(
|
||||
'Would you like to call or text ${business.contactName}?'),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Text'),
|
||||
onPressed: () {
|
||||
launchUrl(Uri.parse(
|
||||
'sms:${business.contactPhone}'));
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Call'),
|
||||
onPressed: () async {
|
||||
launchUrl(Uri.parse(
|
||||
'tel:${business.contactPhone}'));
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
],
|
||||
);
|
||||
});
|
||||
},
|
||||
),
|
||||
if (business.contactEmail != null)
|
||||
IconButton(
|
||||
icon: const Icon(Icons.email),
|
||||
onPressed: () {
|
||||
launchUrl(
|
||||
Uri.parse('mailto:${business.contactEmail}'));
|
||||
},
|
||||
),
|
||||
],
|
||||
)),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _businessListItem(Business business, JobType? jobType) {
|
||||
return Card(
|
||||
child: ListTile(
|
||||
leading: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(3.0),
|
||||
child: Image.network('$apiAddress/logos/${business.id}',
|
||||
height: 24, width: 24, errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return Icon(getIconFromBusinessType(business.type!));
|
||||
})),
|
||||
title: Text(business.listings![0].name),
|
||||
subtitle: Text(business.listings![0].description,
|
||||
maxLines: 2, overflow: TextOverflow.ellipsis),
|
||||
onTap: () {
|
||||
Navigator.of(context).push(MaterialPageRoute(
|
||||
builder: (context) => JobListingDetail(
|
||||
listing: business.listings![0],
|
||||
fromBusiness: business,
|
||||
)));
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
@ -1,12 +1,10 @@
|
||||
import 'package:fbla_ui/api_logic.dart';
|
||||
import 'package:fbla_ui/shared.dart';
|
||||
import 'package:fbla_ui/shared/api_logic.dart';
|
||||
import 'package:fbla_ui/shared/global_vars.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
bool loggedIn = false;
|
||||
|
||||
class SignInPage extends StatefulWidget {
|
||||
final void Function() refreshAccount;
|
||||
final void Function(bool) refreshAccount;
|
||||
|
||||
const SignInPage({super.key, required this.refreshAccount});
|
||||
|
||||
@ -96,8 +94,7 @@ class _SignInPageState extends State<SignInPage> {
|
||||
await prefs.setString('username', username);
|
||||
await prefs.setString('password', password);
|
||||
await prefs.setBool('rememberMe', rememberMe);
|
||||
loggedIn = true;
|
||||
widget.refreshAccount();
|
||||
widget.refreshAccount(true);
|
||||
Navigator.of(context).pop();
|
||||
} else {
|
||||
setState(() {
|
||||
@ -182,8 +179,7 @@ class _SignInPageState extends State<SignInPage> {
|
||||
await prefs.setString('username', username);
|
||||
await prefs.setString('password', password);
|
||||
await prefs.setBool('rememberMe', rememberMe);
|
||||
loggedIn = true;
|
||||
widget.refreshAccount();
|
||||
widget.refreshAccount(true);
|
||||
Navigator.of(context).pop();
|
||||
} else {
|
||||
setState(() {
|
||||
|
||||
@ -1,951 +0,0 @@
|
||||
import 'package:fbla_ui/api_logic.dart';
|
||||
import 'package:fbla_ui/pages/business_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_sticky_header/flutter_sticky_header.dart';
|
||||
import 'package:pdf/pdf.dart';
|
||||
import 'package:pdf/widgets.dart' as pw;
|
||||
import 'package:sliver_tools/sliver_tools.dart';
|
||||
import 'package:url_launcher/url_launcher.dart';
|
||||
|
||||
late String jwt;
|
||||
String searchFilter = '';
|
||||
Set<Business> selectedBusinesses = <Business>{};
|
||||
|
||||
enum DataTypeBusiness {
|
||||
logo,
|
||||
name,
|
||||
description,
|
||||
website,
|
||||
contactName,
|
||||
contactEmail,
|
||||
contactPhone,
|
||||
notes,
|
||||
}
|
||||
|
||||
enum DataTypeJob {
|
||||
businessName,
|
||||
name,
|
||||
description,
|
||||
wage,
|
||||
link,
|
||||
}
|
||||
|
||||
Map<DataTypeBusiness, int> dataTypePriorityBusiness = {
|
||||
DataTypeBusiness.logo: 0,
|
||||
DataTypeBusiness.name: 1,
|
||||
DataTypeBusiness.description: 2,
|
||||
// DataType.type: 3,
|
||||
DataTypeBusiness.website: 4,
|
||||
DataTypeBusiness.contactName: 5,
|
||||
DataTypeBusiness.contactEmail: 6,
|
||||
DataTypeBusiness.contactPhone: 7,
|
||||
DataTypeBusiness.notes: 8
|
||||
};
|
||||
|
||||
Map<DataTypeBusiness, String> dataTypeFriendlyBusiness = {
|
||||
DataTypeBusiness.logo: 'Logo',
|
||||
DataTypeBusiness.name: 'Name',
|
||||
DataTypeBusiness.description: 'Description',
|
||||
// DataType.type: 'Type',
|
||||
DataTypeBusiness.website: 'Website',
|
||||
DataTypeBusiness.contactName: 'Contact Name',
|
||||
DataTypeBusiness.contactEmail: 'Contact Email',
|
||||
DataTypeBusiness.contactPhone: 'Contact Phone',
|
||||
DataTypeBusiness.notes: 'Notes'
|
||||
};
|
||||
|
||||
Map<DataTypeJob, int> dataTypePriorityJob = {
|
||||
DataTypeJob.businessName: 1,
|
||||
DataTypeJob.name: 2,
|
||||
DataTypeJob.description: 3,
|
||||
DataTypeJob.wage: 4,
|
||||
DataTypeJob.link: 5,
|
||||
};
|
||||
|
||||
Map<DataTypeJob, String> dataTypeFriendlyJob = {
|
||||
DataTypeJob.businessName: 'Business Name',
|
||||
DataTypeJob.name: 'Listing Name',
|
||||
DataTypeJob.description: 'Description',
|
||||
DataTypeJob.wage: 'Wage',
|
||||
DataTypeJob.link: 'Link',
|
||||
};
|
||||
|
||||
Set<DataTypeBusiness> sortDataTypesBusiness(Set<DataTypeBusiness> set) {
|
||||
List<DataTypeBusiness> list = set.toList();
|
||||
list.sort((a, b) {
|
||||
return dataTypePriorityBusiness[a]!.compareTo(dataTypePriorityBusiness[b]!);
|
||||
});
|
||||
set = list.toSet();
|
||||
return set;
|
||||
}
|
||||
|
||||
Set<DataTypeJob> sortDataTypesJob(Set<DataTypeJob> set) {
|
||||
List<DataTypeJob> list = set.toList();
|
||||
list.sort((a, b) {
|
||||
return dataTypePriorityJob[a]!.compareTo(dataTypePriorityJob[b]!);
|
||||
});
|
||||
set = list.toSet();
|
||||
return set;
|
||||
}
|
||||
|
||||
enum BusinessType {
|
||||
food,
|
||||
shop,
|
||||
outdoors,
|
||||
manufacturing,
|
||||
entertainment,
|
||||
other,
|
||||
}
|
||||
|
||||
enum JobType { cashier, server, mechanic, other }
|
||||
|
||||
class JobListing {
|
||||
int? id;
|
||||
int? businessId;
|
||||
String name;
|
||||
String description;
|
||||
JobType type;
|
||||
String? wage;
|
||||
String? link;
|
||||
|
||||
JobListing(
|
||||
{this.id,
|
||||
this.businessId,
|
||||
required this.name,
|
||||
required this.description,
|
||||
required this.type,
|
||||
this.wage,
|
||||
this.link});
|
||||
|
||||
factory JobListing.copy(JobListing input) {
|
||||
return JobListing(
|
||||
id: input.id,
|
||||
businessId: input.businessId,
|
||||
name: input.name,
|
||||
description: input.description,
|
||||
type: input.type,
|
||||
wage: input.wage,
|
||||
link: input.link,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class Business {
|
||||
int id;
|
||||
String name;
|
||||
String description;
|
||||
String website;
|
||||
String? contactName;
|
||||
String contactEmail;
|
||||
String? contactPhone;
|
||||
String? notes;
|
||||
String locationName;
|
||||
String? locationAddress;
|
||||
List<JobListing>? listings;
|
||||
|
||||
Business(
|
||||
{required this.id,
|
||||
required this.name,
|
||||
required this.description,
|
||||
required this.website,
|
||||
this.contactName,
|
||||
required this.contactEmail,
|
||||
this.contactPhone,
|
||||
this.notes,
|
||||
required this.locationName,
|
||||
this.locationAddress,
|
||||
this.listings});
|
||||
|
||||
factory Business.fromJson(Map<String, dynamic> json) {
|
||||
List<JobListing>? listings;
|
||||
if (json['listings'] != null) {
|
||||
listings = [];
|
||||
for (int i = 0; i < json['listings'].length; i++) {
|
||||
listings.add(JobListing(
|
||||
id: json['listings'][i]['id'],
|
||||
businessId: json['listings'][i]['businessId'],
|
||||
name: json['listings'][i]['name'],
|
||||
description: json['listings'][i]['description'],
|
||||
type: JobType.values.byName(json['listings'][i]['type']),
|
||||
wage: json['listings'][i]['wage'],
|
||||
link: json['listings'][i]['link']));
|
||||
}
|
||||
}
|
||||
|
||||
return Business(
|
||||
id: json['id'],
|
||||
name: json['name'],
|
||||
description: json['description'],
|
||||
website: json['website'],
|
||||
contactName: json['contactName'],
|
||||
contactEmail: json['contactEmail'],
|
||||
contactPhone: json['contactPhone'],
|
||||
notes: json['notes'],
|
||||
locationName: json['locationName'],
|
||||
locationAddress: json['locationAddress'],
|
||||
listings: listings);
|
||||
}
|
||||
|
||||
factory Business.copy(Business input) {
|
||||
return Business(
|
||||
id: input.id,
|
||||
name: input.name,
|
||||
description: input.description,
|
||||
website: input.website,
|
||||
contactName: input.contactName,
|
||||
contactEmail: input.contactEmail,
|
||||
contactPhone: input.contactPhone,
|
||||
notes: input.notes,
|
||||
locationName: input.locationName,
|
||||
locationAddress: input.locationAddress,
|
||||
listings: input.listings);
|
||||
}
|
||||
}
|
||||
|
||||
// Map<BusinessType, List<Business>> groupBusinesses(List<Business> businesses) {
|
||||
// Map<BusinessType, List<Business>> groupedBusinesses =
|
||||
// groupBy<Business, BusinessType>(businesses, (business) => business.type!);
|
||||
//
|
||||
// return groupedBusinesses;
|
||||
// }
|
||||
|
||||
Icon getIconFromBusinessType(BusinessType type, double size, Color color) {
|
||||
switch (type) {
|
||||
case BusinessType.food:
|
||||
return Icon(
|
||||
Icons.restaurant,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
case BusinessType.shop:
|
||||
return Icon(
|
||||
Icons.store,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
case BusinessType.outdoors:
|
||||
return Icon(
|
||||
Icons.forest,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
case BusinessType.manufacturing:
|
||||
return Icon(
|
||||
Icons.factory,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
case BusinessType.entertainment:
|
||||
return Icon(
|
||||
Icons.live_tv,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
case BusinessType.other:
|
||||
return Icon(
|
||||
Icons.business,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Icon getIconFromJobType(JobType type, double size, Color color) {
|
||||
switch (type) {
|
||||
case JobType.cashier:
|
||||
return Icon(
|
||||
Icons.shopping_bag,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
case JobType.server:
|
||||
return Icon(
|
||||
Icons.restaurant,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
case JobType.mechanic:
|
||||
return Icon(
|
||||
Icons.construction,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
case JobType.other:
|
||||
return Icon(
|
||||
Icons.work,
|
||||
size: size,
|
||||
color: color,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
pw.Icon getPwIconFromBusinessType(
|
||||
BusinessType type, double size, PdfColor color) {
|
||||
switch (type) {
|
||||
case BusinessType.food:
|
||||
return pw.Icon(const pw.IconData(0xe56c), size: size, color: color);
|
||||
case BusinessType.shop:
|
||||
return pw.Icon(const pw.IconData(0xea12), size: size, color: color);
|
||||
case BusinessType.outdoors:
|
||||
return pw.Icon(const pw.IconData(0xea99), size: size, color: color);
|
||||
case BusinessType.manufacturing:
|
||||
return pw.Icon(const pw.IconData(0xebbc), size: size, color: color);
|
||||
case BusinessType.entertainment:
|
||||
return pw.Icon(const pw.IconData(0xe639), size: size, color: color);
|
||||
case BusinessType.other:
|
||||
return pw.Icon(const pw.IconData(0xe0af), size: size, color: color);
|
||||
}
|
||||
}
|
||||
|
||||
pw.Icon getPwIconFromJobType(JobType type, double size, PdfColor color) {
|
||||
switch (type) {
|
||||
case JobType.cashier:
|
||||
return pw.Icon(const pw.IconData(0xf1cc), size: size, color: color);
|
||||
case JobType.server:
|
||||
return pw.Icon(const pw.IconData(0xe56c), size: size, color: color);
|
||||
case JobType.mechanic:
|
||||
return pw.Icon(const pw.IconData(0xea3c), size: size, color: color);
|
||||
case JobType.other:
|
||||
return pw.Icon(const pw.IconData(0xe8f9), size: size, color: color);
|
||||
}
|
||||
}
|
||||
|
||||
String getNameFromBusinessType(BusinessType type) {
|
||||
switch (type) {
|
||||
case BusinessType.food:
|
||||
return 'Food Related';
|
||||
case BusinessType.shop:
|
||||
return 'Shops';
|
||||
case BusinessType.outdoors:
|
||||
return 'Outdoors';
|
||||
case BusinessType.manufacturing:
|
||||
return 'Manufacturing';
|
||||
case BusinessType.entertainment:
|
||||
return 'Entertainment';
|
||||
case BusinessType.other:
|
||||
return 'Other';
|
||||
}
|
||||
}
|
||||
|
||||
String getNameFromJobType(JobType type) {
|
||||
switch (type) {
|
||||
case JobType.cashier:
|
||||
return 'Cashier';
|
||||
case JobType.server:
|
||||
return 'Server';
|
||||
case JobType.mechanic:
|
||||
return 'Mechanic';
|
||||
case JobType.other:
|
||||
return 'Other';
|
||||
}
|
||||
}
|
||||
|
||||
Icon getIconFromThemeMode(ThemeMode theme) {
|
||||
switch (theme) {
|
||||
case ThemeMode.dark:
|
||||
return const Icon(Icons.dark_mode);
|
||||
case ThemeMode.light:
|
||||
return const Icon(Icons.light_mode);
|
||||
case ThemeMode.system:
|
||||
return const Icon(Icons.brightness_4);
|
||||
}
|
||||
}
|
||||
|
||||
class BusinessDisplayPanel extends StatefulWidget {
|
||||
final Map<JobType, List<Business>> groupedBusinesses;
|
||||
final bool widescreen;
|
||||
final bool selectable;
|
||||
|
||||
const BusinessDisplayPanel(
|
||||
{super.key,
|
||||
required this.groupedBusinesses,
|
||||
required this.widescreen,
|
||||
required this.selectable});
|
||||
|
||||
@override
|
||||
State<BusinessDisplayPanel> createState() => _BusinessDisplayPanelState();
|
||||
}
|
||||
|
||||
class _BusinessDisplayPanelState extends State<BusinessDisplayPanel> {
|
||||
Set<Business> selectedBusinesses = <Business>{};
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
List<BusinessHeader> headers = [];
|
||||
// List<Business> filteredBusinesses = [];
|
||||
// for (var business in widget.groupedBusinesses.) {
|
||||
// if (business.name.toLowerCase().contains(searchFilter.toLowerCase())) {
|
||||
// filteredBusinesses.add(business);
|
||||
// }
|
||||
// }
|
||||
|
||||
// if (filters.isNotEmpty) {
|
||||
// isFiltered = true;
|
||||
// }
|
||||
|
||||
// for (var i = 0; i < businessTypes.length; i++) {
|
||||
// if (filters.contains(businessTypes[i])) {
|
||||
// isFiltered = true;
|
||||
// }
|
||||
// }
|
||||
|
||||
// if (isFiltered) {
|
||||
// for (JobType jobType in widget.groupedBusinesses.keys) {
|
||||
// if (filters.contains(jobType)) {
|
||||
// headers.add(BusinessHeader(
|
||||
// type: jobType,
|
||||
// widescreen: widget.widescreen,
|
||||
// selectable: widget.selectable,
|
||||
// selectedBusinesses: selectedBusinesses,
|
||||
// businesses: widget.groupedBusinesses[jobType]!));
|
||||
// }
|
||||
// }
|
||||
// } else {
|
||||
for (JobType jobType in widget.groupedBusinesses.keys) {
|
||||
headers.add(BusinessHeader(
|
||||
type: jobType,
|
||||
widescreen: widget.widescreen,
|
||||
selectable: widget.selectable,
|
||||
selectedBusinesses: selectedBusinesses,
|
||||
businesses: widget.groupedBusinesses[jobType]!));
|
||||
}
|
||||
// }
|
||||
headers.sort((a, b) => a.type.index.compareTo(b.type.index));
|
||||
return MultiSliver(children: headers);
|
||||
}
|
||||
}
|
||||
|
||||
class BusinessHeader extends StatefulWidget {
|
||||
final JobType type;
|
||||
final List<Business> businesses;
|
||||
final Set<Business> selectedBusinesses;
|
||||
final bool widescreen;
|
||||
final bool selectable;
|
||||
|
||||
const BusinessHeader({
|
||||
super.key,
|
||||
required this.type,
|
||||
required this.businesses,
|
||||
required this.selectedBusinesses,
|
||||
required this.widescreen,
|
||||
required this.selectable,
|
||||
});
|
||||
|
||||
@override
|
||||
State<BusinessHeader> createState() => _BusinessHeaderState();
|
||||
}
|
||||
|
||||
class _BusinessHeaderState extends State<BusinessHeader> {
|
||||
refresh() {
|
||||
setState(() {});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return SliverStickyHeader(
|
||||
header: Container(
|
||||
height: 55.0,
|
||||
color: Theme.of(context).colorScheme.primary,
|
||||
padding: const EdgeInsets.symmetric(horizontal: 16.0),
|
||||
alignment: Alignment.centerLeft,
|
||||
child: _getHeaderRow(widget.selectable),
|
||||
),
|
||||
sliver: _getChildSliver(
|
||||
widget.businesses, widget.widescreen, widget.selectable),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _getHeaderRow(bool selectable) {
|
||||
if (selectable) {
|
||||
return Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
children: [
|
||||
Row(
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(left: 4.0, right: 12.0),
|
||||
child: getIconFromJobType(
|
||||
widget.type, 24, Theme.of(context).colorScheme.onPrimary),
|
||||
),
|
||||
Text(getNameFromJobType(widget.type)),
|
||||
],
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(right: 12.0),
|
||||
child: Checkbox(
|
||||
checkColor: Theme.of(context).colorScheme.primary,
|
||||
activeColor: Theme.of(context).colorScheme.onPrimary,
|
||||
value: selectedBusinesses.containsAll(widget.businesses),
|
||||
onChanged: (value) {
|
||||
if (value!) {
|
||||
setState(() {
|
||||
selectedBusinesses.addAll(widget.businesses);
|
||||
});
|
||||
} else {
|
||||
setState(() {
|
||||
selectedBusinesses.removeAll(widget.businesses);
|
||||
});
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
} else {
|
||||
return Row(
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(left: 4.0, right: 12.0),
|
||||
child: getIconFromJobType(
|
||||
widget.type, 24, Theme.of(context).colorScheme.onPrimary),
|
||||
),
|
||||
Text(
|
||||
getNameFromJobType(widget.type),
|
||||
style: TextStyle(color: Theme.of(context).colorScheme.onPrimary),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Widget _getChildSliver(
|
||||
List<Business> businesses, bool widescreen, bool selectable) {
|
||||
if (widescreen) {
|
||||
return SliverGrid(
|
||||
gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent(
|
||||
mainAxisExtent: 250.0,
|
||||
maxCrossAxisExtent: 400.0,
|
||||
mainAxisSpacing: 10.0,
|
||||
crossAxisSpacing: 10.0,
|
||||
// childAspectRatio: 4.0,
|
||||
),
|
||||
delegate: SliverChildBuilderDelegate(
|
||||
childCount: businesses.length,
|
||||
(BuildContext context, int index) {
|
||||
return BusinessCard(
|
||||
business: businesses[index],
|
||||
selectable: selectable,
|
||||
widescreen: widescreen,
|
||||
callback: refresh,
|
||||
type: widget.type,
|
||||
);
|
||||
},
|
||||
),
|
||||
);
|
||||
} else {
|
||||
return SliverList(
|
||||
delegate: SliverChildBuilderDelegate(
|
||||
childCount: businesses.length,
|
||||
(BuildContext context, int index) {
|
||||
return BusinessCard(
|
||||
business: businesses[index],
|
||||
selectable: selectable,
|
||||
widescreen: widescreen,
|
||||
callback: refresh,
|
||||
type: widget.type,
|
||||
);
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class BusinessCard extends StatefulWidget {
|
||||
final Business business;
|
||||
final bool widescreen;
|
||||
final bool selectable;
|
||||
final Function callback;
|
||||
final JobType type;
|
||||
|
||||
const BusinessCard(
|
||||
{super.key,
|
||||
required this.business,
|
||||
required this.widescreen,
|
||||
required this.selectable,
|
||||
required this.callback,
|
||||
required this.type});
|
||||
|
||||
@override
|
||||
State<BusinessCard> createState() => _BusinessCardState();
|
||||
}
|
||||
|
||||
class _BusinessCardState extends State<BusinessCard> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
if (widget.widescreen) {
|
||||
return _businessTile(widget.business, widget.selectable, widget.type);
|
||||
} else {
|
||||
return _businessListItem(
|
||||
widget.business, widget.selectable, widget.callback, widget.type);
|
||||
}
|
||||
}
|
||||
|
||||
Widget _businessTile(Business business, bool selectable, JobType type) {
|
||||
return MouseRegion(
|
||||
cursor: SystemMouseCursors.click,
|
||||
child: GestureDetector(
|
||||
onTap: () {
|
||||
Navigator.of(context).push(MaterialPageRoute(
|
||||
builder: (context) => BusinessDetail(
|
||||
id: business.id,
|
||||
name: business.name,
|
||||
clickFromType: type,
|
||||
)));
|
||||
},
|
||||
child: Card(
|
||||
clipBehavior: Clip.antiAlias,
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.center,
|
||||
children: [
|
||||
_getTileRow(business, selectable, widget.callback, type),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
business.description,
|
||||
maxLines: selectable ? 7 : 5,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
),
|
||||
const Spacer(),
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: !selectable
|
||||
? Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||
children: [
|
||||
IconButton(
|
||||
icon: const Icon(Icons.link),
|
||||
onPressed: () {
|
||||
launchUrl(
|
||||
Uri.parse('https://${business.website}'));
|
||||
},
|
||||
),
|
||||
if (business.locationName != '')
|
||||
IconButton(
|
||||
icon: const Icon(Icons.location_on),
|
||||
onPressed: () {
|
||||
launchUrl(Uri.parse(Uri.encodeFull(
|
||||
'https://www.google.com/maps/search/?api=1&query=${business.locationName}')));
|
||||
},
|
||||
),
|
||||
if ((business.contactPhone != null) &&
|
||||
(business.contactPhone != ''))
|
||||
IconButton(
|
||||
icon: const Icon(Icons.phone),
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
backgroundColor: Theme.of(context)
|
||||
.colorScheme
|
||||
.surface,
|
||||
title: Text((business.contactName ==
|
||||
null ||
|
||||
business.contactName == '')
|
||||
? 'Contact ${business.name}?'
|
||||
: 'Contact ${business.contactName}'),
|
||||
content: Text((business.contactName ==
|
||||
null ||
|
||||
business.contactName == '')
|
||||
? 'Would you like to call or text ${business.name}?'
|
||||
: 'Would you like to call or text ${business.contactName}?'),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Text'),
|
||||
onPressed: () {
|
||||
launchUrl(Uri.parse(
|
||||
'sms:${business.contactPhone}'));
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Call'),
|
||||
onPressed: () async {
|
||||
launchUrl(Uri.parse(
|
||||
'tel:${business.contactPhone}'));
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
],
|
||||
);
|
||||
});
|
||||
},
|
||||
),
|
||||
if (business.contactEmail != '')
|
||||
IconButton(
|
||||
icon: const Icon(Icons.email),
|
||||
onPressed: () {
|
||||
launchUrl(Uri.parse(
|
||||
'mailto:${business.contactEmail}'));
|
||||
},
|
||||
),
|
||||
],
|
||||
)
|
||||
: null),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _getTileRow(
|
||||
Business business, bool selectable, Function callback, JobType type) {
|
||||
if (selectable) {
|
||||
return Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(6.0),
|
||||
child: Image.network('$apiAddress/logos/${business.id}',
|
||||
height: 48, width: 48, errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return getIconFromJobType(
|
||||
type, 48, Theme.of(context).colorScheme.onSurface);
|
||||
}),
|
||||
),
|
||||
),
|
||||
Flexible(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
business.name,
|
||||
style:
|
||||
const TextStyle(fontSize: 18, fontWeight: FontWeight.bold),
|
||||
maxLines: 2,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(right: 24.0),
|
||||
child: _checkbox(callback),
|
||||
)
|
||||
],
|
||||
);
|
||||
} else {
|
||||
return Row(
|
||||
children: [
|
||||
Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(6.0),
|
||||
child: Image.network('$apiAddress/logos/${business.id}',
|
||||
height: 48, width: 48, errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return getIconFromJobType(
|
||||
type, 48, Theme.of(context).colorScheme.onSurface);
|
||||
}),
|
||||
)),
|
||||
Flexible(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(8.0),
|
||||
child: Text(
|
||||
business.name,
|
||||
style:
|
||||
const TextStyle(fontSize: 18, fontWeight: FontWeight.bold),
|
||||
maxLines: 2,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Widget _businessListItem(
|
||||
Business business, bool selectable, Function callback, JobType type) {
|
||||
return Card(
|
||||
child: ListTile(
|
||||
leading: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(3.0),
|
||||
child: Image.network('$apiAddress/logos/${business.id}',
|
||||
height: 24, width: 24, errorBuilder: (BuildContext context,
|
||||
Object exception, StackTrace? stackTrace) {
|
||||
return getIconFromJobType(
|
||||
type, 24, Theme.of(context).colorScheme.onSurface);
|
||||
})),
|
||||
title: Text(business.name),
|
||||
subtitle: Text(business.description,
|
||||
maxLines: 1, overflow: TextOverflow.ellipsis),
|
||||
trailing: _getCheckbox(selectable, callback),
|
||||
onTap: () {
|
||||
Navigator.of(context).push(MaterialPageRoute(
|
||||
builder: (context) => BusinessDetail(
|
||||
id: business.id,
|
||||
name: business.name,
|
||||
clickFromType: type,
|
||||
)));
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _checkbox(Function callback) {
|
||||
return Checkbox(
|
||||
value: selectedBusinesses.contains(widget.business),
|
||||
onChanged: (value) {
|
||||
if (value!) {
|
||||
setState(() {
|
||||
selectedBusinesses.add(widget.business);
|
||||
});
|
||||
} else {
|
||||
setState(() {
|
||||
selectedBusinesses.remove(widget.business);
|
||||
});
|
||||
}
|
||||
callback();
|
||||
},
|
||||
);
|
||||
}
|
||||
|
||||
Widget? _getCheckbox(bool selectable, Function callback) {
|
||||
if (selectable) {
|
||||
return _checkbox(callback);
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class BusinessSearchBar extends StatefulWidget {
|
||||
final Set<JobType> filters;
|
||||
final Future<void> Function(Set<JobType>) setFiltersCallback;
|
||||
final Future<void> Function(String) setSearchCallback;
|
||||
|
||||
const BusinessSearchBar(
|
||||
{super.key,
|
||||
required this.filters,
|
||||
required this.setFiltersCallback,
|
||||
required this.setSearchCallback});
|
||||
|
||||
@override
|
||||
State<BusinessSearchBar> createState() => _BusinessSearchBarState();
|
||||
}
|
||||
|
||||
class _BusinessSearchBarState extends State<BusinessSearchBar> {
|
||||
bool isFiltered = false;
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
Set<JobType> selectedChips = Set.from(widget.filters);
|
||||
return SizedBox(
|
||||
width: 800,
|
||||
height: 50,
|
||||
child: SearchBar(
|
||||
backgroundColor: WidgetStateProperty.resolveWith((notNeeded) {
|
||||
return Theme.of(context).colorScheme.surfaceContainer;
|
||||
}),
|
||||
onChanged: (query) {
|
||||
widget.setSearchCallback(query);
|
||||
},
|
||||
leading: const Padding(
|
||||
padding: EdgeInsets.only(left: 8.0),
|
||||
child: Icon(Icons.search),
|
||||
),
|
||||
trailing: [
|
||||
IconButton(
|
||||
tooltip: 'Filters',
|
||||
icon: Icon(Icons.filter_list,
|
||||
color: isFiltered
|
||||
? Theme.of(context).colorScheme.primary
|
||||
: Theme.of(context).colorScheme.onSurface),
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
// DO NOT MOVE TO SEPARATE WIDGET, setState is needed in main tree
|
||||
title: const Text('Filter Options'),
|
||||
content: FilterChips(
|
||||
selectedChips: selectedChips,
|
||||
),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Reset'),
|
||||
onPressed: () async {
|
||||
setState(() {
|
||||
selectedChips = <JobType>{};
|
||||
isFiltered = false;
|
||||
});
|
||||
widget.setFiltersCallback(<JobType>{});
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Cancel'),
|
||||
onPressed: () {
|
||||
selectedChips = Set.from(widget.filters);
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Apply'),
|
||||
onPressed: () async {
|
||||
widget.setFiltersCallback(
|
||||
Set.from(selectedChips));
|
||||
if (selectedChips.isNotEmpty) {
|
||||
setState(() {
|
||||
isFiltered = true;
|
||||
});
|
||||
} else {
|
||||
setState(() {
|
||||
isFiltered = false;
|
||||
});
|
||||
}
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
],
|
||||
);
|
||||
});
|
||||
},
|
||||
)
|
||||
]),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class FilterChips extends StatefulWidget {
|
||||
final Set<JobType> selectedChips;
|
||||
|
||||
const FilterChips({super.key, required this.selectedChips});
|
||||
|
||||
@override
|
||||
State<FilterChips> createState() => _FilterChipsState();
|
||||
}
|
||||
|
||||
class _FilterChipsState extends State<FilterChips> {
|
||||
List<Padding> filterChips() {
|
||||
List<Padding> chips = [];
|
||||
|
||||
for (var type in JobType.values) {
|
||||
chips.add(Padding(
|
||||
padding: const EdgeInsets.only(left: 4.0, right: 4.0),
|
||||
child: FilterChip(
|
||||
showCheckmark: false,
|
||||
shape:
|
||||
RoundedRectangleBorder(borderRadius: BorderRadius.circular(20)),
|
||||
label: Text(getNameFromJobType(type)),
|
||||
selected: widget.selectedChips.contains(type),
|
||||
onSelected: (bool selected) {
|
||||
setState(() {
|
||||
if (selected) {
|
||||
widget.selectedChips.add(type);
|
||||
} else {
|
||||
widget.selectedChips.remove(type);
|
||||
}
|
||||
});
|
||||
}),
|
||||
));
|
||||
}
|
||||
return chips;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Wrap(
|
||||
children: filterChips(),
|
||||
);
|
||||
}
|
||||
}
|
||||
@ -2,7 +2,8 @@ import 'dart:async';
|
||||
import 'dart:convert';
|
||||
import 'dart:io';
|
||||
|
||||
import 'package:fbla_ui/shared.dart';
|
||||
import 'package:fbla_ui/shared/global_vars.dart';
|
||||
import 'package:fbla_ui/shared/utils.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
|
||||
var apiAddress = 'https://homelab.marinodev.com/fbla-api';
|
||||
@ -49,14 +50,14 @@ Future fetchBusinessNames() async {
|
||||
}
|
||||
}
|
||||
|
||||
Future fetchBusinessDataOverview({List<JobType>? typeFilters}) async {
|
||||
Future fetchBusinessDataOverviewJobs({List<JobType>? typeFilters}) async {
|
||||
try {
|
||||
String? typeString =
|
||||
typeFilters?.map((jobType) => jobType.name).toList().join(',');
|
||||
Uri uri =
|
||||
Uri.parse('$apiAddress/businessdata/overview?filters=$typeString');
|
||||
Uri.parse('$apiAddress/businessdata/overview/jobs?filters=$typeString');
|
||||
if (typeFilters == null || typeFilters.isEmpty) {
|
||||
uri = Uri.parse('$apiAddress/businessdata/overview');
|
||||
uri = Uri.parse('$apiAddress/businessdata/overview/jobs');
|
||||
}
|
||||
var response = await http.get(uri).timeout(const Duration(seconds: 20));
|
||||
if (response.statusCode == 200) {
|
||||
@ -85,6 +86,43 @@ Future fetchBusinessDataOverview({List<JobType>? typeFilters}) async {
|
||||
}
|
||||
}
|
||||
|
||||
Future fetchBusinessDataOverviewTypes({List<BusinessType>? typeFilters}) async {
|
||||
try {
|
||||
String? typeString =
|
||||
typeFilters?.map((jobType) => jobType.name).toList().join(',');
|
||||
Uri uri = Uri.parse(
|
||||
'$apiAddress/businessdata/overview/types?filters=$typeString');
|
||||
if (typeFilters == null || typeFilters.isEmpty) {
|
||||
uri = Uri.parse('$apiAddress/businessdata/overview/types');
|
||||
}
|
||||
var response = await http.get(uri).timeout(const Duration(seconds: 20));
|
||||
if (response.statusCode == 200) {
|
||||
var decodedResponse = json.decode(response.body);
|
||||
Map<BusinessType, List<Business>> groupedBusinesses = {};
|
||||
|
||||
for (String stringType in decodedResponse.keys) {
|
||||
List<Business> businesses = [];
|
||||
|
||||
for (Map<String, dynamic> map in decodedResponse[stringType]) {
|
||||
map.addAll({'type': stringType});
|
||||
Business business = Business.fromJson(map);
|
||||
businesses.add(business);
|
||||
}
|
||||
|
||||
groupedBusinesses
|
||||
.addAll({BusinessType.values.byName(stringType): businesses});
|
||||
}
|
||||
return groupedBusinesses;
|
||||
} else {
|
||||
return 'Error ${response.statusCode}! Please try again later!';
|
||||
}
|
||||
} on TimeoutException {
|
||||
return 'Unable to connect to server (timeout).\nPlease try again later.';
|
||||
} on SocketException {
|
||||
return 'Unable to connect to server (socket exception).\nPlease check your internet connection.\n';
|
||||
}
|
||||
}
|
||||
|
||||
Future fetchBusinesses(List<int> ids) async {
|
||||
try {
|
||||
var response = await http
|
||||
@ -129,13 +167,34 @@ Future fetchBusiness(int id) async {
|
||||
}
|
||||
}
|
||||
|
||||
Future fetchJob(int id) async {
|
||||
try {
|
||||
var response = await http
|
||||
.get(Uri.parse('$apiAddress/businessdata/jobs/$id'))
|
||||
.timeout(const Duration(seconds: 20));
|
||||
if (response.statusCode == 200) {
|
||||
var decodedResponse = json.decode(response.body);
|
||||
Business business = Business.fromJson(decodedResponse);
|
||||
|
||||
return business;
|
||||
} else {
|
||||
return 'Error ${response.statusCode}! Please try again later!';
|
||||
}
|
||||
} on TimeoutException {
|
||||
return 'Unable to connect to server (timeout).\nPlease try again later.';
|
||||
} on SocketException {
|
||||
return 'Unable to connect to server (socket exception).\nPlease check your internet connection.\n';
|
||||
}
|
||||
}
|
||||
|
||||
Future createBusiness(Business business) async {
|
||||
var json = '''
|
||||
{
|
||||
"id": ${business.id},
|
||||
"name": "${business.name}",
|
||||
"description": "${business.description}",
|
||||
"description": "${business.description?.replaceAll('\n', '\\n')}",
|
||||
"website": "${business.website}",
|
||||
"type": "${business.type!.name}",
|
||||
"contactName": "${business.contactName}",
|
||||
"contactEmail": "${business.contactEmail}",
|
||||
"contactPhone": "${business.contactPhone}",
|
||||
@ -165,14 +224,14 @@ Future createListing(JobListing listing) async {
|
||||
"id": ${listing.id},
|
||||
"businessId": ${listing.businessId},
|
||||
"name": "${listing.name}",
|
||||
"description": "${listing.description}",
|
||||
"description": "${listing.description.replaceAll('\n', '\\n')}",
|
||||
"wage": "${listing.wage}",
|
||||
"link": "${listing.link}"
|
||||
}
|
||||
''';
|
||||
|
||||
try {
|
||||
var response = await http.post(Uri.parse('$apiAddress/createbusiness'),
|
||||
var response = await http.post(Uri.parse('$apiAddress/createlisting'),
|
||||
body: json,
|
||||
headers: {'Authorization': jwt}).timeout(const Duration(seconds: 20));
|
||||
if (response.statusCode != 200) {
|
||||
@ -232,8 +291,9 @@ Future editBusiness(Business business) async {
|
||||
{
|
||||
"id": ${business.id},
|
||||
"name": "${business.name}",
|
||||
"description": "${business.description}",
|
||||
"description": "${business.description?.replaceAll('\n', '\\n')}",
|
||||
"website": "${business.website}",
|
||||
"type": "${business.type!.name}",
|
||||
"contactName": "${business.contactName}",
|
||||
"contactEmail": "${business.contactEmail}",
|
||||
"contactPhone": "${business.contactPhone}",
|
||||
@ -262,8 +322,8 @@ Future editListing(JobListing listing) async {
|
||||
"id": ${listing.id},
|
||||
"businessId": ${listing.businessId},
|
||||
"name": "${listing.name}",
|
||||
"description": "${listing.description}",
|
||||
"type": "${listing.type.name}",
|
||||
"description": "${listing.description.replaceAll('\n', '\\n')}",
|
||||
"type": "${listing.type!.name}",
|
||||
"wage": "${listing.wage}",
|
||||
"link": "${listing.link}"
|
||||
}
|
||||
488
fbla_ui/lib/shared/export.dart
Normal file
488
fbla_ui/lib/shared/export.dart
Normal file
@ -0,0 +1,488 @@
|
||||
import 'dart:io';
|
||||
|
||||
import 'package:fbla_ui/shared/api_logic.dart';
|
||||
import 'package:fbla_ui/shared/utils.dart';
|
||||
import 'package:flutter/foundation.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
import 'package:open_filex/open_filex.dart';
|
||||
import 'package:path_provider/path_provider.dart';
|
||||
import 'package:pdf/pdf.dart';
|
||||
import 'package:pdf/widgets.dart' as pw;
|
||||
import 'package:printing/printing.dart';
|
||||
|
||||
class _FilterBusinessDataTypeChips extends StatefulWidget {
|
||||
final Set<DataTypeBusiness> selectedDataTypesBusiness;
|
||||
|
||||
const _FilterBusinessDataTypeChips({required this.selectedDataTypesBusiness});
|
||||
|
||||
@override
|
||||
State<_FilterBusinessDataTypeChips> createState() =>
|
||||
_FilterBusinessDataTypeChipsState();
|
||||
}
|
||||
|
||||
class _FilterBusinessDataTypeChipsState
|
||||
extends State<_FilterBusinessDataTypeChips> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
List<Padding> chips = [];
|
||||
for (var type in DataTypeBusiness.values) {
|
||||
chips.add(Padding(
|
||||
padding:
|
||||
const EdgeInsets.only(left: 3.0, right: 3.0, bottom: 3.0, top: 3.0),
|
||||
child: FilterChip(
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(20),
|
||||
side:
|
||||
BorderSide(color: Theme.of(context).colorScheme.secondary)),
|
||||
label: Text(dataTypeFriendlyBusiness[type]!),
|
||||
showCheckmark: false,
|
||||
selected: widget.selectedDataTypesBusiness.contains(type),
|
||||
onSelected: (bool selected) {
|
||||
setState(() {
|
||||
if (selected) {
|
||||
widget.selectedDataTypesBusiness.add(type);
|
||||
} else {
|
||||
widget.selectedDataTypesBusiness.remove(type);
|
||||
}
|
||||
});
|
||||
}),
|
||||
));
|
||||
}
|
||||
return Wrap(
|
||||
children: chips,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class _FilterJobDataTypeChips extends StatefulWidget {
|
||||
final Set<DataTypeJob> selectedDataTypesJob;
|
||||
|
||||
const _FilterJobDataTypeChips({required this.selectedDataTypesJob});
|
||||
|
||||
@override
|
||||
State<_FilterJobDataTypeChips> createState() =>
|
||||
_FilterJobDataTypeChipsState();
|
||||
}
|
||||
|
||||
class _FilterJobDataTypeChipsState extends State<_FilterJobDataTypeChips> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
List<Padding> chips = [];
|
||||
for (var type in DataTypeJob.values) {
|
||||
chips.add(Padding(
|
||||
padding:
|
||||
const EdgeInsets.only(left: 3.0, right: 3.0, bottom: 3.0, top: 3.0),
|
||||
child: FilterChip(
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(20),
|
||||
side:
|
||||
BorderSide(color: Theme.of(context).colorScheme.secondary)),
|
||||
label: Text(dataTypeFriendlyJob[type]!),
|
||||
showCheckmark: false,
|
||||
selected: widget.selectedDataTypesJob.contains(type),
|
||||
onSelected: (bool selected) {
|
||||
setState(() {
|
||||
if (selected) {
|
||||
widget.selectedDataTypesJob.add(type);
|
||||
} else {
|
||||
widget.selectedDataTypesJob.remove(type);
|
||||
}
|
||||
});
|
||||
}),
|
||||
));
|
||||
}
|
||||
return Wrap(
|
||||
children: chips,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> generatePDF(
|
||||
{required BuildContext context,
|
||||
required int documentTypeIndex,
|
||||
Set<Business>? selectedBusinesses,
|
||||
Set<Business>? selectedJobs}) async {
|
||||
List<pw.Widget> headerColumns = [];
|
||||
List<pw.TableRow> tableRows = [];
|
||||
|
||||
Set<DataTypeBusiness> dataTypesBusiness = {};
|
||||
Set<DataTypeJob> dataTypesJob = {};
|
||||
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
title: const Text('Export Settings'),
|
||||
content: SizedBox(
|
||||
width: 400,
|
||||
height: 200,
|
||||
child: Column(
|
||||
children: [
|
||||
const Padding(
|
||||
padding: EdgeInsets.all(8.0),
|
||||
child: Text('Data columns you want to export:'),
|
||||
),
|
||||
documentTypeIndex == 0
|
||||
? _FilterBusinessDataTypeChips(
|
||||
selectedDataTypesBusiness: dataTypesBusiness,
|
||||
)
|
||||
: _FilterJobDataTypeChips(
|
||||
selectedDataTypesJob: dataTypesJob)
|
||||
],
|
||||
),
|
||||
),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Cancel'),
|
||||
onPressed: () {
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Generate'),
|
||||
onPressed: () async {
|
||||
if (documentTypeIndex == 0) {
|
||||
List<Business> businesses = await fetchBusinesses(
|
||||
selectedBusinesses!
|
||||
.map((business) => business.id)
|
||||
.toList());
|
||||
if (dataTypesBusiness.isEmpty) {
|
||||
dataTypesBusiness.addAll(DataTypeBusiness.values);
|
||||
}
|
||||
dataTypesBusiness =
|
||||
sortDataTypesBusiness(dataTypesBusiness);
|
||||
|
||||
for (Business business in businesses) {
|
||||
List<pw.Widget> businessRow = [];
|
||||
if (dataTypesBusiness.contains(DataTypeBusiness.logo)) {
|
||||
var apiLogo = await getLogo(business.id);
|
||||
if (apiLogo.runtimeType != String) {
|
||||
businessRow.add(pw.Padding(
|
||||
child: pw.ClipRRect(
|
||||
child: pw.Image(pw.MemoryImage(apiLogo),
|
||||
height: 24, width: 24),
|
||||
horizontalRadius: 4,
|
||||
verticalRadius: 4),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
} else {
|
||||
businessRow.add(pw.Padding(
|
||||
child: pw.Icon(
|
||||
getPwIconFromBusinessType(business.type!),
|
||||
size: 24),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
}
|
||||
}
|
||||
for (DataTypeBusiness dataType in dataTypesBusiness) {
|
||||
if (dataType != DataTypeBusiness.logo) {
|
||||
var currentValue =
|
||||
businessValueFromDataType(business, dataType);
|
||||
if (currentValue != null) {
|
||||
businessRow.add(pw.Padding(
|
||||
child: pw.Text(businessValueFromDataType(
|
||||
business, dataType)),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
} else {
|
||||
businessRow.add(pw.Container());
|
||||
}
|
||||
}
|
||||
}
|
||||
tableRows.add(pw.TableRow(children: businessRow));
|
||||
}
|
||||
|
||||
for (var filter in dataTypesBusiness) {
|
||||
headerColumns.add(pw.Padding(
|
||||
child: pw.Text(dataTypeFriendlyBusiness[filter]!,
|
||||
style: const pw.TextStyle(fontSize: 10)),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
}
|
||||
} else {
|
||||
if (dataTypesJob.isEmpty) {
|
||||
dataTypesJob.addAll(DataTypeJob.values);
|
||||
}
|
||||
dataTypesJob = sortDataTypesJob(dataTypesJob);
|
||||
|
||||
// List<Map<String, dynamic>> nameMapping =
|
||||
// await fetchBusinessNames();
|
||||
|
||||
for (Business business in selectedJobs!) {
|
||||
for (JobListing job in business.listings!) {
|
||||
List<pw.Widget> jobRow = [];
|
||||
for (DataTypeJob dataType in dataTypesJob) {
|
||||
if (dataType != DataTypeJob.businessName) {
|
||||
var currentValue =
|
||||
jobValueFromDataType(job, dataType);
|
||||
if (currentValue != null) {
|
||||
jobRow.add(pw.Padding(
|
||||
child: pw.Text(currentValue),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
} else {
|
||||
jobRow.add(pw.Container());
|
||||
}
|
||||
} else {
|
||||
jobRow.add(pw.Padding(
|
||||
child: pw.Text(business.name!),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
}
|
||||
}
|
||||
tableRows.add(pw.TableRow(children: jobRow));
|
||||
}
|
||||
}
|
||||
|
||||
for (var filter in dataTypesJob) {
|
||||
headerColumns.add(pw.Padding(
|
||||
child: pw.Text(dataTypeFriendlyJob[filter]!,
|
||||
style: const pw.TextStyle(fontSize: 10)),
|
||||
padding: const pw.EdgeInsets.all(4.0)));
|
||||
}
|
||||
}
|
||||
// Final Generation
|
||||
DateTime dateTime = DateTime.now();
|
||||
String minute = '00';
|
||||
if (dateTime.minute.toString().length < 2) {
|
||||
minute = '0${dateTime.minute}';
|
||||
} else {
|
||||
minute = dateTime.minute.toString();
|
||||
}
|
||||
|
||||
String time = dateTime.hour <= 12
|
||||
? '${dateTime.hour}:${minute}AM'
|
||||
: '${dateTime.hour - 12}:${minute}PM';
|
||||
String fileName =
|
||||
'${documentTypeIndex == 0 ? 'Business' : 'Job Listing'} Data - ${dateTime.month}-${dateTime.day}-${dateTime.year} $time.pdf';
|
||||
|
||||
final pdf = pw.Document();
|
||||
var svg = await rootBundle.loadString('assets/MarinoDev.svg');
|
||||
|
||||
var themeIcon = pw.ThemeData.withFont(
|
||||
base: await PdfGoogleFonts.notoSansDisplayMedium(),
|
||||
icons: await PdfGoogleFonts.materialIcons());
|
||||
|
||||
var finalTheme = themeIcon.copyWith(
|
||||
defaultTextStyle: const pw.TextStyle(fontSize: 9),
|
||||
);
|
||||
|
||||
pdf.addPage(pw.MultiPage(
|
||||
theme: finalTheme,
|
||||
pageFormat: PdfPageFormat.letter,
|
||||
orientation: pw.PageOrientation.landscape,
|
||||
margin: const pw.EdgeInsets.all(24),
|
||||
build: (pw.Context context) {
|
||||
return [
|
||||
pw.Row(
|
||||
mainAxisAlignment:
|
||||
pw.MainAxisAlignment.spaceBetween,
|
||||
children: [
|
||||
pw.SvgImage(svg: svg, height: 40),
|
||||
pw.Padding(
|
||||
padding: const pw.EdgeInsets.all(8.0),
|
||||
child: pw.Text(
|
||||
'${documentTypeIndex == 0 ? 'Business' : 'Job Listing'} Datasheet',
|
||||
style: pw.TextStyle(
|
||||
fontSize: 32,
|
||||
fontWeight: pw.FontWeight.bold)),
|
||||
),
|
||||
pw.Text(
|
||||
'Generated on ${dateTime.month}/${dateTime.day}/${dateTime.year} at $time',
|
||||
style: const pw.TextStyle(fontSize: 12),
|
||||
textAlign: pw.TextAlign.right),
|
||||
//
|
||||
]),
|
||||
pw.Table(
|
||||
columnWidths: documentTypeIndex == 0
|
||||
? _businessColumnSizes(dataTypesBusiness)
|
||||
: _jobColumnSizes(dataTypesJob),
|
||||
border: const pw.TableBorder(
|
||||
bottom: pw.BorderSide(),
|
||||
left: pw.BorderSide(),
|
||||
right: pw.BorderSide(),
|
||||
top: pw.BorderSide(),
|
||||
horizontalInside: pw.BorderSide(),
|
||||
verticalInside: pw.BorderSide()),
|
||||
children: [
|
||||
pw.TableRow(
|
||||
decoration: const pw.BoxDecoration(
|
||||
color: PdfColors.blue400),
|
||||
children: headerColumns,
|
||||
repeat: true,
|
||||
),
|
||||
...tableRows,
|
||||
])
|
||||
];
|
||||
}));
|
||||
|
||||
Uint8List pdfBytes = await pdf.save();
|
||||
|
||||
if (kIsWeb) {
|
||||
await Printing.sharePdf(
|
||||
bytes: await pdf.save(),
|
||||
filename: fileName,
|
||||
);
|
||||
} else {
|
||||
var dir = await getTemporaryDirectory();
|
||||
var tempDir = dir.path;
|
||||
|
||||
File pdfFile = File('$tempDir/$fileName');
|
||||
pdfFile.writeAsBytesSync(pdfBytes);
|
||||
|
||||
OpenFilex.open(pdfFile.path);
|
||||
}
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
],
|
||||
);
|
||||
});
|
||||
}
|
||||
|
||||
Map<int, pw.TableColumnWidth> _businessColumnSizes(
|
||||
Set<DataTypeBusiness> dataTypes) {
|
||||
double space = 744.0;
|
||||
List<DataTypeBusiness> sorted = sortDataTypesBusiness(dataTypes).toList();
|
||||
Map<int, pw.TableColumnWidth> map = {};
|
||||
|
||||
if (sorted.contains(DataTypeBusiness.logo)) {
|
||||
space -= 32;
|
||||
map.addAll(
|
||||
{sorted.indexOf(DataTypeBusiness.logo): const pw.FixedColumnWidth(32)});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.contactName)) {
|
||||
space -= 72;
|
||||
map.addAll({
|
||||
sorted.indexOf(DataTypeBusiness.contactName):
|
||||
const pw.FixedColumnWidth(72)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.contactPhone)) {
|
||||
space -= 76;
|
||||
map.addAll({
|
||||
sorted.indexOf(DataTypeBusiness.contactPhone):
|
||||
const pw.FixedColumnWidth(76)
|
||||
});
|
||||
}
|
||||
double leftNum = 0;
|
||||
if (dataTypes.contains(DataTypeBusiness.name)) {
|
||||
leftNum += 1;
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.website)) {
|
||||
leftNum += 1;
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.contactEmail)) {
|
||||
leftNum += 1;
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.notes)) {
|
||||
leftNum += 2;
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.description)) {
|
||||
leftNum += 3;
|
||||
}
|
||||
leftNum = space / leftNum;
|
||||
if (dataTypes.contains(DataTypeBusiness.name)) {
|
||||
map.addAll(
|
||||
{sorted.indexOf(DataTypeBusiness.name): pw.FixedColumnWidth(leftNum)});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.website)) {
|
||||
map.addAll({
|
||||
sorted.indexOf(DataTypeBusiness.website): pw.FixedColumnWidth(leftNum)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.contactEmail)) {
|
||||
map.addAll({
|
||||
sorted.indexOf(DataTypeBusiness.contactEmail):
|
||||
pw.FixedColumnWidth(leftNum)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.notes)) {
|
||||
map.addAll({
|
||||
sorted.indexOf(DataTypeBusiness.notes): pw.FixedColumnWidth(leftNum * 2)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeBusiness.description)) {
|
||||
map.addAll({
|
||||
sorted.indexOf(DataTypeBusiness.description):
|
||||
pw.FixedColumnWidth(leftNum * 3)
|
||||
});
|
||||
}
|
||||
|
||||
return map;
|
||||
}
|
||||
|
||||
Map<int, pw.TableColumnWidth> _jobColumnSizes(Set<DataTypeJob> dataTypes) {
|
||||
Map<int, pw.TableColumnWidth> map = {};
|
||||
List<DataTypeJob> sortedDataTypes = sortDataTypesJob(dataTypes).toList();
|
||||
|
||||
if (dataTypes.contains(DataTypeJob.businessName)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.businessName)
|
||||
.first): const pw.FractionColumnWidth(0.2)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeJob.name)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.name)
|
||||
.first): const pw.FractionColumnWidth(0.2)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeJob.description)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.description)
|
||||
.first): const pw.FractionColumnWidth(0.4)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeJob.wage)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.wage)
|
||||
.first): const pw.FractionColumnWidth(0.15)
|
||||
});
|
||||
}
|
||||
if (dataTypes.contains(DataTypeJob.link)) {
|
||||
map.addAll({
|
||||
sortedDataTypes.indexOf(sortedDataTypes
|
||||
.where((element) => element == DataTypeJob.link)
|
||||
.first): const pw.FractionColumnWidth(0.2)
|
||||
});
|
||||
}
|
||||
|
||||
return map;
|
||||
}
|
||||
|
||||
dynamic businessValueFromDataType(
|
||||
Business business, DataTypeBusiness dataType) {
|
||||
switch (dataType) {
|
||||
case DataTypeBusiness.name:
|
||||
return business.name;
|
||||
case DataTypeBusiness.description:
|
||||
return business.description;
|
||||
case DataTypeBusiness.type:
|
||||
return business.type;
|
||||
case DataTypeBusiness.website:
|
||||
return business.website;
|
||||
case DataTypeBusiness.contactName:
|
||||
return business.contactName;
|
||||
case DataTypeBusiness.contactEmail:
|
||||
return business.contactEmail;
|
||||
case DataTypeBusiness.contactPhone:
|
||||
return business.contactPhone;
|
||||
case DataTypeBusiness.notes:
|
||||
return business.notes;
|
||||
case DataTypeBusiness.logo:
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
dynamic jobValueFromDataType(JobListing job, DataTypeJob dataType) {
|
||||
switch (dataType) {
|
||||
case DataTypeJob.name:
|
||||
return job.name;
|
||||
case DataTypeJob.description:
|
||||
return job.description;
|
||||
case DataTypeJob.wage:
|
||||
return job.wage;
|
||||
case DataTypeJob.link:
|
||||
return job.link;
|
||||
case DataTypeJob.businessName:
|
||||
return null;
|
||||
}
|
||||
}
|
||||
6
fbla_ui/lib/shared/global_vars.dart
Normal file
6
fbla_ui/lib/shared/global_vars.dart
Normal file
@ -0,0 +1,6 @@
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
late String jwt;
|
||||
const int widescreenWidth = 600;
|
||||
bool loggedIn = false;
|
||||
ThemeMode themeMode = ThemeMode.system;
|
||||
300
fbla_ui/lib/shared/utils.dart
Normal file
300
fbla_ui/lib/shared/utils.dart
Normal file
@ -0,0 +1,300 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:pdf/widgets.dart' as pw;
|
||||
|
||||
enum DataTypeBusiness {
|
||||
logo,
|
||||
name,
|
||||
description,
|
||||
website,
|
||||
contactName,
|
||||
contactEmail,
|
||||
contactPhone,
|
||||
notes,
|
||||
type,
|
||||
}
|
||||
|
||||
enum DataTypeJob {
|
||||
businessName,
|
||||
name,
|
||||
description,
|
||||
wage,
|
||||
link,
|
||||
}
|
||||
|
||||
Map<DataTypeBusiness, int> dataTypePriorityBusiness = {
|
||||
DataTypeBusiness.logo: 0,
|
||||
DataTypeBusiness.name: 1,
|
||||
DataTypeBusiness.description: 2,
|
||||
DataTypeBusiness.type: 3,
|
||||
DataTypeBusiness.website: 4,
|
||||
DataTypeBusiness.contactName: 5,
|
||||
DataTypeBusiness.contactEmail: 6,
|
||||
DataTypeBusiness.contactPhone: 7,
|
||||
DataTypeBusiness.notes: 8
|
||||
};
|
||||
|
||||
Map<DataTypeBusiness, String> dataTypeFriendlyBusiness = {
|
||||
DataTypeBusiness.logo: 'Logo',
|
||||
DataTypeBusiness.name: 'Name',
|
||||
DataTypeBusiness.description: 'Description',
|
||||
DataTypeBusiness.type: 'Type',
|
||||
DataTypeBusiness.website: 'Website',
|
||||
DataTypeBusiness.contactName: 'Contact Name',
|
||||
DataTypeBusiness.contactEmail: 'Contact Email',
|
||||
DataTypeBusiness.contactPhone: 'Contact Phone',
|
||||
DataTypeBusiness.notes: 'Notes'
|
||||
};
|
||||
|
||||
Map<DataTypeJob, int> dataTypePriorityJob = {
|
||||
DataTypeJob.businessName: 1,
|
||||
DataTypeJob.name: 2,
|
||||
DataTypeJob.description: 3,
|
||||
DataTypeJob.wage: 4,
|
||||
DataTypeJob.link: 5,
|
||||
};
|
||||
|
||||
Map<DataTypeJob, String> dataTypeFriendlyJob = {
|
||||
DataTypeJob.businessName: 'Business Name',
|
||||
DataTypeJob.name: 'Job Listing Name',
|
||||
DataTypeJob.description: 'Description',
|
||||
DataTypeJob.wage: 'Wage',
|
||||
DataTypeJob.link: 'Additional Info Link',
|
||||
};
|
||||
|
||||
Set<DataTypeBusiness> sortDataTypesBusiness(Set<DataTypeBusiness> set) {
|
||||
List<DataTypeBusiness> list = set.toList();
|
||||
list.sort((a, b) {
|
||||
return dataTypePriorityBusiness[a]!.compareTo(dataTypePriorityBusiness[b]!);
|
||||
});
|
||||
set = list.toSet();
|
||||
return set;
|
||||
}
|
||||
|
||||
Set<DataTypeJob> sortDataTypesJob(Set<DataTypeJob> set) {
|
||||
List<DataTypeJob> list = set.toList();
|
||||
list.sort((a, b) {
|
||||
return dataTypePriorityJob[a]!.compareTo(dataTypePriorityJob[b]!);
|
||||
});
|
||||
set = list.toSet();
|
||||
return set;
|
||||
}
|
||||
|
||||
enum BusinessType {
|
||||
food,
|
||||
shop,
|
||||
outdoors,
|
||||
manufacturing,
|
||||
entertainment,
|
||||
other,
|
||||
}
|
||||
|
||||
enum JobType { cashier, server, mechanic, other }
|
||||
|
||||
class JobListing {
|
||||
int? id;
|
||||
int? businessId;
|
||||
String name;
|
||||
String description;
|
||||
JobType? type;
|
||||
String? wage;
|
||||
String? link;
|
||||
|
||||
JobListing(
|
||||
{this.id,
|
||||
this.businessId,
|
||||
required this.name,
|
||||
required this.description,
|
||||
this.type,
|
||||
this.wage,
|
||||
this.link});
|
||||
|
||||
factory JobListing.copy(JobListing input) {
|
||||
return JobListing(
|
||||
id: input.id,
|
||||
businessId: input.businessId,
|
||||
name: input.name,
|
||||
description: input.description,
|
||||
type: input.type,
|
||||
wage: input.wage,
|
||||
link: input.link,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class Business {
|
||||
int id;
|
||||
String? name;
|
||||
String? description;
|
||||
BusinessType? type;
|
||||
String? website;
|
||||
String? contactName;
|
||||
String? contactEmail;
|
||||
String? contactPhone;
|
||||
String? notes;
|
||||
String locationName;
|
||||
String? locationAddress;
|
||||
List<JobListing>? listings;
|
||||
|
||||
Business(
|
||||
{required this.id,
|
||||
required this.name,
|
||||
required this.description,
|
||||
required this.website,
|
||||
this.type,
|
||||
this.contactName,
|
||||
this.contactEmail,
|
||||
this.contactPhone,
|
||||
this.notes,
|
||||
required this.locationName,
|
||||
this.locationAddress,
|
||||
this.listings});
|
||||
|
||||
factory Business.fromJson(Map<String, dynamic> json) {
|
||||
List<JobListing>? listings;
|
||||
if (json['listings'] != null) {
|
||||
listings = [];
|
||||
for (int i = 0; i < json['listings'].length; i++) {
|
||||
listings.add(JobListing(
|
||||
id: json['listings'][i]['id'],
|
||||
businessId: json['listings'][i]['businessId'],
|
||||
name: json['listings'][i]['name'],
|
||||
description: json['listings'][i]['description'],
|
||||
type: JobType.values.byName(json['listings'][i]['type']),
|
||||
wage: json['listings'][i]['wage'],
|
||||
link: json['listings'][i]['link']));
|
||||
}
|
||||
}
|
||||
|
||||
return Business(
|
||||
id: json['id'],
|
||||
name: json['name'],
|
||||
description: json['description'],
|
||||
type: json['type'] != null
|
||||
? BusinessType.values.byName(json['type'])
|
||||
: null,
|
||||
website: json['website'],
|
||||
contactName: json['contactName'],
|
||||
contactEmail: json['contactEmail'],
|
||||
contactPhone: json['contactPhone'],
|
||||
notes: json['notes'],
|
||||
locationName: json['locationName'],
|
||||
locationAddress: json['locationAddress'],
|
||||
listings: listings);
|
||||
}
|
||||
|
||||
factory Business.copy(Business input) {
|
||||
return Business(
|
||||
id: input.id,
|
||||
name: input.name,
|
||||
description: input.description,
|
||||
website: input.website,
|
||||
contactName: input.contactName,
|
||||
contactEmail: input.contactEmail,
|
||||
contactPhone: input.contactPhone,
|
||||
notes: input.notes,
|
||||
locationName: input.locationName,
|
||||
locationAddress: input.locationAddress,
|
||||
listings: input.listings);
|
||||
}
|
||||
}
|
||||
|
||||
IconData getIconFromBusinessType(BusinessType type) {
|
||||
switch (type) {
|
||||
case BusinessType.food:
|
||||
return Icons.restaurant;
|
||||
case BusinessType.shop:
|
||||
return Icons.store;
|
||||
case BusinessType.outdoors:
|
||||
return Icons.forest;
|
||||
case BusinessType.manufacturing:
|
||||
return Icons.factory;
|
||||
case BusinessType.entertainment:
|
||||
return Icons.live_tv;
|
||||
case BusinessType.other:
|
||||
return Icons.business;
|
||||
}
|
||||
}
|
||||
|
||||
IconData getIconFromJobType(JobType type) {
|
||||
switch (type) {
|
||||
case JobType.cashier:
|
||||
return Icons.shopping_bag;
|
||||
case JobType.server:
|
||||
return Icons.restaurant;
|
||||
case JobType.mechanic:
|
||||
return Icons.construction;
|
||||
case JobType.other:
|
||||
return Icons.work;
|
||||
}
|
||||
}
|
||||
|
||||
pw.IconData getPwIconFromBusinessType(BusinessType type) {
|
||||
switch (type) {
|
||||
case BusinessType.food:
|
||||
return const pw.IconData(0xe56c);
|
||||
case BusinessType.shop:
|
||||
return const pw.IconData(0xea12);
|
||||
case BusinessType.outdoors:
|
||||
return const pw.IconData(0xea99);
|
||||
case BusinessType.manufacturing:
|
||||
return const pw.IconData(0xebbc);
|
||||
case BusinessType.entertainment:
|
||||
return const pw.IconData(0xe639);
|
||||
case BusinessType.other:
|
||||
return const pw.IconData(0xe0af);
|
||||
}
|
||||
}
|
||||
|
||||
pw.IconData getPwIconFromJobType(JobType type) {
|
||||
switch (type) {
|
||||
case JobType.cashier:
|
||||
return const pw.IconData(0xf1cc);
|
||||
case JobType.server:
|
||||
return const pw.IconData(0xe56c);
|
||||
case JobType.mechanic:
|
||||
return const pw.IconData(0xea3c);
|
||||
case JobType.other:
|
||||
return const pw.IconData(0xe8f9);
|
||||
}
|
||||
}
|
||||
|
||||
String getNameFromBusinessType(BusinessType type) {
|
||||
switch (type) {
|
||||
case BusinessType.food:
|
||||
return 'Food Related';
|
||||
case BusinessType.shop:
|
||||
return 'Shops';
|
||||
case BusinessType.outdoors:
|
||||
return 'Outdoors';
|
||||
case BusinessType.manufacturing:
|
||||
return 'Manufacturing';
|
||||
case BusinessType.entertainment:
|
||||
return 'Entertainment';
|
||||
case BusinessType.other:
|
||||
return 'Other';
|
||||
}
|
||||
}
|
||||
|
||||
String getNameFromJobType(JobType type) {
|
||||
switch (type) {
|
||||
case JobType.cashier:
|
||||
return 'Cashier';
|
||||
case JobType.server:
|
||||
return 'Server';
|
||||
case JobType.mechanic:
|
||||
return 'Mechanic';
|
||||
case JobType.other:
|
||||
return 'Other';
|
||||
}
|
||||
}
|
||||
|
||||
IconData getIconFromThemeMode(ThemeMode theme) {
|
||||
switch (theme) {
|
||||
case ThemeMode.dark:
|
||||
return Icons.dark_mode;
|
||||
case ThemeMode.light:
|
||||
return Icons.light_mode;
|
||||
case ThemeMode.system:
|
||||
return Icons.brightness_4;
|
||||
}
|
||||
}
|
||||
821
fbla_ui/lib/shared/widgets.dart
Normal file
821
fbla_ui/lib/shared/widgets.dart
Normal file
@ -0,0 +1,821 @@
|
||||
import 'package:dart_jsonwebtoken/dart_jsonwebtoken.dart';
|
||||
import 'package:fbla_ui/pages/signin_page.dart';
|
||||
import 'package:fbla_ui/shared/global_vars.dart';
|
||||
import 'package:fbla_ui/shared/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
import 'package:url_launcher/url_launcher.dart';
|
||||
|
||||
// class BusinessDisplayPanel extends StatefulWidget {
|
||||
// final Map<JobType, List<Business>>? jobGroupedBusinesses;
|
||||
// final Map<BusinessType, List<Business>>? businessGroupedBusinesses;
|
||||
// final bool widescreen;
|
||||
// final Set<Business>? selectedBusinesses;
|
||||
//
|
||||
// const BusinessDisplayPanel(
|
||||
// {super.key,
|
||||
// this.jobGroupedBusinesses,
|
||||
// this.businessGroupedBusinesses,
|
||||
// required this.widescreen,
|
||||
// this.selectedBusinesses});
|
||||
//
|
||||
// @override
|
||||
// State<BusinessDisplayPanel> createState() => _BusinessDisplayPanelState();
|
||||
// }
|
||||
//
|
||||
// class _BusinessDisplayPanelState extends State<BusinessDisplayPanel> {
|
||||
// @override
|
||||
// Widget build(BuildContext context) {
|
||||
// if ((widget.businessGroupedBusinesses?.keys ?? <BusinessType>[]).isEmpty &&
|
||||
// (widget.jobGroupedBusinesses?.keys ?? <JobType>[]).isEmpty) {
|
||||
// return const SliverToBoxAdapter(
|
||||
// child: Center(
|
||||
// child: Padding(
|
||||
// padding: EdgeInsets.all(16.0),
|
||||
// child: Text(
|
||||
// 'No results found!\nPlease change your search filters.',
|
||||
// textAlign: TextAlign.center,
|
||||
// style: TextStyle(fontSize: 18),
|
||||
// ),
|
||||
// ),
|
||||
// ),
|
||||
// );
|
||||
// }
|
||||
//
|
||||
// List<BusinessHeader> headers = [];
|
||||
// if (widget.jobGroupedBusinesses != null) {
|
||||
// for (JobType jobType in widget.jobGroupedBusinesses!.keys) {
|
||||
// headers.add(BusinessHeader(
|
||||
// jobType: jobType,
|
||||
// widescreen: widget.widescreen,
|
||||
// // selectable: widget.selectable,
|
||||
// selectedBusinesses: widget.selectedBusinesses,
|
||||
// // updateSelectedBusinessesCallback:
|
||||
// // widget.updateSelectedBusinessesCallback,
|
||||
// businesses: widget.jobGroupedBusinesses![jobType]!));
|
||||
// }
|
||||
// headers.sort((a, b) => a.jobType!.index.compareTo(b.jobType!.index));
|
||||
// return MultiSliver(children: headers);
|
||||
// } else if (widget.businessGroupedBusinesses != null) {
|
||||
// for (BusinessType businessType
|
||||
// in widget.businessGroupedBusinesses!.keys) {
|
||||
// headers.add(BusinessHeader(
|
||||
// businessType: businessType,
|
||||
// widescreen: widget.widescreen,
|
||||
// selectedBusinesses: widget.selectedBusinesses,
|
||||
// businesses: widget.businessGroupedBusinesses![businessType]!));
|
||||
// }
|
||||
// headers.sort(
|
||||
// (a, b) => a.businessType!.index.compareTo(b.businessType!.index));
|
||||
// return MultiSliver(children: headers);
|
||||
// }
|
||||
// return const Text('Error with input data!');
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// class BusinessHeader extends StatefulWidget {
|
||||
// final JobType? jobType;
|
||||
// final BusinessType? businessType;
|
||||
// final List<Business> businesses;
|
||||
// final Set<Business>? selectedBusinesses;
|
||||
// final bool widescreen;
|
||||
// final void Function()? updateSelectedBusinessesCallback;
|
||||
//
|
||||
// const BusinessHeader({
|
||||
// super.key,
|
||||
// this.jobType,
|
||||
// this.businessType,
|
||||
// required this.businesses,
|
||||
// required this.widescreen,
|
||||
// this.updateSelectedBusinessesCallback,
|
||||
// this.selectedBusinesses,
|
||||
// });
|
||||
//
|
||||
// @override
|
||||
// State<BusinessHeader> createState() => _BusinessHeaderState();
|
||||
// }
|
||||
//
|
||||
// class _BusinessHeaderState extends State<BusinessHeader> {
|
||||
// refresh() {
|
||||
// setState(() {});
|
||||
// }
|
||||
//
|
||||
// @override
|
||||
// Widget build(BuildContext context) {
|
||||
// return SliverStickyHeader(
|
||||
// header: Container(
|
||||
// height: 55.0,
|
||||
// color: Theme.of(context).colorScheme.primary,
|
||||
// padding: const EdgeInsets.symmetric(horizontal: 16.0),
|
||||
// alignment: Alignment.centerLeft,
|
||||
// child: _getHeaderRow(widget.selectedBusinesses),
|
||||
// ),
|
||||
// sliver: _getChildSliver(
|
||||
// widget.businesses, widget.widescreen, widget.selectedBusinesses),
|
||||
// );
|
||||
// }
|
||||
//
|
||||
// Widget _getHeaderRow(Set<Business>? selectedBusinesses) {
|
||||
// if (selectedBusinesses != null) {
|
||||
// return Row(
|
||||
// mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
// children: [
|
||||
// Row(
|
||||
// children: [
|
||||
// Padding(
|
||||
// padding: const EdgeInsets.only(left: 4.0, right: 12.0),
|
||||
// child: Icon(
|
||||
// widget.jobType != null
|
||||
// ? getIconFromJobType(widget.jobType!)
|
||||
// : getIconFromBusinessType(widget.businessType!),
|
||||
// color: Theme.of(context).colorScheme.onPrimary,
|
||||
// )),
|
||||
// Text(widget.jobType != null
|
||||
// ? getNameFromJobType(widget.jobType!)
|
||||
// : getNameFromBusinessType(widget.businessType!)),
|
||||
// ],
|
||||
// ),
|
||||
// Padding(
|
||||
// padding: const EdgeInsets.only(right: 12.0),
|
||||
// child: Checkbox(
|
||||
// checkColor: Theme.of(context).colorScheme.primary,
|
||||
// activeColor: Theme.of(context).colorScheme.onPrimary,
|
||||
// value: widget.selectedBusinesses!.containsAll(widget.businesses),
|
||||
// onChanged: (value) {
|
||||
// if (value!) {
|
||||
// setState(() {
|
||||
// widget.selectedBusinesses!.addAll(widget.businesses);
|
||||
// });
|
||||
// } else {
|
||||
// setState(() {
|
||||
// widget.selectedBusinesses!.removeAll(widget.businesses);
|
||||
// });
|
||||
// }
|
||||
// },
|
||||
// ),
|
||||
// ),
|
||||
// ],
|
||||
// );
|
||||
// } else {
|
||||
// return Row(
|
||||
// children: [
|
||||
// Padding(
|
||||
// padding: const EdgeInsets.only(left: 4.0, right: 12.0),
|
||||
// child: Icon(
|
||||
// widget.jobType != null
|
||||
// ? getIconFromJobType(widget.jobType!)
|
||||
// : getIconFromBusinessType(widget.businessType!),
|
||||
// color: Theme.of(context).colorScheme.onPrimary,
|
||||
// ),
|
||||
// ),
|
||||
// Text(
|
||||
// widget.jobType != null
|
||||
// ? getNameFromJobType(widget.jobType!)
|
||||
// : getNameFromBusinessType(widget.businessType!),
|
||||
// style: TextStyle(color: Theme.of(context).colorScheme.onPrimary),
|
||||
// ),
|
||||
// ],
|
||||
// );
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// Widget _getChildSliver(List<Business> businesses, bool widescreen,
|
||||
// Set<Business>? selectedBusinesses) {
|
||||
// if (widescreen) {
|
||||
// return SliverPadding(
|
||||
// padding: const EdgeInsets.all(4),
|
||||
// sliver: SliverGrid(
|
||||
// gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent(
|
||||
// mainAxisExtent: 250.0,
|
||||
// maxCrossAxisExtent: 400.0,
|
||||
// mainAxisSpacing: 4.0,
|
||||
// crossAxisSpacing: 4.0,
|
||||
// ),
|
||||
// delegate: SliverChildBuilderDelegate(
|
||||
// childCount: businesses.length,
|
||||
// (BuildContext context, int index) {
|
||||
// return BusinessCard(
|
||||
// business: businesses[index],
|
||||
// selectedBusinesses: selectedBusinesses,
|
||||
// widescreen: widescreen,
|
||||
// callback: refresh,
|
||||
// jobType: widget.jobType,
|
||||
// );
|
||||
// },
|
||||
// ),
|
||||
// ),
|
||||
// );
|
||||
// } else {
|
||||
// return SliverList(
|
||||
// delegate: SliverChildBuilderDelegate(
|
||||
// childCount: businesses.length,
|
||||
// (BuildContext context, int index) {
|
||||
// return BusinessCard(
|
||||
// business: businesses[index],
|
||||
// selectedBusinesses: selectedBusinesses,
|
||||
// widescreen: widescreen,
|
||||
// callback: refresh,
|
||||
// jobType: widget.jobType,
|
||||
// );
|
||||
// },
|
||||
// ),
|
||||
// );
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// class BusinessCard extends StatefulWidget {
|
||||
// final Business business;
|
||||
// final bool widescreen;
|
||||
// final Set<Business>? selectedBusinesses;
|
||||
// final Function callback;
|
||||
// final JobType? jobType;
|
||||
// final BusinessType? businessType;
|
||||
//
|
||||
// const BusinessCard({
|
||||
// super.key,
|
||||
// required this.business,
|
||||
// required this.widescreen,
|
||||
// required this.callback,
|
||||
// this.jobType,
|
||||
// this.businessType,
|
||||
// this.selectedBusinesses,
|
||||
// });
|
||||
//
|
||||
// @override
|
||||
// State<BusinessCard> createState() => _BusinessCardState();
|
||||
// }
|
||||
//
|
||||
// class _BusinessCardState extends State<BusinessCard> {
|
||||
// @override
|
||||
// Widget build(BuildContext context) {
|
||||
// if (widget.widescreen) {
|
||||
// return _businessTile(widget.business, widget.selectedBusinesses,
|
||||
// widget.jobType, widget.businessType);
|
||||
// } else {
|
||||
// return _businessListItem(widget.business, widget.selectedBusinesses,
|
||||
// widget.callback, widget.jobType, widget.businessType);
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// Widget _businessTile(Business business, Set<Business>? selectedBusinesses,
|
||||
// JobType? jobType, BusinessType? businessType) {
|
||||
// return MouseRegion(
|
||||
// cursor: SystemMouseCursors.click,
|
||||
// child: GestureDetector(
|
||||
// onTap: () {
|
||||
// Navigator.of(context).push(MaterialPageRoute(
|
||||
// builder: (context) => BusinessDetail(
|
||||
// id: business.id,
|
||||
// name: business.name!,
|
||||
// )));
|
||||
// },
|
||||
// child: Card(
|
||||
// clipBehavior: Clip.antiAlias,
|
||||
// child: Column(
|
||||
// crossAxisAlignment: CrossAxisAlignment.center,
|
||||
// children: [
|
||||
// _getTileRow(business, selectedBusinesses, widget.callback),
|
||||
// Padding(
|
||||
// padding: const EdgeInsets.all(8.0),
|
||||
// child: Text(
|
||||
// business.description!,
|
||||
// maxLines: selectedBusinesses != null ? 7 : 5,
|
||||
// overflow: TextOverflow.ellipsis,
|
||||
// ),
|
||||
// ),
|
||||
// const Spacer(),
|
||||
// Padding(
|
||||
// padding: const EdgeInsets.all(8.0),
|
||||
// child: selectedBusinesses == null
|
||||
// ? Row(
|
||||
// mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
||||
// children: [
|
||||
// IconButton(
|
||||
// icon: const Icon(Icons.link),
|
||||
// onPressed: () {
|
||||
// launchUrl(
|
||||
// Uri.parse('https://${business.website}'));
|
||||
// },
|
||||
// ),
|
||||
// if (business.locationName != '')
|
||||
// IconButton(
|
||||
// icon: const Icon(Icons.location_on),
|
||||
// onPressed: () {
|
||||
// launchUrl(Uri.parse(Uri.encodeFull(
|
||||
// 'https://www.google.com/maps/search/?api=1&query=${business.locationName}')));
|
||||
// },
|
||||
// ),
|
||||
// if ((business.contactPhone != null) &&
|
||||
// (business.contactPhone != ''))
|
||||
// IconButton(
|
||||
// icon: const Icon(Icons.phone),
|
||||
// onPressed: () {
|
||||
// showDialog(
|
||||
// context: context,
|
||||
// builder: (BuildContext context) {
|
||||
// return AlertDialog(
|
||||
// backgroundColor: Theme.of(context)
|
||||
// .colorScheme
|
||||
// .surface,
|
||||
// title: Text((business.contactName ==
|
||||
// null ||
|
||||
// business.contactName == '')
|
||||
// ? 'Contact ${business.name}?'
|
||||
// : 'Contact ${business.contactName}'),
|
||||
// content: Text((business.contactName ==
|
||||
// null ||
|
||||
// business.contactName == '')
|
||||
// ? 'Would you like to call or text ${business.name}?'
|
||||
// : 'Would you like to call or text ${business.contactName}?'),
|
||||
// actions: [
|
||||
// TextButton(
|
||||
// child: const Text('Text'),
|
||||
// onPressed: () {
|
||||
// launchUrl(Uri.parse(
|
||||
// 'sms:${business.contactPhone}'));
|
||||
// Navigator.of(context).pop();
|
||||
// }),
|
||||
// TextButton(
|
||||
// child: const Text('Call'),
|
||||
// onPressed: () async {
|
||||
// launchUrl(Uri.parse(
|
||||
// 'tel:${business.contactPhone}'));
|
||||
// Navigator.of(context).pop();
|
||||
// }),
|
||||
// ],
|
||||
// );
|
||||
// });
|
||||
// },
|
||||
// ),
|
||||
// if (business.contactEmail != '')
|
||||
// IconButton(
|
||||
// icon: const Icon(Icons.email),
|
||||
// onPressed: () {
|
||||
// launchUrl(Uri.parse(
|
||||
// 'mailto:${business.contactEmail}'));
|
||||
// },
|
||||
// ),
|
||||
// ],
|
||||
// )
|
||||
// : null),
|
||||
// ],
|
||||
// ),
|
||||
// ),
|
||||
// ),
|
||||
// );
|
||||
// }
|
||||
//
|
||||
// Widget _getTileRow(
|
||||
// Business business, Set<Business>? selectedBusinesses, Function callback) {
|
||||
// if (selectedBusinesses != null) {
|
||||
// return Row(
|
||||
// mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
// children: [
|
||||
// Padding(
|
||||
// padding: const EdgeInsets.all(8.0),
|
||||
// child: ClipRRect(
|
||||
// borderRadius: BorderRadius.circular(6.0),
|
||||
// child: Image.network('$apiAddress/logos/${business.id}',
|
||||
// height: 48, width: 48, errorBuilder: (BuildContext context,
|
||||
// Object exception, StackTrace? stackTrace) {
|
||||
// return Icon(
|
||||
// getIconFromBusinessType(business.type!),
|
||||
// size: 48,
|
||||
// );
|
||||
// }),
|
||||
// ),
|
||||
// ),
|
||||
// Flexible(
|
||||
// child: Padding(
|
||||
// padding: const EdgeInsets.all(8.0),
|
||||
// child: Text(
|
||||
// business.name!,
|
||||
// style:
|
||||
// const TextStyle(fontSize: 18, fontWeight: FontWeight.bold),
|
||||
// maxLines: 2,
|
||||
// overflow: TextOverflow.ellipsis,
|
||||
// ),
|
||||
// ),
|
||||
// ),
|
||||
// Padding(
|
||||
// padding: const EdgeInsets.only(right: 24.0),
|
||||
// child: _checkbox(callback, selectedBusinesses),
|
||||
// )
|
||||
// ],
|
||||
// );
|
||||
// } else {
|
||||
// return Row(
|
||||
// children: [
|
||||
// Padding(
|
||||
// padding: const EdgeInsets.all(8.0),
|
||||
// child: ClipRRect(
|
||||
// borderRadius: BorderRadius.circular(6.0),
|
||||
// child: Image.network('$apiAddress/logos/${business.id}',
|
||||
// height: 48, width: 48, errorBuilder: (BuildContext context,
|
||||
// Object exception, StackTrace? stackTrace) {
|
||||
// return Icon(getIconFromBusinessType(business.type!),
|
||||
// size: 48);
|
||||
// }),
|
||||
// )),
|
||||
// Flexible(
|
||||
// child: Padding(
|
||||
// padding: const EdgeInsets.all(8.0),
|
||||
// child: Text(
|
||||
// business.name!,
|
||||
// style:
|
||||
// const TextStyle(fontSize: 18, fontWeight: FontWeight.bold),
|
||||
// maxLines: 2,
|
||||
// overflow: TextOverflow.ellipsis,
|
||||
// ),
|
||||
// ),
|
||||
// ),
|
||||
// ],
|
||||
// );
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// Widget _businessListItem(Business business, Set<Business>? selectedBusinesses,
|
||||
// Function callback, JobType? jobType, BusinessType? businessType) {
|
||||
// return Card(
|
||||
// child: ListTile(
|
||||
// leading: ClipRRect(
|
||||
// borderRadius: BorderRadius.circular(3.0),
|
||||
// child: Image.network('$apiAddress/logos/${business.id}',
|
||||
// height: 24, width: 24, errorBuilder: (BuildContext context,
|
||||
// Object exception, StackTrace? stackTrace) {
|
||||
// return Icon(getIconFromBusinessType(business.type!));
|
||||
// })),
|
||||
// title: Text(business.name!),
|
||||
// subtitle: Text(business.description!,
|
||||
// maxLines: 1, overflow: TextOverflow.ellipsis),
|
||||
// trailing: _getCheckbox(selectedBusinesses, callback),
|
||||
// onTap: () {
|
||||
// Navigator.of(context).push(MaterialPageRoute(
|
||||
// builder: (context) => BusinessDetail(
|
||||
// id: business.id,
|
||||
// name: business.name!,
|
||||
// )));
|
||||
// },
|
||||
// ),
|
||||
// );
|
||||
// }
|
||||
//
|
||||
// Widget _checkbox(Function callback, Set<Business> selectedBusinesses) {
|
||||
// return Checkbox(
|
||||
// value: selectedBusinesses.contains(widget.business),
|
||||
// onChanged: (value) {
|
||||
// if (value!) {
|
||||
// setState(() {
|
||||
// selectedBusinesses.add(widget.business);
|
||||
// });
|
||||
// } else {
|
||||
// setState(() {
|
||||
// selectedBusinesses.remove(widget.business);
|
||||
// });
|
||||
// }
|
||||
// callback();
|
||||
// },
|
||||
// );
|
||||
// }
|
||||
//
|
||||
// Widget? _getCheckbox(Set<Business>? selectedBusinesses, Function callback) {
|
||||
// if (selectedBusinesses != null) {
|
||||
// return _checkbox(callback, selectedBusinesses);
|
||||
// } else {
|
||||
// return null;
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
|
||||
class BusinessSearchBar extends StatefulWidget {
|
||||
final String searchTextHint;
|
||||
final Widget filterIconButton;
|
||||
final void Function(String) setSearchCallback;
|
||||
|
||||
const BusinessSearchBar(
|
||||
{super.key,
|
||||
required this.setSearchCallback,
|
||||
required this.searchTextHint,
|
||||
required this.filterIconButton});
|
||||
|
||||
@override
|
||||
State<BusinessSearchBar> createState() => _BusinessSearchBarState();
|
||||
}
|
||||
|
||||
class _BusinessSearchBarState extends State<BusinessSearchBar> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return SizedBox(
|
||||
width: 450,
|
||||
height: 50,
|
||||
child: SearchBar(
|
||||
hintText: widget.searchTextHint,
|
||||
backgroundColor: WidgetStateProperty.resolveWith((notNeeded) {
|
||||
return Theme.of(context).colorScheme.surfaceContainer;
|
||||
}),
|
||||
onChanged: (query) {
|
||||
widget.setSearchCallback(query);
|
||||
},
|
||||
leading: const Padding(
|
||||
padding: EdgeInsets.only(left: 8.0),
|
||||
child: Icon(Icons.search),
|
||||
),
|
||||
trailing: [widget.filterIconButton]),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class FilterChips extends StatefulWidget {
|
||||
final Set<JobType>? selectedJobChips;
|
||||
final Set<BusinessType>? selectedBusinessChips;
|
||||
|
||||
const FilterChips(
|
||||
{super.key, this.selectedJobChips, this.selectedBusinessChips});
|
||||
|
||||
@override
|
||||
State<FilterChips> createState() => _FilterChipsState();
|
||||
}
|
||||
|
||||
class _FilterChipsState extends State<FilterChips> {
|
||||
List<Padding> filterChips() {
|
||||
List<Padding> chips = [];
|
||||
|
||||
if (widget.selectedJobChips != null) {
|
||||
for (var type in JobType.values) {
|
||||
chips.add(Padding(
|
||||
padding: const EdgeInsets.only(left: 4.0, right: 4.0),
|
||||
child: FilterChip(
|
||||
showCheckmark: false,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(20)),
|
||||
label: Text(getNameFromJobType(type)),
|
||||
selected: widget.selectedJobChips!.contains(type),
|
||||
onSelected: (bool selected) {
|
||||
setState(() {
|
||||
if (selected) {
|
||||
widget.selectedJobChips!.add(type);
|
||||
} else {
|
||||
widget.selectedJobChips!.remove(type);
|
||||
}
|
||||
});
|
||||
}),
|
||||
));
|
||||
}
|
||||
} else if (widget.selectedBusinessChips != null) {
|
||||
for (var type in BusinessType.values) {
|
||||
chips.add(Padding(
|
||||
padding: const EdgeInsets.only(left: 4.0, right: 4.0),
|
||||
child: FilterChip(
|
||||
showCheckmark: false,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(20)),
|
||||
label: Text(getNameFromBusinessType(type)),
|
||||
selected: widget.selectedBusinessChips!.contains(type),
|
||||
onSelected: (bool selected) {
|
||||
setState(() {
|
||||
if (selected) {
|
||||
widget.selectedBusinessChips!.add(type);
|
||||
} else {
|
||||
widget.selectedBusinessChips!.remove(type);
|
||||
}
|
||||
});
|
||||
}),
|
||||
));
|
||||
}
|
||||
}
|
||||
return chips;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Wrap(
|
||||
children: filterChips(),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class MainSliverAppBar extends StatefulWidget {
|
||||
final bool widescreen;
|
||||
final Widget filterIconButton;
|
||||
final void Function(String) setSearch;
|
||||
final void Function() themeCallback;
|
||||
final void Function() generatePDF;
|
||||
final void Function(bool) updateLoggedIn;
|
||||
final String searchHintText;
|
||||
|
||||
const MainSliverAppBar({
|
||||
super.key,
|
||||
required this.widescreen,
|
||||
required this.setSearch,
|
||||
required this.searchHintText,
|
||||
required this.themeCallback,
|
||||
required this.filterIconButton,
|
||||
required this.updateLoggedIn,
|
||||
required this.generatePDF,
|
||||
});
|
||||
|
||||
@override
|
||||
State<MainSliverAppBar> createState() => _MainSliverAppBarState();
|
||||
}
|
||||
|
||||
class _MainSliverAppBarState extends State<MainSliverAppBar> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return SliverAppBar(
|
||||
title: widget.widescreen
|
||||
? BusinessSearchBar(
|
||||
setSearchCallback: widget.setSearch,
|
||||
searchTextHint: widget.searchHintText,
|
||||
filterIconButton: widget.filterIconButton,
|
||||
)
|
||||
: const Text('Job Link'),
|
||||
toolbarHeight: 70,
|
||||
stretch: false,
|
||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
||||
pinned: true,
|
||||
// floating: true,
|
||||
scrolledUnderElevation: 0,
|
||||
centerTitle: !widget.widescreen,
|
||||
expandedHeight: widget.widescreen ? 70 : 120,
|
||||
bottom: _getBottom(widget.widescreen),
|
||||
leading: !widget.widescreen
|
||||
? IconButton(
|
||||
icon: Icon(getIconFromThemeMode(themeMode)),
|
||||
onPressed: () {
|
||||
setState(() {
|
||||
widget.themeCallback();
|
||||
});
|
||||
},
|
||||
)
|
||||
: null,
|
||||
actions: [
|
||||
IconButton(
|
||||
icon: const Icon(Icons.file_download_outlined),
|
||||
onPressed: widget.generatePDF,
|
||||
),
|
||||
IconButton(
|
||||
icon: const Icon(Icons.help),
|
||||
onPressed: () {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
title: const Text('About'),
|
||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
||||
content: SizedBox(
|
||||
width: 500,
|
||||
child: IntrinsicHeight(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
const Text(
|
||||
'Welcome to my FBLA 2024 Coding and Programming submission!\n\n'
|
||||
'MarinoDev Job Link aims to provide comprehensive details of businesses and community partners'
|
||||
' for Waukesha West High School\'s Career and Technical Education Department.\n\n'),
|
||||
MouseRegion(
|
||||
cursor: SystemMouseCursors.click,
|
||||
child: GestureDetector(
|
||||
child: const Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text('Git Repo:'),
|
||||
Text(
|
||||
'https://git.marinodev.com/MarinoDev/FBLA24\n',
|
||||
style: TextStyle(color: Colors.blue)),
|
||||
],
|
||||
),
|
||||
onTap: () {
|
||||
launchUrl(Uri.https('git.marinodev.com',
|
||||
'/MarinoDev/FBLA24'));
|
||||
},
|
||||
),
|
||||
),
|
||||
MouseRegion(
|
||||
cursor: SystemMouseCursors.click,
|
||||
child: GestureDetector(
|
||||
child: const Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text('Please direct any questions to'),
|
||||
Text('drake@marinodev.com',
|
||||
style: TextStyle(color: Colors.blue)),
|
||||
],
|
||||
),
|
||||
onTap: () {
|
||||
launchUrl(
|
||||
Uri.parse('mailto:drake@marinodev.com'));
|
||||
},
|
||||
),
|
||||
)
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('OK'),
|
||||
onPressed: () {
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
],
|
||||
);
|
||||
});
|
||||
},
|
||||
),
|
||||
// IconButton(
|
||||
// icon: const Icon(Icons.picture_as_pdf),
|
||||
// onPressed: () async {
|
||||
// if (!_isPreviousData) {
|
||||
// ScaffoldMessenger.of(context).clearSnackBars();
|
||||
// ScaffoldMessenger.of(context).showSnackBar(
|
||||
// const SnackBar(
|
||||
// width: 300,
|
||||
// behavior: SnackBarBehavior.floating,
|
||||
// content: Text('There is no data!'),
|
||||
// duration: Duration(seconds: 2),
|
||||
// ),
|
||||
// );
|
||||
// } else {
|
||||
// selectedDataTypesBusiness = <DataTypeBusiness>{};
|
||||
// Navigator.push(
|
||||
// context,
|
||||
// MaterialPageRoute(
|
||||
// builder: (context) => ExportData(
|
||||
// groupedBusinesses: overviewBusinesses)));
|
||||
// }
|
||||
// },
|
||||
// ),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(right: 8.0),
|
||||
child: IconButton(
|
||||
icon: loggedIn
|
||||
? const Icon(Icons.account_circle)
|
||||
: const Icon(Icons.login),
|
||||
onPressed: () {
|
||||
if (loggedIn) {
|
||||
var payload = JWT.decode(jwt).payload;
|
||||
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
||||
title: Text('Hi, ${payload['username']}!'),
|
||||
content: Text(
|
||||
'You are logged in as an admin with username ${payload['username']}.'),
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('Cancel'),
|
||||
onPressed: () {
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
TextButton(
|
||||
child: const Text('Logout'),
|
||||
onPressed: () async {
|
||||
final prefs =
|
||||
await SharedPreferences.getInstance();
|
||||
prefs.setBool('rememberMe', false);
|
||||
prefs.setString('username', '');
|
||||
prefs.setString('password', '');
|
||||
|
||||
widget.updateLoggedIn(false);
|
||||
Navigator.of(context).pop();
|
||||
}),
|
||||
],
|
||||
);
|
||||
});
|
||||
} else {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) =>
|
||||
SignInPage(refreshAccount: widget.updateLoggedIn)));
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
|
||||
PreferredSizeWidget? _getBottom(bool widescreen) {
|
||||
if (!widescreen) {
|
||||
return PreferredSize(
|
||||
preferredSize: const Size.fromHeight(0),
|
||||
child: SizedBox(
|
||||
height: 70,
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(10),
|
||||
child: BusinessSearchBar(
|
||||
filterIconButton: widget.filterIconButton,
|
||||
setSearchCallback: widget.setSearch,
|
||||
searchTextHint: widget.searchHintText,
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
}
|
||||
@ -73,6 +73,8 @@ flutter:
|
||||
# To add assets to your application, add an assets section, like this:
|
||||
assets:
|
||||
- assets/mdev_triangle_loading.riv
|
||||
- assets/MarinoDev.svg
|
||||
- assets/Triangle256.png
|
||||
# - images/a_dot_burr.jpeg
|
||||
# - images/a_dot_ham.jpeg
|
||||
|
||||
|
||||
Loading…
Reference in New Issue
Block a user