Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add conversion assists for various widgets, including Hook-based widgets #2306

Merged
merged 30 commits into from
Apr 8, 2023
Merged
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
cd7f967
enable riverpod_lint
K9i-0 Mar 12, 2023
b4fea06
remove unnecessary Consumer
K9i-0 Mar 12, 2023
42f6739
move field
K9i-0 Mar 12, 2023
40f6379
add convert to widget assists
K9i-0 Mar 12, 2023
c8b2e46
implement ConvertToStatelessWidget
K9i-0 Mar 12, 2023
2024145
implement ConvertToStatefulWidget
K9i-0 Mar 12, 2023
38dfc92
fix file name
K9i-0 Mar 12, 2023
6e67631
add current test result
K9i-0 Mar 12, 2023
e08805d
Merge branch 'master' into add_new_lints
K9i-0 Mar 14, 2023
ee9bae7
Implement ConvertToStatefulHookWidget's method
K9i-0 Mar 17, 2023
3ed6c52
Implement ConvertToStatefulHookConsumerWidget's method
K9i-0 Mar 17, 2023
ec624b2
Implement ConvertToHookWidget's method
K9i-0 Mar 17, 2023
6cac1ac
Implement ConvertToHookConsumerWidget's method
K9i-0 Mar 17, 2023
feff63e
Add test
K9i-0 Mar 21, 2023
a29515c
Refactor
K9i-0 Mar 21, 2023
177eda9
remove
K9i-0 Mar 21, 2023
f23df4c
Merge branch 'master' into add_new_lints
K9i-0 Apr 7, 2023
74524df
Display hook-related assists only when there is a dependency on hooks…
K9i-0 Apr 8, 2023
5e50047
Change priorities
K9i-0 Apr 8, 2023
02b77fa
update golden files
K9i-0 Apr 8, 2023
f8f4597
change test file name
K9i-0 Apr 8, 2023
5eb0935
Provide assists for hook-related functions only when there is a depen…
K9i-0 Apr 8, 2023
a224e24
change name
K9i-0 Apr 8, 2023
d9d8a34
Merge branch 'master' of https://github.com/rrousselGit/riverpod into…
rrousselGit Apr 8, 2023
1929cba
Remove dynamic
rrousselGit Apr 8, 2023
0c12926
Update packages/riverpod_lint_flutter_test/test/assists/convert_to_wi…
rrousselGit Apr 8, 2023
c38f878
Use named parameters
K9i-0 Apr 8, 2023
94e67f0
Refactor
rrousselGit Apr 8, 2023
b2c8fb9
Merge branch 'add_new_lints' of https://github.com/k9i-0/riverpod int…
rrousselGit Apr 8, 2023
3975a36
Changelog
rrousselGit Apr 8, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions packages/riverpod_lint/lib/riverpod_lint.dart
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,12 @@ import 'package:custom_lint_builder/custom_lint_builder.dart';

