Skip to content

Commit bdb52a7

Browse files
authored
refactor: categories to first class objects (#1348)
1 parent 2fafc08 commit bdb52a7

File tree

68 files changed

+484
-419
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

68 files changed

+484
-419
lines changed

flutter_news_example/api/lib/src/client/flutter_news_example_api_client.dart

+3-3
Original file line numberDiff line numberDiff line change
@@ -155,18 +155,18 @@ class FlutterNewsExampleApiClient {
155155
/// Requests news feed metadata.
156156
///
157157
/// Supported parameters:
158-
/// * [category] - The desired news [Category].
158+
/// * [categoryId] - The desired id of news category.
159159
/// * [limit] - The number of results to return.
160160
/// * [offset] - The (zero-based) offset of the first item
161161
/// in the collection to return.
162162
Future<FeedResponse> getFeed({
163-
Category? category,
163+
String? categoryId,
164164
int? limit,
165165
int? offset,
166166
}) async {
167167
final uri = Uri.parse('$_baseUrl/api/v1/feed').replace(
168168
queryParameters: <String, String>{
169-
if (category != null) 'category': category.name,
169+
if (categoryId != null) 'category': categoryId,
170170
if (limit != null) 'limit': '$limit',
171171
if (offset != null) 'offset': '$offset',
172172
},

flutter_news_example/api/lib/src/data/in_memory_news_data_source.dart

+2-2
Original file line numberDiff line numberDiff line change
@@ -102,12 +102,12 @@ class InMemoryNewsDataSource implements NewsDataSource {
102102

103103
@override
104104
Future<Feed> getFeed({
105-
Category category = Category.top,
105+
required String categoryId,
106106
int limit = 20,
107107
int offset = 0,
108108
}) async {
109109
final feed =
110-
_newsFeedData[category] ?? const Feed(blocks: [], totalBlocks: 0);
110+
_newsFeedData[categoryId] ?? const Feed(blocks: [], totalBlocks: 0);
111111
final totalBlocks = feed.totalBlocks;
112112
final normalizedOffset = math.min(offset, totalBlocks);
113113
final blocks = feed.blocks.sublist(normalizedOffset).take(limit).toList();

flutter_news_example/api/lib/src/data/news_data_source.dart

+2-3
Original file line numberDiff line numberDiff line change
@@ -59,8 +59,7 @@ abstract class NewsDataSource {
5959
int offset = 0,
6060
});
6161

62-
/// Returns a news [Feed] for the provided [category].
63-
/// By default [Category.top] is used.
62+
/// Returns a news [Feed] for the provided [categoryId].
6463
///
6564
/// In addition, the feed can be paginated by supplying
6665
/// [limit] and [offset].
@@ -69,7 +68,7 @@ abstract class NewsDataSource {
6968
/// * [offset] - The (zero-based) offset of the first item
7069
/// in the collection to return.
7170
Future<Feed> getFeed({
72-
Category category = Category.top,
71+
required String categoryId,
7372
int limit = 20,
7473
int offset = 0,
7574
});

flutter_news_example/api/lib/src/data/static_news_data.dart

+11-11
Original file line numberDiff line numberDiff line change
@@ -1383,19 +1383,19 @@ List<NewsItem> get _newsItems {
13831383
];
13841384
}
13851385

1386-
final _newsFeedData = <Category, Feed>{
1387-
_topCategory: topNewsFeedBlocks.toFeed(),
1388-
_technologyCategory: technologyFeedBlocks.toFeed(),
1389-
_sportsCategory: sportsFeedBlocks.toFeed(),
1390-
_healthCategory: healthFeedBlocks.toFeed(),
1391-
_scienceCategory: scienceFeedBlocks.toFeed(),
1386+
final _newsFeedData = <String, Feed>{
1387+
_topCategory.id: topNewsFeedBlocks.toFeed(),
1388+
_technologyCategory.id: technologyFeedBlocks.toFeed(),
1389+
_sportsCategory.id: sportsFeedBlocks.toFeed(),
1390+
_healthCategory.id: healthFeedBlocks.toFeed(),
1391+
_scienceCategory.id: scienceFeedBlocks.toFeed(),
13921392
};
13931393

1394-
const _topCategory = Category.top;
1395-
const _sportsCategory = Category.sports;
1396-
const _technologyCategory = Category.technology;
1397-
const _healthCategory = Category.health;
1398-
const _scienceCategory = Category.science;
1394+
const _topCategory = Category(id: 'top', name: 'Top');
1395+
const _sportsCategory = Category(id: 'sports', name: 'Sports');
1396+
const _technologyCategory = Category(id: 'technology', name: 'Technology');
1397+
const _healthCategory = Category(id: 'health', name: 'Health');
1398+
const _scienceCategory = Category(id: 'science', name: 'Science');
13991399

14001400
const _categories = [
14011401
_topCategory,

flutter_news_example/api/lib/src/models/categories_response/categories_response.dart

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ part 'categories_response.g.dart';
77
/// {@template categories_response}
88
/// A news categories response object which contains available news categories.
99
/// {@endtemplate}
10-
@JsonSerializable()
10+
@JsonSerializable(explicitToJson: true)
1111
class CategoriesResponse extends Equatable {
1212
/// {@macro categories_response}
1313
const CategoriesResponse({required this.categories});

flutter_news_example/api/lib/src/models/categories_response/categories_response.g.dart

+2-13
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

flutter_news_example/api/packages/news_blocks/lib/src/block_action.g.dart

+2-12
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -1,30 +1,33 @@
1-
/// The supported news category types.
2-
enum Category {
3-
/// News relating to business.
4-
business,
5-
6-
/// News relating to entertainment.
7-
entertainment,
8-
9-
/// Breaking news.
10-
top,
11-
12-
/// News relating to health.
13-
health,
14-
15-
/// News relating to science.
16-
science,
17-
18-
/// News relating to sports.
19-
sports,
20-
21-
/// News relating to technology.
22-
technology;
23-
24-
/// The id of the category.
25-
String get id => name;
26-
27-
/// Returns a [Category] for the [categoryName].
28-
static Category fromString(String categoryName) =>
29-
Category.values.firstWhere((category) => category.name == categoryName);
1+
import 'package:equatable/equatable.dart';
2+
import 'package:json_annotation/json_annotation.dart';
3+
4+
part 'category.g.dart';
5+
6+
/// {@template category}
7+
/// Represents a news category.
8+
/// {@endtemplate}
9+
@JsonSerializable()
10+
class Category extends Equatable {
11+
/// {@macro category}
12+
const Category({
13+
required this.id,
14+
required this.name,
15+
});
16+
17+
/// Converts a `Map<String, dynamic>` into
18+
/// a [Category] instance.
19+
factory Category.fromJson(Map<String, dynamic> json) =>
20+
_$CategoryFromJson(json);
21+
22+
/// Category id.
23+
final String id;
24+
25+
/// Category name.
26+
final String name;
27+
28+
/// Converts the current instance to a `Map<String, dynamic>`.
29+
Map<String, dynamic> toJson() => _$CategoryToJson(this);
30+
31+
@override
32+
List<Object?> get props => [id, name];
3033
}

flutter_news_example/api/packages/news_blocks/lib/src/category.g.dart

+26
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

flutter_news_example/api/packages/news_blocks/test/src/article_introduction_block_test.dart

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ import 'package:test/test.dart';
44
void main() {
55
group('ArticleIntroductionBlock', () {
66
test('can be (de)serialized', () {
7-
const category = Category.technology;
7+
const category = Category(id: 'technology', name: 'Technology');
88
final block = ArticleIntroductionBlock(
99
categoryId: category.id,
1010
author: 'author',

flutter_news_example/api/packages/news_blocks/test/src/block_action_converter_test.dart

+2-1
Original file line numberDiff line numberDiff line change
@@ -7,10 +7,11 @@ void main() {
77
group('BlockActionConverter', () {
88
test('can (de)serialize BlockAction', () {
99
final converter = BlockActionConverter();
10+
const category = Category(id: 'sports', name: 'Sports');
1011

1112
const actions = <BlockAction>[
1213
NavigateToArticleAction(articleId: 'articleId'),
13-
NavigateToFeedCategoryAction(category: Category.top),
14+
NavigateToFeedCategoryAction(category: category),
1415
];
1516

1617
for (final action in actions) {

flutter_news_example/api/packages/news_blocks/test/src/block_action_test.dart

+6-2
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,9 @@ void main() {
4646
});
4747

4848
test('returns NavigateToFeedCategoryAction', () {
49-
final action = NavigateToFeedCategoryAction(category: Category.top);
49+
const category = Category(id: 'sports', name: 'Sports');
50+
51+
final action = NavigateToFeedCategoryAction(category: category);
5052
expect(BlockAction.fromJson(action.toJson()), equals(action));
5153
});
5254

@@ -91,7 +93,9 @@ void main() {
9193

9294
group('NavigateToFeedCategoryAction', () {
9395
test('can be (de)serialized', () {
94-
final action = NavigateToFeedCategoryAction(category: Category.top);
96+
const category = Category(id: 'sports', name: 'Sports');
97+
98+
final action = NavigateToFeedCategoryAction(category: category);
9599
expect(
96100
NavigateToFeedCategoryAction.fromJson(action.toJson()),
97101
equals(action),

flutter_news_example/api/packages/news_blocks/test/src/category_test.dart

+14-49
Original file line numberDiff line numberDiff line change
@@ -2,56 +2,21 @@ import 'package:news_blocks/news_blocks.dart';
22
import 'package:test/test.dart';
33

44
void main() {
5-
group('Category', () {
6-
group('fromString', () {
7-
test('returns business', () {
8-
expect(
9-
Category.fromString('business'),
10-
equals(Category.business),
11-
);
12-
});
13-
14-
test('returns business', () {
15-
expect(
16-
Category.fromString('entertainment'),
17-
equals(Category.entertainment),
18-
);
19-
});
20-
21-
test('returns top', () {
22-
expect(
23-
Category.fromString('top'),
24-
equals(Category.top),
25-
);
26-
});
27-
28-
test('returns health', () {
29-
expect(
30-
Category.fromString('health'),
31-
equals(Category.health),
32-
);
33-
});
34-
35-
test('returns science', () {
36-
expect(
37-
Category.fromString('science'),
38-
equals(Category.science),
39-
);
40-
});
41-
42-
test('returns sports', () {
43-
expect(
44-
Category.fromString('sports'),
45-
equals(Category.sports),
46-
);
47-
});
5+
group(Category, () {
6+
test('can be (de)serialized', () {
7+
const category = Category(id: 'sports', name: 'Sports');
8+
expect(
9+
Category.fromJson(const {'id': 'sports', 'name': 'Sports'}),
10+
equals(category),
11+
);
12+
});
4813

49-
test('returns technology', () {
50-
expect(
51-
Category.fromString('technology'),
52-
equals(Category.technology),
53-
);
54-
});
14+
test('can be serialized', () {
15+
const category = Category(id: 'sports', name: 'Sports');
16+
expect(
17+
category.toJson(),
18+
equals(const {'id': 'sports', 'name': 'Sports'}),
19+
);
5520
});
5621
});
5722
}

0 commit comments

Comments
 (0)