Major Job Listings refactor
This commit is contained in:
parent
9076765aae
commit
32e3cc574c
@ -65,8 +65,8 @@ class Business {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class JobListing {
|
class JobListing {
|
||||||
String? id;
|
int? id;
|
||||||
String? businessId;
|
int? businessId;
|
||||||
String name;
|
String name;
|
||||||
String description;
|
String description;
|
||||||
JobType type;
|
JobType type;
|
||||||
@ -188,6 +188,26 @@ void main() async {
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
app.get('/fbla-api/businessdata/businessnames', (Request request) async {
|
||||||
|
print('business names request received');
|
||||||
|
|
||||||
|
var postgresResult = (await postgres.query('''
|
||||||
|
SELECT json_agg(
|
||||||
|
json_build_object(
|
||||||
|
'id', id,
|
||||||
|
'name', name
|
||||||
|
)
|
||||||
|
) FROM public.businesses
|
||||||
|
'''))[0][0];
|
||||||
|
|
||||||
|
return Response.ok(
|
||||||
|
json.encode(postgresResult),
|
||||||
|
headers: {
|
||||||
|
'Access-Control-Allow-Origin': '*',
|
||||||
|
'Content-Type': 'text/plain'
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
app.get('/fbla-api/businessdata/business/<business>',
|
app.get('/fbla-api/businessdata/business/<business>',
|
||||||
(Request request, String business) async {
|
(Request request, String business) async {
|
||||||
print('idividual business data request received');
|
print('idividual business data request received');
|
||||||
@ -209,6 +229,7 @@ void main() async {
|
|||||||
json_agg(
|
json_agg(
|
||||||
json_build_object(
|
json_build_object(
|
||||||
'id', l.id,
|
'id', l.id,
|
||||||
|
'businessId', l."businessId",
|
||||||
'name', l.name,
|
'name', l.name,
|
||||||
'description', l.description,
|
'description', l.description,
|
||||||
'type', l.type,
|
'type', l.type,
|
||||||
@ -231,11 +252,85 @@ void main() async {
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
app.get('/fbla-api/businessdata/businesses', (Request request) async {
|
||||||
|
print('list of business data request received');
|
||||||
|
|
||||||
|
if (request.url.queryParameters['businesses'] == null) {
|
||||||
|
return Response.badRequest(
|
||||||
|
body: 'query \'businesses\' required',
|
||||||
|
headers: {
|
||||||
|
'Access-Control-Allow-Origin': '*',
|
||||||
|
'Content-Type': 'text/plain'
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
var filters = request.url.queryParameters['businesses']!.split(',');
|
||||||
|
|
||||||
|
var result = (await postgres.query('''
|
||||||
|
SELECT
|
||||||
|
json_build_object(
|
||||||
|
'id', b.id,
|
||||||
|
'name', b.name,
|
||||||
|
'description', b.description,
|
||||||
|
'website', b.website,
|
||||||
|
'contactName', b."contactName",
|
||||||
|
'contactEmail', b."contactEmail",
|
||||||
|
'contactPhone', b."contactPhone",
|
||||||
|
'notes', b.notes,
|
||||||
|
'locationName', b."locationName",
|
||||||
|
'locationAddress', b."locationAddress",
|
||||||
|
'listings',
|
||||||
|
json_agg(
|
||||||
|
json_build_object(
|
||||||
|
'id', l.id,
|
||||||
|
'businessId', l."businessId",
|
||||||
|
'name', l.name,
|
||||||
|
'description', l.description,
|
||||||
|
'type', l.type,
|
||||||
|
'wage', l.wage,
|
||||||
|
'link', l.link
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
FROM businesses b
|
||||||
|
LEFT JOIN listings l ON b.id = l."businessId"
|
||||||
|
WHERE b.id IN ${'$filters'.replaceAll('[', '(').replaceAll(']', ')')}
|
||||||
|
GROUP BY b.id;
|
||||||
|
'''));
|
||||||
|
|
||||||
|
var output = result.map((element) => element[0]).toList();
|
||||||
|
|
||||||
|
return Response.ok(
|
||||||
|
json.encode(output),
|
||||||
|
headers: {
|
||||||
|
'Access-Control-Allow-Origin': '*',
|
||||||
|
'Content-Type': 'text/plain'
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
app.get('/fbla-api/businessdata', (Request request) async {
|
app.get('/fbla-api/businessdata', (Request request) async {
|
||||||
print('business data request received');
|
print('business data request received');
|
||||||
final output = await fetchBusinessData();
|
final result = await postgres.query('''
|
||||||
|
SELECT json_agg(
|
||||||
|
json_build_object(
|
||||||
|
'id', id,
|
||||||
|
'name', name,
|
||||||
|
'description', description,
|
||||||
|
'type', type,
|
||||||
|
'website', website,
|
||||||
|
'contactName', "contactName",
|
||||||
|
'contactEmail', "contactEmail",
|
||||||
|
'contactPhone', "contactPhone",
|
||||||
|
'notes', notes,
|
||||||
|
'locationName', "locationName",
|
||||||
|
'locationAddress', "locationAddress"
|
||||||
|
)
|
||||||
|
) FROM businesses
|
||||||
|
''');
|
||||||
|
|
||||||
|
var encoded = json.encode(result[0][0]);
|
||||||
return Response.ok(
|
return Response.ok(
|
||||||
output.toString(),
|
encoded,
|
||||||
headers: {
|
headers: {
|
||||||
'Access-Control-Allow-Origin': '*',
|
'Access-Control-Allow-Origin': '*',
|
||||||
'Content-Type': 'text/plain'
|
'Content-Type': 'text/plain'
|
||||||
|
|||||||
@ -5,9 +5,9 @@ import 'dart:io';
|
|||||||
import 'package:fbla_ui/shared.dart';
|
import 'package:fbla_ui/shared.dart';
|
||||||
import 'package:http/http.dart' as http;
|
import 'package:http/http.dart' as http;
|
||||||
|
|
||||||
var apiAddress = 'https://homelab.marinodev.com/fbla-api';
|
// var apiAddress = 'https://homelab.marinodev.com/fbla-api';
|
||||||
|
var apiAddress = 'http://192.168.0.114:8000/fbla-api';
|
||||||
var client = http.Client();
|
var client = http.Client();
|
||||||
// var apiAddress = '192.168.0.114:8000';
|
|
||||||
|
|
||||||
Future fetchBusinessData() async {
|
Future fetchBusinessData() async {
|
||||||
try {
|
try {
|
||||||
@ -29,11 +29,36 @@ Future fetchBusinessData() async {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future fetchBusinessDataOverview() async {
|
Future fetchBusinessNames() async {
|
||||||
try {
|
try {
|
||||||
var response = await http
|
var response = await http
|
||||||
.get(Uri.parse('$apiAddress/businessdata/overview'))
|
.get(Uri.parse('$apiAddress/businessdata/businessnames'))
|
||||||
.timeout(const Duration(seconds: 20));
|
.timeout(const Duration(seconds: 20));
|
||||||
|
if (response.statusCode == 200) {
|
||||||
|
List<Map<String, dynamic>> decodedResponse =
|
||||||
|
json.decode(response.body).cast<Map<String, dynamic>>();
|
||||||
|
|
||||||
|
return decodedResponse;
|
||||||
|
} 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 fetchBusinessDataOverview({List<JobType>? typeFilters}) async {
|
||||||
|
try {
|
||||||
|
String? typeString =
|
||||||
|
typeFilters?.map((jobType) => jobType.name).toList().join(',');
|
||||||
|
Uri uri =
|
||||||
|
Uri.parse('$apiAddress/businessdata/overview?filters=$typeString');
|
||||||
|
if (typeFilters == null || typeFilters.isEmpty) {
|
||||||
|
uri = Uri.parse('$apiAddress/businessdata/overview');
|
||||||
|
}
|
||||||
|
var response = await http.get(uri).timeout(const Duration(seconds: 20));
|
||||||
if (response.statusCode == 200) {
|
if (response.statusCode == 200) {
|
||||||
var decodedResponse = json.decode(response.body);
|
var decodedResponse = json.decode(response.body);
|
||||||
Map<JobType, List<Business>> groupedBusinesses = {};
|
Map<JobType, List<Business>> groupedBusinesses = {};
|
||||||
@ -49,13 +74,6 @@ Future fetchBusinessDataOverview() async {
|
|||||||
groupedBusinesses
|
groupedBusinesses
|
||||||
.addAll({JobType.values.byName(stringType): businesses});
|
.addAll({JobType.values.byName(stringType): businesses});
|
||||||
}
|
}
|
||||||
|
|
||||||
// for (JobType type in decodedResponse.keys) {
|
|
||||||
// groupedBusinesses.addAll({
|
|
||||||
// JobType.values.byName(decodedResponse[i]):
|
|
||||||
// decodedResponse.map((json) => Business.fromJson(json)).toList()
|
|
||||||
// });
|
|
||||||
// }
|
|
||||||
return groupedBusinesses;
|
return groupedBusinesses;
|
||||||
} else {
|
} else {
|
||||||
return 'Error ${response.statusCode}! Please try again later!';
|
return 'Error ${response.statusCode}! Please try again later!';
|
||||||
@ -67,6 +85,30 @@ Future fetchBusinessDataOverview() async {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future fetchBusinesses(List<int> ids) async {
|
||||||
|
try {
|
||||||
|
var response = await http
|
||||||
|
.get(Uri.parse(
|
||||||
|
'$apiAddress/businessdata/businesses?businesses=${ids.join(',')}'))
|
||||||
|
.timeout(const Duration(seconds: 20));
|
||||||
|
if (response.statusCode == 200) {
|
||||||
|
List<dynamic> decodedResponse = json.decode(response.body);
|
||||||
|
|
||||||
|
List<Business> businesses = decodedResponse
|
||||||
|
.map<Business>((json) => Business.fromJson(json))
|
||||||
|
.toList();
|
||||||
|
|
||||||
|
return businesses;
|
||||||
|
} 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 fetchBusiness(int id) async {
|
Future fetchBusiness(int id) async {
|
||||||
try {
|
try {
|
||||||
var response = await http
|
var response = await http
|
||||||
@ -87,7 +129,7 @@ Future fetchBusiness(int id) async {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future createBusiness(Business business, String jwt) async {
|
Future createBusiness(Business business) async {
|
||||||
var json = '''
|
var json = '''
|
||||||
{
|
{
|
||||||
"id": ${business.id},
|
"id": ${business.id},
|
||||||
@ -117,7 +159,33 @@ Future createBusiness(Business business, String jwt) async {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future deleteBusiness(int id, String jwt) async {
|
Future createListing(JobListing listing) async {
|
||||||
|
var json = '''
|
||||||
|
{
|
||||||
|
"id": ${listing.id},
|
||||||
|
"businessId": ${listing.businessId},
|
||||||
|
"name": "${listing.name}",
|
||||||
|
"description": "${listing.description}",
|
||||||
|
"wage": "${listing.wage}",
|
||||||
|
"link": "${listing.link}"
|
||||||
|
}
|
||||||
|
''';
|
||||||
|
|
||||||
|
try {
|
||||||
|
var response = await http.post(Uri.parse('$apiAddress/createbusiness'),
|
||||||
|
body: json,
|
||||||
|
headers: {'Authorization': jwt}).timeout(const Duration(seconds: 20));
|
||||||
|
if (response.statusCode != 200) {
|
||||||
|
return response.body;
|
||||||
|
}
|
||||||
|
} on TimeoutException {
|
||||||
|
return 'Unable to connect to server (timeout). Please try again later';
|
||||||
|
} on SocketException {
|
||||||
|
return 'Unable to connect to server (socket exception). Please check your internet connection.';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future deleteBusiness(int id) async {
|
||||||
var json = '''
|
var json = '''
|
||||||
{
|
{
|
||||||
"id": $id
|
"id": $id
|
||||||
@ -138,7 +206,28 @@ Future deleteBusiness(int id, String jwt) async {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future editBusiness(Business business, String jwt) async {
|
Future deleteListing(int id) async {
|
||||||
|
var json = '''
|
||||||
|
{
|
||||||
|
"id": $id
|
||||||
|
}
|
||||||
|
''';
|
||||||
|
|
||||||
|
try {
|
||||||
|
var response = await http.post(Uri.parse('$apiAddress/deletelisting'),
|
||||||
|
body: json,
|
||||||
|
headers: {'Authorization': jwt}).timeout(const Duration(seconds: 20));
|
||||||
|
if (response.statusCode != 200) {
|
||||||
|
return response.body;
|
||||||
|
}
|
||||||
|
} on TimeoutException {
|
||||||
|
return 'Unable to connect to server (timeout). Please try again later';
|
||||||
|
} on SocketException {
|
||||||
|
return 'Unable to connect to server (socket exception). Please check your internet connection.';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future editBusiness(Business business) async {
|
||||||
var json = '''
|
var json = '''
|
||||||
{
|
{
|
||||||
"id": ${business.id},
|
"id": ${business.id},
|
||||||
@ -167,6 +256,32 @@ Future editBusiness(Business business, String jwt) async {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future editListing(JobListing listing) async {
|
||||||
|
var json = '''
|
||||||
|
{
|
||||||
|
"id": ${listing.id},
|
||||||
|
"businessId": ${listing.businessId},
|
||||||
|
"name": "${listing.name}",
|
||||||
|
"description": "${listing.description}",
|
||||||
|
"type": "${listing.type.name}",
|
||||||
|
"wage": "${listing.wage}",
|
||||||
|
"link": "${listing.link}"
|
||||||
|
}
|
||||||
|
''';
|
||||||
|
try {
|
||||||
|
var response = await http.post(Uri.parse('$apiAddress/editlisting'),
|
||||||
|
body: json,
|
||||||
|
headers: {'Authorization': jwt}).timeout(const Duration(seconds: 20));
|
||||||
|
if (response.statusCode != 200) {
|
||||||
|
return response.body;
|
||||||
|
}
|
||||||
|
} on TimeoutException {
|
||||||
|
return 'Unable to connect to server (timeout). Please try again later';
|
||||||
|
} on SocketException {
|
||||||
|
return 'Unable to connect to server (socket exception). Please check your internet connection.';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Future signIn(String username, String password) async {
|
Future signIn(String username, String password) async {
|
||||||
var json = '''
|
var json = '''
|
||||||
{
|
{
|
||||||
|
|||||||
@ -10,10 +10,8 @@ import 'package:rive/rive.dart';
|
|||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
import 'package:url_launcher/url_launcher.dart';
|
import 'package:url_launcher/url_launcher.dart';
|
||||||
|
|
||||||
typedef Callback = void Function();
|
|
||||||
|
|
||||||
class Home extends StatefulWidget {
|
class Home extends StatefulWidget {
|
||||||
final Callback themeCallback;
|
final void Function() themeCallback;
|
||||||
|
|
||||||
const Home({super.key, required this.themeCallback});
|
const Home({super.key, required this.themeCallback});
|
||||||
|
|
||||||
@ -25,6 +23,51 @@ class _HomeState extends State<Home> {
|
|||||||
late Future refreshBusinessDataOverviewFuture;
|
late Future refreshBusinessDataOverviewFuture;
|
||||||
bool _isPreviousData = false;
|
bool _isPreviousData = false;
|
||||||
late Map<JobType, List<Business>> overviewBusinesses;
|
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(() {
|
||||||
|
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
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
@ -61,26 +104,27 @@ class _HomeState extends State<Home> {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
bool widescreen = MediaQuery.sizeOf(context).width >= 1000;
|
bool widescreen = MediaQuery.sizeOf(context).width >= 1000;
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
|
// backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
||||||
floatingActionButton: _getFAB(),
|
floatingActionButton: _getFAB(),
|
||||||
body: RefreshIndicator(
|
body: RefreshIndicator(
|
||||||
edgeOffset: 120,
|
edgeOffset: 120,
|
||||||
onRefresh: () async {
|
onRefresh: () async {
|
||||||
var refreshedData = fetchBusinessDataOverview();
|
_updateOverviewBusinesses();
|
||||||
await refreshedData;
|
|
||||||
setState(() {
|
|
||||||
refreshBusinessDataOverviewFuture = refreshedData;
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
child: CustomScrollView(
|
child: CustomScrollView(
|
||||||
slivers: [
|
slivers: [
|
||||||
SliverAppBar(
|
SliverAppBar(
|
||||||
title: widescreen ? _searchBar() : const Text('Job Link'),
|
title: widescreen
|
||||||
|
? BusinessSearchBar(
|
||||||
|
filters: jobTypeFilters,
|
||||||
|
setFiltersCallback: _setFilters,
|
||||||
|
setSearchCallback: _setSearch)
|
||||||
|
: const Text('Job Link'),
|
||||||
toolbarHeight: 70,
|
toolbarHeight: 70,
|
||||||
pinned: true,
|
pinned: true,
|
||||||
scrolledUnderElevation: 0,
|
scrolledUnderElevation: 0,
|
||||||
centerTitle: true,
|
centerTitle: true,
|
||||||
expandedHeight: widescreen ? 70 : 120,
|
expandedHeight: widescreen ? 70 : 120,
|
||||||
backgroundColor: Theme.of(context).colorScheme.surface,
|
|
||||||
bottom: _getBottom(),
|
bottom: _getBottom(),
|
||||||
leading: IconButton(
|
leading: IconButton(
|
||||||
icon: getIconFromThemeMode(themeMode),
|
icon: getIconFromThemeMode(themeMode),
|
||||||
@ -100,7 +144,7 @@ class _HomeState extends State<Home> {
|
|||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
title: const Text('About'),
|
title: const Text('About'),
|
||||||
backgroundColor:
|
backgroundColor:
|
||||||
Theme.of(context).colorScheme.background,
|
Theme.of(context).colorScheme.surface,
|
||||||
content: SizedBox(
|
content: SizedBox(
|
||||||
width: 500,
|
width: 500,
|
||||||
child: IntrinsicHeight(
|
child: IntrinsicHeight(
|
||||||
@ -181,7 +225,7 @@ class _HomeState extends State<Home> {
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
selectedDataTypes = <DataType>{};
|
selectedDataTypesBusiness = <DataTypeBusiness>{};
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(
|
MaterialPageRoute(
|
||||||
@ -205,7 +249,7 @@ class _HomeState extends State<Home> {
|
|||||||
builder: (BuildContext context) {
|
builder: (BuildContext context) {
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
backgroundColor:
|
backgroundColor:
|
||||||
Theme.of(context).colorScheme.background,
|
Theme.of(context).colorScheme.surface,
|
||||||
title: Text('Hi, ${payload['username']}!'),
|
title: Text('Hi, ${payload['username']}!'),
|
||||||
content: Text(
|
content: Text(
|
||||||
'You are logged in as an admin with username ${payload['username']}.'),
|
'You are logged in as an admin with username ${payload['username']}.'),
|
||||||
@ -263,12 +307,7 @@ class _HomeState extends State<Home> {
|
|||||||
child: FilledButton(
|
child: FilledButton(
|
||||||
child: const Text('Retry'),
|
child: const Text('Retry'),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
var refreshedData =
|
_updateOverviewBusinesses();
|
||||||
fetchBusinessDataOverview();
|
|
||||||
setState(() {
|
|
||||||
refreshBusinessDataOverviewFuture =
|
|
||||||
refreshedData;
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -280,7 +319,8 @@ class _HomeState extends State<Home> {
|
|||||||
_isPreviousData = true;
|
_isPreviousData = true;
|
||||||
|
|
||||||
return BusinessDisplayPanel(
|
return BusinessDisplayPanel(
|
||||||
groupedBusinesses: overviewBusinesses,
|
groupedBusinesses:
|
||||||
|
_filterBySearch(overviewBusinesses),
|
||||||
widescreen: widescreen,
|
widescreen: widescreen,
|
||||||
selectable: false);
|
selectable: false);
|
||||||
} else if (snapshot.hasError) {
|
} else if (snapshot.hasError) {
|
||||||
@ -295,7 +335,8 @@ class _HomeState extends State<Home> {
|
|||||||
ConnectionState.waiting) {
|
ConnectionState.waiting) {
|
||||||
if (_isPreviousData) {
|
if (_isPreviousData) {
|
||||||
return BusinessDisplayPanel(
|
return BusinessDisplayPanel(
|
||||||
groupedBusinesses: overviewBusinesses,
|
groupedBusinesses:
|
||||||
|
_filterBySearch(overviewBusinesses),
|
||||||
widescreen: widescreen,
|
widescreen: widescreen,
|
||||||
selectable: false);
|
selectable: false);
|
||||||
} else {
|
} else {
|
||||||
@ -303,15 +344,11 @@ class _HomeState extends State<Home> {
|
|||||||
child: Container(
|
child: Container(
|
||||||
padding: const EdgeInsets.all(8.0),
|
padding: const EdgeInsets.all(8.0),
|
||||||
alignment: Alignment.center,
|
alignment: Alignment.center,
|
||||||
// child: const CircularProgressIndicator(),
|
|
||||||
child: const SizedBox(
|
child: const SizedBox(
|
||||||
width: 75,
|
width: 75,
|
||||||
height: 75,
|
height: 75,
|
||||||
child: RiveAnimation.asset(
|
child: RiveAnimation.asset(
|
||||||
'assets/mdev_triangle_loading.riv'),
|
'assets/mdev_triangle_loading.riv'),
|
||||||
// child: RiveAnimation.file(
|
|
||||||
// 'assets/mdev_triangle_loading.riv',
|
|
||||||
// ),
|
|
||||||
),
|
),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
@ -353,85 +390,6 @@ class _HomeState extends State<Home> {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _searchBar() {
|
|
||||||
return SizedBox(
|
|
||||||
width: 800,
|
|
||||||
height: 50,
|
|
||||||
child: TextField(
|
|
||||||
onChanged: (query) {
|
|
||||||
setState(() {
|
|
||||||
searchFilter = query;
|
|
||||||
});
|
|
||||||
},
|
|
||||||
decoration: InputDecoration(
|
|
||||||
labelText: 'Search',
|
|
||||||
hintText: 'Search',
|
|
||||||
prefixIcon: const Padding(
|
|
||||||
padding: EdgeInsets.only(left: 8.0),
|
|
||||||
child: Icon(Icons.search),
|
|
||||||
),
|
|
||||||
border: const OutlineInputBorder(
|
|
||||||
borderRadius: BorderRadius.all(Radius.circular(25.0)),
|
|
||||||
),
|
|
||||||
suffixIcon: Padding(
|
|
||||||
padding: const EdgeInsets.only(right: 8.0),
|
|
||||||
child: IconButton(
|
|
||||||
tooltip: 'Filters',
|
|
||||||
icon: Icon(Icons.filter_list,
|
|
||||||
color: isFiltered
|
|
||||||
? Theme.of(context).colorScheme.primary
|
|
||||||
: Theme.of(context).colorScheme.onBackground),
|
|
||||||
onPressed: () {
|
|
||||||
showDialog(
|
|
||||||
context: context,
|
|
||||||
builder: (BuildContext context) {
|
|
||||||
return AlertDialog(
|
|
||||||
// DO NOT MOVE TO SEPARATE WIDGET, setState is needed in main tree
|
|
||||||
backgroundColor:
|
|
||||||
Theme.of(context).colorScheme.background,
|
|
||||||
title: const Text('Filter Options'),
|
|
||||||
content: const FilterChips(),
|
|
||||||
actions: [
|
|
||||||
TextButton(
|
|
||||||
child: const Text('Reset'),
|
|
||||||
onPressed: () {
|
|
||||||
setState(() {
|
|
||||||
filters = <BusinessType>{};
|
|
||||||
selectedChips = <BusinessType>{};
|
|
||||||
isFiltered = false;
|
|
||||||
});
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
}),
|
|
||||||
TextButton(
|
|
||||||
child: const Text('Cancel'),
|
|
||||||
onPressed: () {
|
|
||||||
selectedChips = Set.from(filters);
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
}),
|
|
||||||
TextButton(
|
|
||||||
child: const Text('Apply'),
|
|
||||||
onPressed: () {
|
|
||||||
setState(() {
|
|
||||||
filters = Set.from(selectedChips);
|
|
||||||
if (filters.isNotEmpty) {
|
|
||||||
isFiltered = true;
|
|
||||||
} else {
|
|
||||||
isFiltered = false;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
}),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
PreferredSizeWidget? _getBottom() {
|
PreferredSizeWidget? _getBottom() {
|
||||||
if (MediaQuery.sizeOf(context).width <= 1000) {
|
if (MediaQuery.sizeOf(context).width <= 1000) {
|
||||||
return PreferredSize(
|
return PreferredSize(
|
||||||
@ -441,7 +399,10 @@ class _HomeState extends State<Home> {
|
|||||||
height: 70,
|
height: 70,
|
||||||
child: Padding(
|
child: Padding(
|
||||||
padding: const EdgeInsets.all(10),
|
padding: const EdgeInsets.all(10),
|
||||||
child: _searchBar(),
|
child: BusinessSearchBar(
|
||||||
|
filters: jobTypeFilters,
|
||||||
|
setFiltersCallback: _setFilters,
|
||||||
|
setSearchCallback: _setSearch),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@ -69,14 +69,14 @@ class _MainAppState extends State<MainApp> {
|
|||||||
return MaterialApp(
|
return MaterialApp(
|
||||||
title: 'Job Link',
|
title: 'Job Link',
|
||||||
themeMode: themeMode,
|
themeMode: themeMode,
|
||||||
// themeMode: ThemeMode.light,
|
|
||||||
darkTheme: ThemeData(
|
darkTheme: ThemeData(
|
||||||
colorScheme: ColorScheme.dark(
|
colorScheme: ColorScheme.dark(
|
||||||
brightness: Brightness.dark,
|
brightness: Brightness.dark,
|
||||||
primary: Colors.blue,
|
primary: Colors.blue,
|
||||||
onPrimary: Colors.white,
|
onPrimary: Colors.white,
|
||||||
secondary: Colors.blue.shade900,
|
secondary: Colors.blue.shade900,
|
||||||
background: const Color.fromARGB(255, 31, 31, 31),
|
surface: const Color.fromARGB(255, 31, 31, 31),
|
||||||
|
surfaceContainer: const Color.fromARGB(255, 40, 40, 40),
|
||||||
tertiary: Colors.green.shade900,
|
tertiary: Colors.green.shade900,
|
||||||
),
|
),
|
||||||
iconTheme: const IconThemeData(color: Colors.white),
|
iconTheme: const IconThemeData(color: Colors.white),
|
||||||
@ -89,7 +89,8 @@ class _MainAppState extends State<MainApp> {
|
|||||||
primary: Colors.blue,
|
primary: Colors.blue,
|
||||||
onPrimary: Colors.white,
|
onPrimary: Colors.white,
|
||||||
secondary: Colors.blue.shade200,
|
secondary: Colors.blue.shade200,
|
||||||
background: Colors.grey.shade300,
|
surface: Colors.grey.shade200,
|
||||||
|
surfaceContainer: Colors.grey.shade300,
|
||||||
tertiary: Colors.green,
|
tertiary: Colors.green,
|
||||||
),
|
),
|
||||||
iconTheme: const IconThemeData(color: Colors.black),
|
iconTheme: const IconThemeData(color: Colors.black),
|
||||||
|
|||||||
@ -1,6 +1,8 @@
|
|||||||
import 'package:fbla_ui/api_logic.dart';
|
import 'package:fbla_ui/api_logic.dart';
|
||||||
import 'package:fbla_ui/main.dart';
|
import 'package:fbla_ui/main.dart';
|
||||||
import 'package:fbla_ui/pages/create_edit_business.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/pages/signin_page.dart';
|
||||||
import 'package:fbla_ui/shared.dart';
|
import 'package:fbla_ui/shared.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
@ -111,7 +113,9 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
return ListView(
|
return ListView(
|
||||||
children: [
|
children: [
|
||||||
// Title, logo, desc, website
|
// Title, logo, desc, website
|
||||||
Card(
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 4.0),
|
||||||
|
child: Card(
|
||||||
clipBehavior: Clip.antiAlias,
|
clipBehavior: Clip.antiAlias,
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
@ -138,63 +142,33 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
ListTile(
|
ListTile(
|
||||||
leading: const Icon(Icons.link),
|
leading: const Icon(Icons.link),
|
||||||
title: const Text('Website'),
|
title: const Text('Website'),
|
||||||
subtitle: Text(business.website!,
|
subtitle: Text(
|
||||||
|
business.website
|
||||||
|
.replaceAll('https://', '')
|
||||||
|
.replaceAll('http://', '')
|
||||||
|
.replaceAll('www.', ''),
|
||||||
style: const TextStyle(color: Colors.blue)),
|
style: const TextStyle(color: Colors.blue)),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
launchUrl(Uri.parse('https://${business.website}'));
|
launchUrl(Uri.parse(business.website));
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
),
|
||||||
// Available positions
|
// Available positions
|
||||||
Card(
|
Card(
|
||||||
child: Padding(
|
clipBehavior: Clip.antiAlias,
|
||||||
padding: const EdgeInsets.only(left: 16.0, top: 8.0),
|
|
||||||
child:
|
child:
|
||||||
Column(crossAxisAlignment: CrossAxisAlignment.start, children: [
|
Column(crossAxisAlignment: CrossAxisAlignment.start, children: [
|
||||||
const Text(
|
Padding(
|
||||||
'Available Postitions',
|
padding: const EdgeInsets.only(left: 16, top: 4),
|
||||||
style: TextStyle(fontSize: 20, fontWeight: FontWeight.bold),
|
child: _GetListingsTitle(business)),
|
||||||
),
|
_JobList(business: business)
|
||||||
// Container(
|
|
||||||
// height: 400,
|
|
||||||
// width: 300,
|
|
||||||
ListView(
|
|
||||||
scrollDirection: Axis.vertical,
|
|
||||||
shrinkWrap: true,
|
|
||||||
children: [
|
|
||||||
ListTile(
|
|
||||||
title: Text('Postition 1'),
|
|
||||||
leading: Icon(Icons.work),
|
|
||||||
onTap: () {
|
|
||||||
// launchUrl(Uri.parse(''));
|
|
||||||
},
|
|
||||||
),
|
|
||||||
ListTile(
|
|
||||||
title: Text('Postition 2'),
|
|
||||||
leading: Icon(Icons.work),
|
|
||||||
onTap: () {
|
|
||||||
// launchUrl(Uri.parse(''));
|
|
||||||
},
|
|
||||||
),
|
|
||||||
ListTile(
|
|
||||||
title: Text('Postition 3'),
|
|
||||||
leading: Icon(Icons.work),
|
|
||||||
onTap: () {
|
|
||||||
// launchUrl(Uri.parse(''));
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
]),
|
]),
|
||||||
),
|
),
|
||||||
),
|
|
||||||
// Contact info
|
// Contact info
|
||||||
Visibility(
|
Card(
|
||||||
visible:
|
|
||||||
(business.contactEmail != null || business.contactPhone != null),
|
|
||||||
child: Card(
|
|
||||||
clipBehavior: Clip.antiAlias,
|
clipBehavior: Clip.antiAlias,
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
@ -203,7 +177,7 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.only(left: 16.0, top: 8.0),
|
padding: const EdgeInsets.only(left: 16.0, top: 8.0),
|
||||||
child: Text(
|
child: Text(
|
||||||
business.contactName ?? 'Contact ${business.name}',
|
business.contactName!,
|
||||||
textAlign: TextAlign.left,
|
textAlign: TextAlign.left,
|
||||||
style: const TextStyle(
|
style: const TextStyle(
|
||||||
fontSize: 20, fontWeight: FontWeight.bold),
|
fontSize: 20, fontWeight: FontWeight.bold),
|
||||||
@ -214,7 +188,7 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
Visibility(
|
Visibility(
|
||||||
visible: business.contactPhone != null,
|
visible: business.contactPhone != null,
|
||||||
child: ListTile(
|
child: ListTile(
|
||||||
leading: Icon(Icons.phone),
|
leading: const Icon(Icons.phone),
|
||||||
title: Text(business.contactPhone!),
|
title: Text(business.contactPhone!),
|
||||||
// maybe replace ! with ?? ''. same is true for below
|
// maybe replace ! with ?? ''. same is true for below
|
||||||
onTap: () {
|
onTap: () {
|
||||||
@ -223,13 +197,10 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
builder: (BuildContext context) {
|
builder: (BuildContext context) {
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
backgroundColor:
|
backgroundColor:
|
||||||
Theme.of(context).colorScheme.background,
|
Theme.of(context).colorScheme.surface,
|
||||||
title: Text(business.contactName!.isEmpty
|
title: Text('Contact ${business.contactName}'),
|
||||||
? 'Contact ${business.name}?'
|
content: Text(
|
||||||
: 'Contact ${business.contactName}'),
|
'Would you like to call or text ${business.contactName}?'),
|
||||||
content: Text(business.contactName!.isEmpty
|
|
||||||
? 'Would you like to call or text ${business.name}?'
|
|
||||||
: 'Would you like to call or text ${business.contactName}?'),
|
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(
|
TextButton(
|
||||||
child: const Text('Text'),
|
child: const Text('Text'),
|
||||||
@ -251,27 +222,23 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Visibility(
|
ListTile(
|
||||||
visible: business.contactEmail != null,
|
|
||||||
child: ListTile(
|
|
||||||
leading: const Icon(Icons.email),
|
leading: const Icon(Icons.email),
|
||||||
title: Text(business.contactEmail!),
|
title: Text(business.contactEmail),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
launchUrl(Uri.parse('mailto:${business.contactEmail}'));
|
launchUrl(Uri.parse('mailto:${business.contactEmail}'));
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
|
||||||
// Location
|
// Location
|
||||||
Visibility(
|
Visibility(
|
||||||
child: Card(
|
child: Card(
|
||||||
clipBehavior: Clip.antiAlias,
|
clipBehavior: Clip.antiAlias,
|
||||||
child: ListTile(
|
child: ListTile(
|
||||||
leading: const Icon(Icons.location_on),
|
leading: const Icon(Icons.location_on),
|
||||||
title: Text(business.locationName!),
|
title: Text(business.locationName),
|
||||||
subtitle: Text(business.locationAddress!),
|
subtitle: Text(business.locationAddress!),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
launchUrl(Uri.parse(Uri.encodeFull(
|
launchUrl(Uri.parse(Uri.encodeFull(
|
||||||
@ -282,12 +249,12 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
),
|
),
|
||||||
// Notes
|
// Notes
|
||||||
Visibility(
|
Visibility(
|
||||||
visible: business.notes != null,
|
visible: business.notes != null && business.notes != '',
|
||||||
child: Card(
|
child: Card(
|
||||||
child: ListTile(
|
child: ListTile(
|
||||||
leading: const Icon(Icons.notes),
|
leading: const Icon(Icons.notes),
|
||||||
title: const Text(
|
title: const Text(
|
||||||
'Additional Notes:',
|
'Additional Notes',
|
||||||
style: TextStyle(fontSize: 20, fontWeight: FontWeight.bold),
|
style: TextStyle(fontSize: 20, fontWeight: FontWeight.bold),
|
||||||
),
|
),
|
||||||
subtitle: Text(business.notes!),
|
subtitle: Text(business.notes!),
|
||||||
@ -318,7 +285,7 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
context: context,
|
context: context,
|
||||||
builder: (BuildContext context) {
|
builder: (BuildContext context) {
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
backgroundColor: Theme.of(context).colorScheme.background,
|
backgroundColor: Theme.of(context).colorScheme.surface,
|
||||||
title: const Text('Are You Sure?'),
|
title: const Text('Are You Sure?'),
|
||||||
content:
|
content:
|
||||||
Text('This will permanently delete ${business.name}.'),
|
Text('This will permanently delete ${business.name}.'),
|
||||||
@ -332,7 +299,7 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
child: const Text('Yes'),
|
child: const Text('Yes'),
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
String? deleteResult =
|
String? deleteResult =
|
||||||
await deleteBusiness(business.id, jwt);
|
await deleteBusiness(business.id);
|
||||||
if (deleteResult != null) {
|
if (deleteResult != null) {
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
SnackBar(
|
SnackBar(
|
||||||
@ -356,3 +323,105 @@ class _CreateBusinessDetailState extends State<BusinessDetail> {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class _JobList extends StatelessWidget {
|
||||||
|
final Business business;
|
||||||
|
|
||||||
|
const _JobList({required this.business});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
List<_JobListItem> listItems = [];
|
||||||
|
for (JobListing listing in business.listings!) {
|
||||||
|
listItems.add(_JobListItem(
|
||||||
|
jobListing: listing,
|
||||||
|
fromBusiness: business,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
return ListView(
|
||||||
|
shrinkWrap: true,
|
||||||
|
children: listItems,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class _JobListItem extends StatelessWidget {
|
||||||
|
final JobListing jobListing;
|
||||||
|
final Business fromBusiness;
|
||||||
|
|
||||||
|
const _JobListItem({required this.jobListing, required this.fromBusiness});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return ListTile(
|
||||||
|
leading: getIconFromJobType(
|
||||||
|
jobListing.type, 24, Theme.of(context).colorScheme.onSurface),
|
||||||
|
title: Text(jobListing.name),
|
||||||
|
subtitle: Text(
|
||||||
|
jobListing.description,
|
||||||
|
style: const TextStyle(overflow: TextOverflow.ellipsis),
|
||||||
|
),
|
||||||
|
trailing: _getEditIcon(context, fromBusiness, jobListing),
|
||||||
|
onTap: () {
|
||||||
|
Navigator.of(context).push(MaterialPageRoute(
|
||||||
|
builder: (context) => JobListingDetail(
|
||||||
|
listing: jobListing,
|
||||||
|
fromBusiness: fromBusiness,
|
||||||
|
)));
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget? _getEditIcon(
|
||||||
|
BuildContext context, Business fromBusiness, JobListing inputListing) {
|
||||||
|
if (loggedIn) {
|
||||||
|
return IconButton(
|
||||||
|
icon: const Icon(
|
||||||
|
Icons.edit,
|
||||||
|
),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.of(context).push(MaterialPageRoute(
|
||||||
|
builder: (context) => CreateEditJobListing(
|
||||||
|
inputBusiness: fromBusiness,
|
||||||
|
inputJobListing: inputListing,
|
||||||
|
)));
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class _GetListingsTitle extends StatelessWidget {
|
||||||
|
final Business fromBusiness;
|
||||||
|
|
||||||
|
const _GetListingsTitle(this.fromBusiness);
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
if (!loggedIn) {
|
||||||
|
return const Text('Available Postitions',
|
||||||
|
style: TextStyle(fontSize: 20, fontWeight: FontWeight.bold));
|
||||||
|
} else {
|
||||||
|
return Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
const Text('Available Postitions',
|
||||||
|
style: TextStyle(fontSize: 20, fontWeight: FontWeight.bold)),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 24.0),
|
||||||
|
child: IconButton(
|
||||||
|
icon: const Icon(Icons.add),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.of(context).push(MaterialPageRoute(
|
||||||
|
builder: (context) => CreateEditJobListing(
|
||||||
|
inputBusiness: fromBusiness,
|
||||||
|
)));
|
||||||
|
},
|
||||||
|
),
|
||||||
|
)
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@ -101,9 +101,9 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
|||||||
// business.contactName = 'Contact ${business.name}';
|
// business.contactName = 'Contact ${business.name}';
|
||||||
// }
|
// }
|
||||||
if (widget.inputBusiness != null) {
|
if (widget.inputBusiness != null) {
|
||||||
result = await editBusiness(business, jwt);
|
result = await editBusiness(business);
|
||||||
} else {
|
} else {
|
||||||
result = await createBusiness(business, jwt);
|
result = await createBusiness(business);
|
||||||
}
|
}
|
||||||
setState(() {
|
setState(() {
|
||||||
_isLoading = false;
|
_isLoading = false;
|
||||||
@ -159,7 +159,7 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
|||||||
return getIconFromJobType(
|
return getIconFromJobType(
|
||||||
widget.clickFromType ?? JobType.other,
|
widget.clickFromType ?? JobType.other,
|
||||||
48,
|
48,
|
||||||
Theme.of(context).colorScheme.onBackground);
|
Theme.of(context).colorScheme.onSurface);
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@ -202,13 +202,13 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
|||||||
AutovalidateMode.onUserInteraction,
|
AutovalidateMode.onUserInteraction,
|
||||||
keyboardType: TextInputType.url,
|
keyboardType: TextInputType.url,
|
||||||
onChanged: (inputUrl) {
|
onChanged: (inputUrl) {
|
||||||
setState(() {
|
business.website = Uri.encodeFull(inputUrl);
|
||||||
business.website = Uri.encodeFull(inputUrl
|
if (!business.website.contains('http://') &&
|
||||||
.toLowerCase()
|
!business.website
|
||||||
.replaceAll('https://', '')
|
.contains('https://')) {
|
||||||
.replaceAll('http://', '')
|
business.website =
|
||||||
.replaceAll('www.', ''));
|
'https://${business.website}';
|
||||||
});
|
}
|
||||||
},
|
},
|
||||||
onTapOutside: (PointerDownEvent event) {
|
onTapOutside: (PointerDownEvent event) {
|
||||||
FocusScope.of(context).unfocus();
|
FocusScope.of(context).unfocus();
|
||||||
@ -362,48 +362,6 @@ class _CreateEditBusinessState extends State<CreateEditBusiness> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
// Business Type Dropdown
|
|
||||||
|
|
||||||
// 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,
|
|
||||||
// controller: businessTypeController,
|
|
||||||
// label: const Text('Business Type'),
|
|
||||||
// dropdownMenuEntries: const [
|
|
||||||
// DropdownMenuEntry(
|
|
||||||
// value: BusinessType.food,
|
|
||||||
// label: 'Food Related'),
|
|
||||||
// DropdownMenuEntry(
|
|
||||||
// value: BusinessType.shop,
|
|
||||||
// label: 'Shop'),
|
|
||||||
// DropdownMenuEntry(
|
|
||||||
// value: BusinessType.outdoors,
|
|
||||||
// label: 'Outdoors'),
|
|
||||||
// DropdownMenuEntry(
|
|
||||||
// value: BusinessType.manufacturing,
|
|
||||||
// label: 'Manufacturing'),
|
|
||||||
// DropdownMenuEntry(
|
|
||||||
// value: BusinessType.entertainment,
|
|
||||||
// label: 'Entertainment'),
|
|
||||||
// DropdownMenuEntry(
|
|
||||||
// value: BusinessType.other,
|
|
||||||
// label: 'Other'),
|
|
||||||
// ],
|
|
||||||
// onSelected: (inputType) {
|
|
||||||
// business.type = inputType!;
|
|
||||||
// },
|
|
||||||
// ),
|
|
||||||
// ],
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.only(
|
padding: const EdgeInsets.only(
|
||||||
left: 8.0, right: 8.0, bottom: 8.0),
|
left: 8.0, right: 8.0, bottom: 8.0),
|
||||||
|
|||||||
431
fbla_ui/lib/pages/create_edit_listing.dart
Normal file
431
fbla_ui/lib/pages/create_edit_listing.dart
Normal file
@ -0,0 +1,431 @@
|
|||||||
|
import 'package:fbla_ui/api_logic.dart';
|
||||||
|
import 'package:fbla_ui/main.dart';
|
||||||
|
import 'package:fbla_ui/shared.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:rive/rive.dart';
|
||||||
|
|
||||||
|
class CreateEditJobListing extends StatefulWidget {
|
||||||
|
final JobListing? inputJobListing;
|
||||||
|
final Business inputBusiness;
|
||||||
|
|
||||||
|
const CreateEditJobListing(
|
||||||
|
{super.key, this.inputJobListing, required this.inputBusiness});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<CreateEditJobListing> createState() => _CreateEditJobListingState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CreateEditJobListingState extends State<CreateEditJobListing> {
|
||||||
|
late Future getBusinessNameMapping;
|
||||||
|
late TextEditingController _nameController;
|
||||||
|
late TextEditingController _descriptionController;
|
||||||
|
late TextEditingController _wageController;
|
||||||
|
late TextEditingController _linkController;
|
||||||
|
List nameMapping = [];
|
||||||
|
String? businessErrorText;
|
||||||
|
|
||||||
|
JobListing listing = JobListing(
|
||||||
|
id: null,
|
||||||
|
businessId: null,
|
||||||
|
name: 'Job Listing',
|
||||||
|
description: 'Add details about the business below.',
|
||||||
|
type: JobType.other,
|
||||||
|
wage: null,
|
||||||
|
link: null);
|
||||||
|
bool _isLoading = false;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
if (widget.inputJobListing != null) {
|
||||||
|
listing = JobListing.copy(widget.inputJobListing!);
|
||||||
|
_nameController = TextEditingController(text: listing.name);
|
||||||
|
_descriptionController = TextEditingController(text: listing.description);
|
||||||
|
} else {
|
||||||
|
_nameController = TextEditingController();
|
||||||
|
_descriptionController = TextEditingController();
|
||||||
|
}
|
||||||
|
_wageController = TextEditingController(text: listing.wage);
|
||||||
|
_linkController = TextEditingController(text: listing.link);
|
||||||
|
getBusinessNameMapping = fetchBusinessNames();
|
||||||
|
}
|
||||||
|
|
||||||
|
final formKey = GlobalKey<FormState>();
|
||||||
|
final TextEditingController jobTypeController = TextEditingController();
|
||||||
|
final TextEditingController businessController = TextEditingController();
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
listing.businessId = widget.inputBusiness.id;
|
||||||
|
return PopScope(
|
||||||
|
canPop: !_isLoading,
|
||||||
|
onPopInvoked: _handlePop,
|
||||||
|
child: Form(
|
||||||
|
key: formKey,
|
||||||
|
child: Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: (widget.inputJobListing != null)
|
||||||
|
? Text('Edit ${widget.inputJobListing?.name}', maxLines: 1)
|
||||||
|
: const Text('Add New Job Listing'),
|
||||||
|
),
|
||||||
|
floatingActionButton: FloatingActionButton(
|
||||||
|
child: _isLoading
|
||||||
|
? const Padding(
|
||||||
|
padding: EdgeInsets.all(16.0),
|
||||||
|
child: CircularProgressIndicator(
|
||||||
|
color: Colors.white,
|
||||||
|
strokeWidth: 3.0,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
: const Icon(Icons.save),
|
||||||
|
onPressed: () async {
|
||||||
|
if (formKey.currentState!.validate()) {
|
||||||
|
formKey.currentState?.save();
|
||||||
|
setState(() {
|
||||||
|
_isLoading = true;
|
||||||
|
});
|
||||||
|
String? result;
|
||||||
|
if (widget.inputJobListing != null) {
|
||||||
|
result = await editListing(listing);
|
||||||
|
} else {
|
||||||
|
result = await createListing(listing);
|
||||||
|
}
|
||||||
|
setState(() {
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
if (result != null) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(
|
||||||
|
width: 400,
|
||||||
|
behavior: SnackBarBehavior.floating,
|
||||||
|
content: Text(result)));
|
||||||
|
} else {
|
||||||
|
Navigator.pushReplacement(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => const MainApp()));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: const Text('Check field inputs!'),
|
||||||
|
width: 200,
|
||||||
|
behavior: SnackBarBehavior.floating,
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.circular(10)),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
body: FutureBuilder(
|
||||||
|
future: getBusinessNameMapping,
|
||||||
|
builder: (context, snapshot) {
|
||||||
|
if (snapshot.connectionState == ConnectionState.done) {
|
||||||
|
if (snapshot.hasData) {
|
||||||
|
if (snapshot.data.runtimeType == String) {
|
||||||
|
return 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: () async {
|
||||||
|
var refreshedData = fetchBusinessNames();
|
||||||
|
await refreshedData;
|
||||||
|
setState(() {
|
||||||
|
getBusinessNameMapping = refreshedData;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
nameMapping = snapshot.data;
|
||||||
|
|
||||||
|
return ListView(
|
||||||
|
children: [
|
||||||
|
Center(
|
||||||
|
child: SizedBox(
|
||||||
|
width: 1000,
|
||||||
|
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(
|
||||||
|
borderRadius: BorderRadius.circular(6.0),
|
||||||
|
child: Image.network(
|
||||||
|
width: 48,
|
||||||
|
height: 48,
|
||||||
|
listing.businessId != null
|
||||||
|
? '$apiAddress/logos/${listing.businessId}'
|
||||||
|
: '',
|
||||||
|
errorBuilder: (BuildContext context,
|
||||||
|
Object exception,
|
||||||
|
StackTrace? stackTrace) {
|
||||||
|
return getIconFromJobType(
|
||||||
|
listing.type,
|
||||||
|
48,
|
||||||
|
Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onSurface);
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
// Business Type Dropdown
|
||||||
|
Card(
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(
|
||||||
|
left: 8.0,
|
||||||
|
right: 8.0,
|
||||||
|
bottom: 8.0,
|
||||||
|
top: 8),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment:
|
||||||
|
MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
const Text('Type of Job',
|
||||||
|
style:
|
||||||
|
TextStyle(fontSize: 16)),
|
||||||
|
DropdownMenu<JobType>(
|
||||||
|
initialSelection: listing.type,
|
||||||
|
controller: jobTypeController,
|
||||||
|
label: const Text('Job Type'),
|
||||||
|
dropdownMenuEntries: [
|
||||||
|
for (JobType type
|
||||||
|
in JobType.values)
|
||||||
|
DropdownMenuEntry(
|
||||||
|
value: type,
|
||||||
|
label:
|
||||||
|
getNameFromJobType(
|
||||||
|
type))
|
||||||
|
],
|
||||||
|
onSelected: (inputType) {
|
||||||
|
setState(() {
|
||||||
|
listing.type = inputType!;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(
|
||||||
|
left: 8.0,
|
||||||
|
right: 8.0,
|
||||||
|
bottom: 8.0,
|
||||||
|
top: 8),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment:
|
||||||
|
MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
const Text(
|
||||||
|
'Business that has the job',
|
||||||
|
style:
|
||||||
|
TextStyle(fontSize: 16)),
|
||||||
|
DropdownMenu<int>(
|
||||||
|
initialSelection:
|
||||||
|
widget.inputBusiness.id,
|
||||||
|
controller: businessController,
|
||||||
|
label: const Text('Business'),
|
||||||
|
dropdownMenuEntries: [
|
||||||
|
for (Map<String, dynamic> map
|
||||||
|
in nameMapping)
|
||||||
|
DropdownMenuEntry(
|
||||||
|
value: map['id']!,
|
||||||
|
label: map['name'])
|
||||||
|
],
|
||||||
|
onSelected: (inputType) {
|
||||||
|
setState(() {
|
||||||
|
listing.businessId =
|
||||||
|
inputType!;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(
|
||||||
|
left: 8.0, right: 8.0),
|
||||||
|
child: TextFormField(
|
||||||
|
controller: _nameController,
|
||||||
|
autovalidateMode: AutovalidateMode
|
||||||
|
.onUserInteraction,
|
||||||
|
maxLength: 30,
|
||||||
|
onChanged: (inputName) {
|
||||||
|
setState(() {
|
||||||
|
listing.name = inputName;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
onTapOutside:
|
||||||
|
(PointerDownEvent event) {
|
||||||
|
FocusScope.of(context).unfocus();
|
||||||
|
},
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
labelText:
|
||||||
|
'Job Listing Name (required)',
|
||||||
|
),
|
||||||
|
validator: (value) {
|
||||||
|
if (value != null &&
|
||||||
|
value.isEmpty) {
|
||||||
|
return 'Name is required';
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(
|
||||||
|
left: 8.0, right: 8.0),
|
||||||
|
child: TextFormField(
|
||||||
|
controller: _descriptionController,
|
||||||
|
autovalidateMode: AutovalidateMode
|
||||||
|
.onUserInteraction,
|
||||||
|
maxLength: 500,
|
||||||
|
maxLines: null,
|
||||||
|
onChanged: (inputDesc) {
|
||||||
|
setState(() {
|
||||||
|
listing.description = inputDesc;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
onTapOutside:
|
||||||
|
(PointerDownEvent event) {
|
||||||
|
FocusScope.of(context).unfocus();
|
||||||
|
},
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
labelText:
|
||||||
|
'Job Listing Description (required)',
|
||||||
|
),
|
||||||
|
validator: (value) {
|
||||||
|
if (value != null &&
|
||||||
|
value.isEmpty) {
|
||||||
|
return 'Description is required';
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(
|
||||||
|
left: 8.0,
|
||||||
|
right: 8.0,
|
||||||
|
bottom: 8.0),
|
||||||
|
child: TextFormField(
|
||||||
|
controller: _wageController,
|
||||||
|
onChanged: (input) {
|
||||||
|
setState(() {
|
||||||
|
listing.wage = input;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
onTapOutside:
|
||||||
|
(PointerDownEvent event) {
|
||||||
|
FocusScope.of(context).unfocus();
|
||||||
|
},
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
labelText: 'Wage Information',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(
|
||||||
|
left: 8.0,
|
||||||
|
right: 8.0,
|
||||||
|
bottom: 8.0),
|
||||||
|
child: TextFormField(
|
||||||
|
controller: _linkController,
|
||||||
|
autovalidateMode: AutovalidateMode
|
||||||
|
.onUserInteraction,
|
||||||
|
keyboardType: TextInputType.url,
|
||||||
|
onChanged: (inputUrl) {
|
||||||
|
if (listing.link != null &&
|
||||||
|
listing.link != '') {
|
||||||
|
listing.link =
|
||||||
|
Uri.encodeFull(inputUrl);
|
||||||
|
if (!listing.link!
|
||||||
|
.contains('http://') &&
|
||||||
|
!listing.link!
|
||||||
|
.contains('https://')) {
|
||||||
|
listing.link =
|
||||||
|
'https://${listing.link}';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onTapOutside:
|
||||||
|
(PointerDownEvent event) {
|
||||||
|
FocusScope.of(context).unfocus();
|
||||||
|
},
|
||||||
|
decoration: const InputDecoration(
|
||||||
|
labelText:
|
||||||
|
'Additional Information Link',
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(
|
||||||
|
height: 75,
|
||||||
|
)
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
} else if (snapshot.hasError) {
|
||||||
|
return 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) {
|
||||||
|
return 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 const Padding(
|
||||||
|
padding: EdgeInsets.only(left: 16.0, right: 16.0),
|
||||||
|
child: Text('Error when loading data!'),
|
||||||
|
);
|
||||||
|
}),
|
||||||
|
)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
void _handlePop(bool didPop) {
|
||||||
|
if (!didPop) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
const SnackBar(
|
||||||
|
width: 400,
|
||||||
|
behavior: SnackBarBehavior.floating,
|
||||||
|
content: Text('Please wait for it to save.'),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
221
fbla_ui/lib/pages/listing_detail.dart
Normal file
221
fbla_ui/lib/pages/listing_detail.dart
Normal file
@ -0,0 +1,221 @@
|
|||||||
|
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:flutter/material.dart';
|
||||||
|
import 'package:url_launcher/url_launcher.dart';
|
||||||
|
|
||||||
|
class JobListingDetail extends StatefulWidget {
|
||||||
|
final JobListing listing;
|
||||||
|
final Business fromBusiness;
|
||||||
|
|
||||||
|
const JobListingDetail(
|
||||||
|
{super.key, required this.listing, required this.fromBusiness});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<JobListingDetail> createState() => _CreateBusinessDetailState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _CreateBusinessDetailState extends State<JobListingDetail> {
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(widget.listing.name),
|
||||||
|
actions: _getActions(widget.listing, widget.fromBusiness),
|
||||||
|
),
|
||||||
|
body: _detailBody(widget.listing),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
ListView _detailBody(JobListing listing) {
|
||||||
|
return ListView(
|
||||||
|
children: [
|
||||||
|
// Title, logo, desc, website
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 4.0),
|
||||||
|
child: Card(
|
||||||
|
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(
|
||||||
|
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);
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Visibility(
|
||||||
|
visible: listing.link != null && listing.link != '',
|
||||||
|
child: ListTile(
|
||||||
|
leading: const Icon(Icons.link),
|
||||||
|
title: const Text('More Information'),
|
||||||
|
subtitle: Text(
|
||||||
|
listing.link!
|
||||||
|
.replaceAll('https://', '')
|
||||||
|
.replaceAll('http://', '')
|
||||||
|
.replaceAll('www.', ''),
|
||||||
|
style: const TextStyle(color: Colors.blue)),
|
||||||
|
onTap: () {
|
||||||
|
launchUrl(Uri.parse(listing.link!));
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
// Wage
|
||||||
|
Visibility(
|
||||||
|
visible: listing.wage != null && listing.wage != '',
|
||||||
|
child: Card(
|
||||||
|
child: ListTile(
|
||||||
|
leading: const Icon(Icons.attach_money),
|
||||||
|
subtitle: Text(listing.wage!),
|
||||||
|
title: const Text('Wage Information'),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Card(
|
||||||
|
clipBehavior: Clip.antiAlias,
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
Row(
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(left: 16.0, top: 8.0),
|
||||||
|
child: Text(
|
||||||
|
widget.fromBusiness.contactName!,
|
||||||
|
textAlign: TextAlign.left,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontSize: 20, fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
Visibility(
|
||||||
|
visible: widget.fromBusiness.contactPhone != null,
|
||||||
|
child: ListTile(
|
||||||
|
leading: const Icon(Icons.phone),
|
||||||
|
title: Text(widget.fromBusiness.contactPhone!),
|
||||||
|
// maybe replace ! with ?? ''. same is true for below
|
||||||
|
onTap: () {
|
||||||
|
showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
backgroundColor:
|
||||||
|
Theme.of(context).colorScheme.surface,
|
||||||
|
title: Text(
|
||||||
|
'Contact ${widget.fromBusiness.contactName}'),
|
||||||
|
content: Text(
|
||||||
|
'Would you like to call or text ${widget.fromBusiness.contactName}?'),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
child: const Text('Text'),
|
||||||
|
onPressed: () {
|
||||||
|
launchUrl(Uri.parse(
|
||||||
|
'sms:${widget.fromBusiness.contactPhone}'));
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
}),
|
||||||
|
TextButton(
|
||||||
|
child: const Text('Call'),
|
||||||
|
onPressed: () async {
|
||||||
|
launchUrl(Uri.parse(
|
||||||
|
'tel:${widget.fromBusiness.contactPhone}'));
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.email),
|
||||||
|
title: Text(widget.fromBusiness.contactEmail),
|
||||||
|
onTap: () {
|
||||||
|
launchUrl(
|
||||||
|
Uri.parse('mailto:${widget.fromBusiness.contactEmail}'));
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
List<Widget>? _getActions(JobListing listing, Business fromBusiness) {
|
||||||
|
if (loggedIn) {
|
||||||
|
return [
|
||||||
|
IconButton(
|
||||||
|
icon: const Icon(Icons.edit),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.of(context).push(MaterialPageRoute(
|
||||||
|
builder: (context) => CreateEditJobListing(
|
||||||
|
inputJobListing: listing,
|
||||||
|
inputBusiness: fromBusiness,
|
||||||
|
)));
|
||||||
|
},
|
||||||
|
),
|
||||||
|
IconButton(
|
||||||
|
icon: const Icon(Icons.delete),
|
||||||
|
onPressed: () {
|
||||||
|
showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
backgroundColor: Theme.of(context).colorScheme.surface,
|
||||||
|
title: const Text('Are You Sure?'),
|
||||||
|
content:
|
||||||
|
Text('This will permanently delete ${listing.name}.'),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
child: const Text('Cancel'),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
}),
|
||||||
|
TextButton(
|
||||||
|
child: const Text('Yes'),
|
||||||
|
onPressed: () async {
|
||||||
|
String? deleteResult =
|
||||||
|
await deleteListing(listing.id!);
|
||||||
|
if (deleteResult != null) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
width: 300,
|
||||||
|
behavior: SnackBarBehavior.floating,
|
||||||
|
content: Text(deleteResult)));
|
||||||
|
} else {
|
||||||
|
Navigator.pushReplacement(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => const MainApp()));
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,5 +1,4 @@
|
|||||||
import 'package:fbla_ui/api_logic.dart';
|
import 'package:fbla_ui/api_logic.dart';
|
||||||
import 'package:fbla_ui/home.dart';
|
|
||||||
import 'package:fbla_ui/shared.dart';
|
import 'package:fbla_ui/shared.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
@ -7,7 +6,7 @@ import 'package:shared_preferences/shared_preferences.dart';
|
|||||||
bool loggedIn = false;
|
bool loggedIn = false;
|
||||||
|
|
||||||
class SignInPage extends StatefulWidget {
|
class SignInPage extends StatefulWidget {
|
||||||
final Callback refreshAccount;
|
final void Function() refreshAccount;
|
||||||
|
|
||||||
const SignInPage({super.key, required this.refreshAccount});
|
const SignInPage({super.key, required this.refreshAccount});
|
||||||
|
|
||||||
|
|||||||
@ -8,19 +8,13 @@ import 'package:sliver_tools/sliver_tools.dart';
|
|||||||
import 'package:url_launcher/url_launcher.dart';
|
import 'package:url_launcher/url_launcher.dart';
|
||||||
|
|
||||||
late String jwt;
|
late String jwt;
|
||||||
Set<BusinessType> filters = <BusinessType>{};
|
|
||||||
Set<BusinessType> selectedChips = <BusinessType>{};
|
|
||||||
String searchFilter = '';
|
String searchFilter = '';
|
||||||
bool isFiltered = false;
|
|
||||||
Set<Business> selectedBusinesses = <Business>{};
|
Set<Business> selectedBusinesses = <Business>{};
|
||||||
Set<DataType> selectedDataTypes = <DataType>{};
|
|
||||||
Set<DataType> dataTypeFilters = <DataType>{};
|
|
||||||
|
|
||||||
enum DataType {
|
enum DataTypeBusiness {
|
||||||
logo,
|
logo,
|
||||||
name,
|
name,
|
||||||
description,
|
description,
|
||||||
// type,
|
|
||||||
website,
|
website,
|
||||||
contactName,
|
contactName,
|
||||||
contactEmail,
|
contactEmail,
|
||||||
@ -28,34 +22,67 @@ enum DataType {
|
|||||||
notes,
|
notes,
|
||||||
}
|
}
|
||||||
|
|
||||||
Map<DataType, int> dataTypeValues = {
|
enum DataTypeJob {
|
||||||
DataType.logo: 0,
|
businessName,
|
||||||
DataType.name: 1,
|
name,
|
||||||
DataType.description: 2,
|
description,
|
||||||
|
wage,
|
||||||
|
link,
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<DataTypeBusiness, int> dataTypePriorityBusiness = {
|
||||||
|
DataTypeBusiness.logo: 0,
|
||||||
|
DataTypeBusiness.name: 1,
|
||||||
|
DataTypeBusiness.description: 2,
|
||||||
// DataType.type: 3,
|
// DataType.type: 3,
|
||||||
DataType.website: 4,
|
DataTypeBusiness.website: 4,
|
||||||
DataType.contactName: 5,
|
DataTypeBusiness.contactName: 5,
|
||||||
DataType.contactEmail: 6,
|
DataTypeBusiness.contactEmail: 6,
|
||||||
DataType.contactPhone: 7,
|
DataTypeBusiness.contactPhone: 7,
|
||||||
DataType.notes: 8
|
DataTypeBusiness.notes: 8
|
||||||
};
|
};
|
||||||
|
|
||||||
Map<DataType, String> dataTypeFriendly = {
|
Map<DataTypeBusiness, String> dataTypeFriendlyBusiness = {
|
||||||
DataType.logo: 'Logo',
|
DataTypeBusiness.logo: 'Logo',
|
||||||
DataType.name: 'Name',
|
DataTypeBusiness.name: 'Name',
|
||||||
DataType.description: 'Description',
|
DataTypeBusiness.description: 'Description',
|
||||||
// DataType.type: 'Type',
|
// DataType.type: 'Type',
|
||||||
DataType.website: 'Website',
|
DataTypeBusiness.website: 'Website',
|
||||||
DataType.contactName: 'Contact Name',
|
DataTypeBusiness.contactName: 'Contact Name',
|
||||||
DataType.contactEmail: 'Contact Email',
|
DataTypeBusiness.contactEmail: 'Contact Email',
|
||||||
DataType.contactPhone: 'Contact Phone',
|
DataTypeBusiness.contactPhone: 'Contact Phone',
|
||||||
DataType.notes: 'Notes'
|
DataTypeBusiness.notes: 'Notes'
|
||||||
};
|
};
|
||||||
|
|
||||||
Set<DataType> sortDataTypes(Set<DataType> set) {
|
Map<DataTypeJob, int> dataTypePriorityJob = {
|
||||||
List<DataType> list = set.toList();
|
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) {
|
list.sort((a, b) {
|
||||||
return dataTypeValues[a]!.compareTo(dataTypeValues[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();
|
set = list.toSet();
|
||||||
return set;
|
return set;
|
||||||
@ -73,8 +100,8 @@ enum BusinessType {
|
|||||||
enum JobType { cashier, server, mechanic, other }
|
enum JobType { cashier, server, mechanic, other }
|
||||||
|
|
||||||
class JobListing {
|
class JobListing {
|
||||||
String? id;
|
int? id;
|
||||||
String? businessId;
|
int? businessId;
|
||||||
String name;
|
String name;
|
||||||
String description;
|
String description;
|
||||||
JobType type;
|
JobType type;
|
||||||
@ -89,18 +116,30 @@ class JobListing {
|
|||||||
required this.type,
|
required this.type,
|
||||||
this.wage,
|
this.wage,
|
||||||
this.link});
|
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 {
|
class Business {
|
||||||
int id;
|
int id;
|
||||||
String name;
|
String name;
|
||||||
String description;
|
String description;
|
||||||
String? website;
|
String website;
|
||||||
String? contactName;
|
String? contactName;
|
||||||
String? contactEmail;
|
String contactEmail;
|
||||||
String? contactPhone;
|
String? contactPhone;
|
||||||
String? notes;
|
String? notes;
|
||||||
String? locationName;
|
String locationName;
|
||||||
String? locationAddress;
|
String? locationAddress;
|
||||||
List<JobListing>? listings;
|
List<JobListing>? listings;
|
||||||
|
|
||||||
@ -108,12 +147,12 @@ class Business {
|
|||||||
{required this.id,
|
{required this.id,
|
||||||
required this.name,
|
required this.name,
|
||||||
required this.description,
|
required this.description,
|
||||||
this.website,
|
required this.website,
|
||||||
this.contactName,
|
this.contactName,
|
||||||
this.contactEmail,
|
required this.contactEmail,
|
||||||
this.contactPhone,
|
this.contactPhone,
|
||||||
this.notes,
|
this.notes,
|
||||||
this.locationName,
|
required this.locationName,
|
||||||
this.locationAddress,
|
this.locationAddress,
|
||||||
this.listings});
|
this.listings});
|
||||||
|
|
||||||
@ -123,11 +162,13 @@ class Business {
|
|||||||
listings = [];
|
listings = [];
|
||||||
for (int i = 0; i < json['listings'].length; i++) {
|
for (int i = 0; i < json['listings'].length; i++) {
|
||||||
listings.add(JobListing(
|
listings.add(JobListing(
|
||||||
name: json['listings']['name'],
|
id: json['listings'][i]['id'],
|
||||||
description: json['listings']['description'],
|
businessId: json['listings'][i]['businessId'],
|
||||||
type: json['listings']['type'],
|
name: json['listings'][i]['name'],
|
||||||
wage: json['listings']['wage'],
|
description: json['listings'][i]['description'],
|
||||||
link: json['listings']['link']));
|
type: JobType.values.byName(json['listings'][i]['type']),
|
||||||
|
wage: json['listings'][i]['wage'],
|
||||||
|
link: json['listings'][i]['link']));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -168,7 +209,7 @@ class Business {
|
|||||||
// return groupedBusinesses;
|
// return groupedBusinesses;
|
||||||
// }
|
// }
|
||||||
|
|
||||||
Icon getIconFromType(BusinessType type, double size, Color color) {
|
Icon getIconFromBusinessType(BusinessType type, double size, Color color) {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case BusinessType.food:
|
case BusinessType.food:
|
||||||
return Icon(
|
return Icon(
|
||||||
@ -238,7 +279,8 @@ Icon getIconFromJobType(JobType type, double size, Color color) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pw.Icon getPwIconFromType(BusinessType type, double size, PdfColor color) {
|
pw.Icon getPwIconFromBusinessType(
|
||||||
|
BusinessType type, double size, PdfColor color) {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case BusinessType.food:
|
case BusinessType.food:
|
||||||
return pw.Icon(const pw.IconData(0xe56c), size: size, color: color);
|
return pw.Icon(const pw.IconData(0xe56c), size: size, color: color);
|
||||||
@ -268,33 +310,33 @@ pw.Icon getPwIconFromJobType(JobType type, double size, PdfColor color) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Text getNameFromType(BusinessType type, Color color) {
|
String getNameFromBusinessType(BusinessType type) {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case BusinessType.food:
|
case BusinessType.food:
|
||||||
return Text('Food Related', style: TextStyle(color: color));
|
return 'Food Related';
|
||||||
case BusinessType.shop:
|
case BusinessType.shop:
|
||||||
return Text('Shops', style: TextStyle(color: color));
|
return 'Shops';
|
||||||
case BusinessType.outdoors:
|
case BusinessType.outdoors:
|
||||||
return Text('Outdoors', style: TextStyle(color: color));
|
return 'Outdoors';
|
||||||
case BusinessType.manufacturing:
|
case BusinessType.manufacturing:
|
||||||
return Text('Manufacturing', style: TextStyle(color: color));
|
return 'Manufacturing';
|
||||||
case BusinessType.entertainment:
|
case BusinessType.entertainment:
|
||||||
return Text('Entertainment', style: TextStyle(color: color));
|
return 'Entertainment';
|
||||||
case BusinessType.other:
|
case BusinessType.other:
|
||||||
return Text('Other', style: TextStyle(color: color));
|
return 'Other';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Text getNameFromJobType(JobType type, Color color) {
|
String getNameFromJobType(JobType type) {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case JobType.cashier:
|
case JobType.cashier:
|
||||||
return Text('Cashier', style: TextStyle(color: color));
|
return 'Cashier';
|
||||||
case JobType.server:
|
case JobType.server:
|
||||||
return Text('Server', style: TextStyle(color: color));
|
return 'Server';
|
||||||
case JobType.mechanic:
|
case JobType.mechanic:
|
||||||
return Text('Mechanic', style: TextStyle(color: color));
|
return 'Mechanic';
|
||||||
case JobType.other:
|
case JobType.other:
|
||||||
return Text('Other', style: TextStyle(color: color));
|
return 'Other';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -337,9 +379,9 @@ class _BusinessDisplayPanelState extends State<BusinessDisplayPanel> {
|
|||||||
// }
|
// }
|
||||||
// }
|
// }
|
||||||
|
|
||||||
if (filters.isNotEmpty) {
|
// if (filters.isNotEmpty) {
|
||||||
isFiltered = true;
|
// isFiltered = true;
|
||||||
}
|
// }
|
||||||
|
|
||||||
// for (var i = 0; i < businessTypes.length; i++) {
|
// for (var i = 0; i < businessTypes.length; i++) {
|
||||||
// if (filters.contains(businessTypes[i])) {
|
// if (filters.contains(businessTypes[i])) {
|
||||||
@ -347,18 +389,18 @@ class _BusinessDisplayPanelState extends State<BusinessDisplayPanel> {
|
|||||||
// }
|
// }
|
||||||
// }
|
// }
|
||||||
|
|
||||||
if (isFiltered) {
|
// if (isFiltered) {
|
||||||
for (JobType jobType in widget.groupedBusinesses.keys) {
|
// for (JobType jobType in widget.groupedBusinesses.keys) {
|
||||||
if (filters.contains(jobType)) {
|
// if (filters.contains(jobType)) {
|
||||||
headers.add(BusinessHeader(
|
// headers.add(BusinessHeader(
|
||||||
type: jobType,
|
// type: jobType,
|
||||||
widescreen: widget.widescreen,
|
// widescreen: widget.widescreen,
|
||||||
selectable: widget.selectable,
|
// selectable: widget.selectable,
|
||||||
selectedBusinesses: selectedBusinesses,
|
// selectedBusinesses: selectedBusinesses,
|
||||||
businesses: widget.groupedBusinesses[jobType]!));
|
// businesses: widget.groupedBusinesses[jobType]!));
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
} else {
|
// } else {
|
||||||
for (JobType jobType in widget.groupedBusinesses.keys) {
|
for (JobType jobType in widget.groupedBusinesses.keys) {
|
||||||
headers.add(BusinessHeader(
|
headers.add(BusinessHeader(
|
||||||
type: jobType,
|
type: jobType,
|
||||||
@ -367,7 +409,7 @@ class _BusinessDisplayPanelState extends State<BusinessDisplayPanel> {
|
|||||||
selectedBusinesses: selectedBusinesses,
|
selectedBusinesses: selectedBusinesses,
|
||||||
businesses: widget.groupedBusinesses[jobType]!));
|
businesses: widget.groupedBusinesses[jobType]!));
|
||||||
}
|
}
|
||||||
}
|
// }
|
||||||
headers.sort((a, b) => a.type.index.compareTo(b.type.index));
|
headers.sort((a, b) => a.type.index.compareTo(b.type.index));
|
||||||
return MultiSliver(children: headers);
|
return MultiSliver(children: headers);
|
||||||
}
|
}
|
||||||
@ -425,8 +467,7 @@ class _BusinessHeaderState extends State<BusinessHeader> {
|
|||||||
child: getIconFromJobType(
|
child: getIconFromJobType(
|
||||||
widget.type, 24, Theme.of(context).colorScheme.onPrimary),
|
widget.type, 24, Theme.of(context).colorScheme.onPrimary),
|
||||||
),
|
),
|
||||||
getNameFromJobType(
|
Text(getNameFromJobType(widget.type)),
|
||||||
widget.type, Theme.of(context).colorScheme.onPrimary),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
Padding(
|
Padding(
|
||||||
@ -458,8 +499,10 @@ class _BusinessHeaderState extends State<BusinessHeader> {
|
|||||||
child: getIconFromJobType(
|
child: getIconFromJobType(
|
||||||
widget.type, 24, Theme.of(context).colorScheme.onPrimary),
|
widget.type, 24, Theme.of(context).colorScheme.onPrimary),
|
||||||
),
|
),
|
||||||
getNameFromJobType(
|
Text(
|
||||||
widget.type, Theme.of(context).colorScheme.onPrimary),
|
getNameFromJobType(widget.type),
|
||||||
|
style: TextStyle(color: Theme.of(context).colorScheme.onPrimary),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -578,8 +621,7 @@ class _BusinessCardState extends State<BusinessCard> {
|
|||||||
Uri.parse('https://${business.website}'));
|
Uri.parse('https://${business.website}'));
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
if ((business.locationName != null) &&
|
if (business.locationName != '')
|
||||||
(business.locationName != ''))
|
|
||||||
IconButton(
|
IconButton(
|
||||||
icon: const Icon(Icons.location_on),
|
icon: const Icon(Icons.location_on),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
@ -598,7 +640,7 @@ class _BusinessCardState extends State<BusinessCard> {
|
|||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
backgroundColor: Theme.of(context)
|
backgroundColor: Theme.of(context)
|
||||||
.colorScheme
|
.colorScheme
|
||||||
.background,
|
.surface,
|
||||||
title: Text((business.contactName ==
|
title: Text((business.contactName ==
|
||||||
null ||
|
null ||
|
||||||
business.contactName == '')
|
business.contactName == '')
|
||||||
@ -629,8 +671,7 @@ class _BusinessCardState extends State<BusinessCard> {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
if ((business.contactEmail != null) &&
|
if (business.contactEmail != '')
|
||||||
(business.contactEmail != ''))
|
|
||||||
IconButton(
|
IconButton(
|
||||||
icon: const Icon(Icons.email),
|
icon: const Icon(Icons.email),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
@ -770,8 +811,105 @@ class _BusinessCardState extends State<BusinessCard> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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 {
|
class FilterChips extends StatefulWidget {
|
||||||
const FilterChips({super.key});
|
final Set<JobType> selectedChips;
|
||||||
|
|
||||||
|
const FilterChips({super.key, required this.selectedChips});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<FilterChips> createState() => _FilterChipsState();
|
State<FilterChips> createState() => _FilterChipsState();
|
||||||
@ -781,22 +919,21 @@ class _FilterChipsState extends State<FilterChips> {
|
|||||||
List<Padding> filterChips() {
|
List<Padding> filterChips() {
|
||||||
List<Padding> chips = [];
|
List<Padding> chips = [];
|
||||||
|
|
||||||
for (var type in BusinessType.values) {
|
for (var type in JobType.values) {
|
||||||
chips.add(Padding(
|
chips.add(Padding(
|
||||||
padding: const EdgeInsets.only(left: 4.0, right: 4.0),
|
padding: const EdgeInsets.only(left: 4.0, right: 4.0),
|
||||||
child: FilterChip(
|
child: FilterChip(
|
||||||
showCheckmark: false,
|
showCheckmark: false,
|
||||||
shape:
|
shape:
|
||||||
RoundedRectangleBorder(borderRadius: BorderRadius.circular(20)),
|
RoundedRectangleBorder(borderRadius: BorderRadius.circular(20)),
|
||||||
label:
|
label: Text(getNameFromJobType(type)),
|
||||||
getNameFromType(type, Theme.of(context).colorScheme.onSurface),
|
selected: widget.selectedChips.contains(type),
|
||||||
selected: selectedChips.contains(type),
|
|
||||||
onSelected: (bool selected) {
|
onSelected: (bool selected) {
|
||||||
setState(() {
|
setState(() {
|
||||||
if (selected) {
|
if (selected) {
|
||||||
selectedChips.add(type);
|
widget.selectedChips.add(type);
|
||||||
} else {
|
} else {
|
||||||
selectedChips.remove(type);
|
widget.selectedChips.remove(type);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}),
|
}),
|
||||||
@ -812,64 +949,3 @@ class _FilterChipsState extends State<FilterChips> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class FilterDataTypeChips extends StatefulWidget {
|
|
||||||
const FilterDataTypeChips({super.key});
|
|
||||||
|
|
||||||
@override
|
|
||||||
State<FilterDataTypeChips> createState() => _FilterDataTypeChipsState();
|
|
||||||
}
|
|
||||||
|
|
||||||
class _FilterDataTypeChipsState extends State<FilterDataTypeChips> {
|
|
||||||
List<Padding> filterDataTypeChips() {
|
|
||||||
List<Padding> chips = [];
|
|
||||||
|
|
||||||
for (var type in DataType.values) {
|
|
||||||
chips.add(Padding(
|
|
||||||
padding:
|
|
||||||
const EdgeInsets.only(left: 3.0, right: 3.0, bottom: 3.0, top: 3.0),
|
|
||||||
// child: ActionChip(
|
|
||||||
// avatar: selectedDataTypes.contains(type) ? Icon(Icons.check_box) : Icon(Icons.check_box_outline_blank),
|
|
||||||
// label: Text(type.name),
|
|
||||||
// onPressed: () {
|
|
||||||
// if (!selectedDataTypes.contains(type)) {
|
|
||||||
// setState(() {
|
|
||||||
// selectedDataTypes.add(type);
|
|
||||||
// });
|
|
||||||
// } else {
|
|
||||||
// setState(() {
|
|
||||||
// selectedDataTypes.remove(type);
|
|
||||||
// });
|
|
||||||
// }
|
|
||||||
// },
|
|
||||||
|
|
||||||
// ),
|
|
||||||
child: FilterChip(
|
|
||||||
shape: RoundedRectangleBorder(
|
|
||||||
borderRadius: BorderRadius.circular(20),
|
|
||||||
side:
|
|
||||||
BorderSide(color: Theme.of(context).colorScheme.secondary)),
|
|
||||||
label: Text(dataTypeFriendly[type]!),
|
|
||||||
showCheckmark: false,
|
|
||||||
selected: selectedDataTypes.contains(type),
|
|
||||||
onSelected: (bool selected) {
|
|
||||||
setState(() {
|
|
||||||
if (selected) {
|
|
||||||
selectedDataTypes.add(type);
|
|
||||||
} else {
|
|
||||||
selectedDataTypes.remove(type);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
return chips;
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) {
|
|
||||||
return Wrap(
|
|
||||||
children: filterDataTypeChips(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user