import 'src/assists/convert_to_consumer_stateful_widget.dart';
import 'src/assists/convert_to_consumer_widget.dart';
import 'src/assists/convert_to_hook_consumer_widget.dart';
import 'src/assists/convert_to_hook_widget.dart';
import 'src/assists/convert_to_stateful_hook_consumer_widget.dart';
import 'src/assists/convert_to_stateful_hook_widget.dart';
import 'src/assists/convert_to_stateful_widget.dart';
import 'src/assists/convert_to_stateless_widget.dart';
import 'src/assists/stateful_to_stateless_provider.dart';
import 'src/assists/stateless_to_stateful_provider.dart';
import 'src/assists/wrap_with_consumer.dart';
Expand Down Expand Up @@ -58,6 +64,12 @@ class _RiverpodPlugin extends PluginBase {
WrapWithProviderScope(),
ConvertToConsumerWidget(),
ConvertToConsumerStatefulWidget(),
ConvertToHookWidget(),
ConvertToStatefulHookWidget(),
ConvertToHookConsumerWidget(),
ConvertToStatefulHookConsumerWidget(),
ConvertToStatelessWidget(),
ConvertToStatefulWidget(),

// StateProvider to SyncStatefulProvider
// convert FutureProvider <> AsyncNotifierProvider
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,25 +7,12 @@ import 'package:custom_lint_builder/custom_lint_builder.dart';

import '../object_utils.dart';
import '../riverpod_custom_lint.dart';
import 'convert_to_consumer_widget.dart';

const statefulConvertPriority = convertPriority - 1;

const _statelessBaseType = TypeChecker.any([
TypeChecker.fromName('StatelessWidget', packageName: 'flutter'),
TypeChecker.fromName('ConsumerWidget', packageName: 'flutter_riverpod'),
TypeChecker.fromName('HookConsumerWidget', packageName: 'hooks_riverpod'),
TypeChecker.fromName('HookWidget', packageName: 'flutter_hooks'),
]);

const _statefulBaseType = TypeChecker.any([
TypeChecker.fromName('StatefulWidget', packageName: 'flutter'),
TypeChecker.fromName(
'StatefulHookConsumerWidget',
packageName: 'hooks_riverpod',
),
TypeChecker.fromName('StatefulHookWidget', packageName: 'flutter_hooks'),
]);
import 'convert_to_widget_utils.dart';

const _convertTarget = ConvertToWidget.consumerStatefulWidget;

final _statelessBaseType = getStatelessBaseType(excludes: [_convertTarget]);
final _statefulBaseType = getStatefulBaseType(excludes: [_convertTarget]);

const _stateType = TypeChecker.fromName('State', packageName: 'flutter');

Expand Down Expand Up @@ -68,8 +55,8 @@ class ConvertToConsumerStatefulWidget extends RiverpodAssist {
Source source,
) {
final changeBuilder = reporter.createChangeBuilder(
message: 'Convert to ConsumerStatefulWidget',
priority: statefulConvertPriority,
message: 'Convert to ${_convertTarget.widgetName}',
priority: _convertTarget.priority,
);

changeBuilder.addDartFileEdit((builder) {
Expand Down Expand Up @@ -140,8 +127,8 @@ class ConvertToConsumerStatefulWidget extends RiverpodAssist {
ExtendsClause node,
) {
final changeBuilder = reporter.createChangeBuilder(
message: 'Convert to ConsumerStatefulWidget',
priority: statefulConvertPriority,
message: 'Convert to ${_convertTarget.widgetName}',
priority: _convertTarget.priority,
);

changeBuilder.addDartFileEdit((builder) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,28 +7,12 @@ import 'package:custom_lint_builder/custom_lint_builder.dart';

import '../object_utils.dart';
import '../riverpod_custom_lint.dart';
import 'convert_to_widget_utils.dart';

/// But the priority above everything else
const convertPriority = 100;

const _statelessBaseType = TypeChecker.any([
TypeChecker.fromName('StatelessWidget', packageName: 'flutter'),
TypeChecker.fromName('HookConsumerWidget', packageName: 'hooks_riverpod'),
TypeChecker.fromName('HookWidget', packageName: 'flutter_hooks'),
]);

const _statefulBaseType = TypeChecker.any([
TypeChecker.fromName('StatefulWidget', packageName: 'flutter'),
TypeChecker.fromName(
'ConsumerStatefulWidget',
packageName: 'flutter_riverpod',
),
TypeChecker.fromName(
'StatefulHookConsumerWidget',
packageName: 'hooks_riverpod',
),
TypeChecker.fromName('StatefulHookWidget', packageName: 'flutter_hooks'),
]);
const _convertTarget = ConvertToWidget.consumerWidget;

final _statelessBaseType = getStatelessBaseType(excludes: [_convertTarget]);
final _statefulBaseType = getStatefulBaseType(excludes: [_convertTarget]);

const _stateType = TypeChecker.fromName('State', packageName: 'flutter');

Expand Down Expand Up @@ -67,8 +51,8 @@ class ConvertToConsumerWidget extends RiverpodAssist {
Source source,
) {
final changeBuilder = reporter.createChangeBuilder(
message: 'Convert to ConsumerWidget',
priority: convertPriority,
message: 'Convert to ${_convertTarget.widgetName}',
priority: _convertTarget.priority,
);

changeBuilder.addDartFileEdit((builder) {
Expand Down Expand Up @@ -170,8 +154,8 @@ class ConvertToConsumerWidget extends RiverpodAssist {
ExtendsClause node,
) {
final changeBuilder = reporter.createChangeBuilder(
message: 'Convert to ConsumerWidget',
priority: convertPriority,
message: 'Convert to ${_convertTarget.widgetName}',
priority: _convertTarget.priority,
);

changeBuilder.addDartFileEdit((builder) {
Expand Down
rrousselGit marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,225 @@
import 'package:analyzer/dart/ast/ast.dart';
import 'package:analyzer/source/source_range.dart';
// ignore: implementation_imports, somehow not exported by analyzer
import 'package:analyzer/src/generated/source.dart' show Source;
import 'package:collection/collection.dart';
import 'package:custom_lint_builder/custom_lint_builder.dart';

import '../object_utils.dart';
import '../riverpod_custom_lint.dart';
import 'convert_to_widget_utils.dart';

const _convertTarget = ConvertToWidget.hookConsumerWidget;

final _statelessBaseType = getStatelessBaseType(excludes: [_convertTarget]);
final _statefulBaseType = getStatefulBaseType(excludes: [_convertTarget]);

const _stateType = TypeChecker.fromName('State', packageName: 'flutter');

class ConvertToHookConsumerWidget extends RiverpodAssist {
ConvertToHookConsumerWidget();

@override
void run(
CustomLintResolver resolver,
ChangeReporter reporter,
CustomLintContext context,
SourceRange target,
) {
context.registry.addExtendsClause((node) {
// Only offer the assist if hovering the extended type
if (!node.superclass.sourceRange.intersects(target)) return;

final type = node.superclass.type;
if (type == null) return;

if (_statelessBaseType.isExactlyType(type)) {
_convertStatelessToHookConsumerWidget(reporter, node);
return;
}

if (_statefulBaseType.isExactlyType(type)) {
_convertStatefulToHookConsumerWidget(
reporter,
node,
resolver.source,
);
return;
}
});
}

void _convertStatelessToHookConsumerWidget(
ChangeReporter reporter,
ExtendsClause node,
) {
final changeBuilder = reporter.createChangeBuilder(
message: 'Convert to ${_convertTarget.widgetName}',
priority: _convertTarget.priority,
);

changeBuilder.addDartFileEdit((builder) {
// Change the extended base class
builder.addSimpleReplacement(
node.superclass.sourceRange,
_convertTarget.widgetName,
);

// Now update "build" to take a "ref" parameter
final buildMethod = node
.thisOrAncestorOfType<ClassDeclaration>()
?.members
.whereType<MethodDeclaration>()
.firstWhereOrNull((element) => element.name.lexeme == 'build');

if (buildMethod == null) return;
final buildParams = buildMethod.parameters;
// If there is more than one parameter, the build method already has the "ref"
if (buildParams == null || buildParams.parameters.length != 1) return;

builder.addSimpleInsertion(
buildParams.parameters.last.end,
', WidgetRef ref',
);
});
}

void _convertStatefulToHookConsumerWidget(
ChangeReporter reporter,
ExtendsClause node,
Source source,
) {
final changeBuilder = reporter.createChangeBuilder(
message: 'Convert to ${_convertTarget.widgetName}',
priority: _convertTarget.priority,
);

changeBuilder.addDartFileEdit((builder) {
// Change the extended base class
builder.addSimpleReplacement(
node.superclass.sourceRange,
_convertTarget.widgetName,
);

final widgetClass = node.thisOrAncestorOfType<ClassDeclaration>();
if (widgetClass == null) return;

// Remove createState method
final createStateMethod = widgetClass.members
.whereType<MethodDeclaration>()
.firstWhereOrNull((element) => element.name.lexeme == 'createState');
if (createStateMethod != null) {
builder.addDeletion(createStateMethod.sourceRange);
}

// Search for the associated State class
final stateClass = _findStateClass(widgetClass);
if (stateClass == null) return;

// Move the build method to the widget class

final buildMethod = stateClass.members
.whereType<MethodDeclaration>()
.firstWhereOrNull((element) => element.name.lexeme == 'build');
if (buildMethod == null) return;

final newBuildMethod = _buildMethodWithRef(buildMethod, source);
if (newBuildMethod == null) return;
builder.addSimpleInsertion(
widgetClass.rightBracket.offset,
newBuildMethod,
);

// Delete the state class
builder.addDeletion(stateClass.sourceRange);
});
}

String? _buildMethodWithRef(
MethodDeclaration buildMethod,
Source source,
) {
final parameters = buildMethod.parameters;
if (parameters == null) return null;

if (parameters.parameters.length == 2) {
// The build method already has a ref parameter, nothing to change
return '${source.contents.data.substring(buildMethod.offset, buildMethod.end)}\n';
}

final buffer = StringBuffer();
final refParamStartOffset = parameters.parameters.firstOrNull?.end ??
parameters.leftParenthesis.offset + 1;

buffer
..write(
source.contents.data.substring(buildMethod.offset, refParamStartOffset),
)
..write(', WidgetRef ref')
..writeln(
source.contents.data.substring(refParamStartOffset, buildMethod.end),
);

return buffer.toString();
}

ClassDeclaration? _findStateClass(ClassDeclaration widgetClass) {
final widgetType = widgetClass.declaredElement?.thisType;
if (widgetType == null) return null;

return widgetClass
.thisOrAncestorOfType<CompilationUnit>()
?.declarations
.whereType<ClassDeclaration>()
.where(
// Is the class a state class?
(e) =>
e.extendsClause?.superclass.type
.let(_stateType.isAssignableFromType) ??
false,
)
.firstWhereOrNull((e) {
final stateWidgetType = e
.extendsClause?.superclass.typeArguments?.arguments.firstOrNull?.type;
if (stateWidgetType == null) return false;

final checker = TypeChecker.fromStatic(widgetType);
return checker.isExactlyType(stateWidgetType);
});
}

void _convertStatelessToConsumerWidget(
ChangeReporter reporter,
ExtendsClause node,
) {
final changeBuilder = reporter.createChangeBuilder(
message: 'Convert to ${_convertTarget.widgetName}',
priority: _convertTarget.priority,
);

changeBuilder.addDartFileEdit((builder) {
// Change the extended base class
builder.addSimpleReplacement(
node.superclass.sourceRange,
'ConsumerWidget',
);

// Now update "build" to take a "ref" parameter
final buildMethod = node
.thisOrAncestorOfType<ClassDeclaration>()
?.members
.whereType<MethodDeclaration>()
.firstWhereOrNull((element) => element.name.lexeme == 'build');

if (buildMethod == null) return;
final buildParams = buildMethod.parameters;
// If there is more than one parameter, the build method already has the "ref"
if (buildParams == null || buildParams.parameters.length != 1) return;

builder.addSimpleInsertion(
buildParams.parameters.last.end,
', WidgetRef ref',
);
});
}
}
Loading