inflearn logo
강의

강의

N
챌린지

챌린지

멘토링

멘토링

N
클립

클립

로드맵

로드맵

지식공유

[코드팩토리] [중급] Flutter 진짜 실전! 상태관리, 캐시관리, Code Generation, GoRouter, 인증로직 등 중수가 되기 위한 필수 스킬들!

일반 Navigation에서 GoRouter로 전환하기

115강 5분 10초, model의 타입을 ProductModel로 지정했을 때 발생하는 타입 및 null 에러가 있습니다

해결된 질문

354

limazero14

작성한 질문수 3

0

class ProductPage extends StatelessWidget {
  const ProductPage({super.key});

  @override
  Widget build(BuildContext context) {
    return PaginationListView<ProductModel>(
      provider: productProvider,
      itemBuilder: <ProductModel>(_, index, ProductModel model) =>
          GestureDetector(
        onTap: () => context.goNamed(
          RestaurantDetailPage.routeName,
          pathParameters: {
            'rid': model.restaurant.id,
          },
        ),
        child: ProductCard.fromProductModel(
          model: model,
        ),
      ),
    );
  }
}

여기서 이상한 에러가 뜹니다.

 

먼저 'rid': model.restaurant.id 이 부분에서는 아래와 같은 에러가 뜹니다.

The property 'restaurant' can't be unconditionally accessed because the receiver can be 'null'.
Try making the access conditional (using '?.') or adding a null check to the target ('!').dartunchecked_use_of_nullable_value

 

그리고 두 번째로, ProductCard.fromProductModel(model: model)에서는 이런 에러가 뜹니다.

The argument type 'ProductModel' can't be assigned to the parameter type 'ProductModel'.dartargument_type_not_assignable

 

두 에러 모두 이해되지 않습니다. ProductModel 타입을 ProductModel 타입으로 Assign할 수 없다니요? 같은 타입인데 이런 에러가 뜹니다.

 

또한 첫 번째 에러의 경우에도, 분명히 nullable 타입이 존재하지 않는데 nullable 체크를 하라고 하고 있습니다. 당황스럽습니다.

 

한편, 강의는 이런 식으로 되어 있습니다.

itemBuilder: <ProductModel>(_, index, model)

model의 타입을 따로 지정해주지 않았습니다. 이렇게 했을경우 model의 타입은 dynamic이 되며, 자동 완성 기능은 수행할 수 없지만, 결론적으로 잘 작동은 합니다.

 

그런데 왜 저런 이상한 에러가 발생하는 지 모르겠습니다.

 

여기 이와 관련된 코드를 덧붙입니다. 그러나 대부분 강의와 동일합니다.

@JsonSerializable()
class ProductModel implements IModelWithId {
  @override
  final String id;

  /// 상품 이름
  final String name;

  /// 상품 상세 정보
  final String detail;

  /// 상품 이미지 URL
  @JsonKey(fromJson: DataUtils.pathToUrl)
  final String imgUrl;

  /// 상품 가격
  final int price;

  /// 레스토랑 정보
  final RestaurantModel restaurant;

  ProductModel({
    required this.id,
    required this.name,
    required this.detail,
    required this.imgUrl,
    required this.price,
    required this.restaurant,
  });

  factory ProductModel.fromJson(Map<String, dynamic> json) =>
      _$ProductModelFromJson(json);
}
typedef PaginationWidgetBuilder<T extends IModelWithId> = Widget Function(
  BuildContext context,
  int index,
  T model,
);

문제가 발생하고 있는 스크린샷:

스크린샷 2023-08-10 오전 7.48.15.png스크린샷 2023-08-10 오전 7.48.36.png

flutter 하이브리드-앱

답변 1

0

코드팩토리

안녕하세요!

혹시 PaginationListView 클래스의 State 제너릭에 타입 선언을 추가해주면 문제가 해결되시나요?

아니라면 다시 질문 주세요!

감사합니다!

0

limazero14

빠른 답변 감사드립니다!

현재 제 PaginationListView 클래스의 상황입니다. 강의랑 최대한 비슷하게 한다고 했는데, 제네릭에서 미스가 났는지도 모르겠네요. 그러나 일단은, 필요한 T는 모두 extends하고 있습니다.

typedef PaginationWidgetBuilder<T extends IModelWithId> = Widget Function(
  BuildContext context,
  int index,
  T model,
);

class PaginationListView<T extends IModelWithId>
    extends ConsumerStatefulWidget {
  final StateNotifierProvider<PaginationProvider, CursorPaginationBase>
      provider;
  final PaginationWidgetBuilder<T> itemBuilder;

  const PaginationListView({
    required this.provider,
    required this.itemBuilder,
    super.key,
  });

  @override
  ConsumerState<PaginationListView> createState() =>
      _PaginationListViewState<T>();
}

class _PaginationListViewState<T extends IModelWithId>
    extends ConsumerState<PaginationListView> {
  final controller = ScrollController();

  @override
  void initState() {
    super.initState();

    controller.addListener(listener);
  }

  void listener() {
    PaginationUtils.paginate(
      controller: controller,
      provider: ref.read(widget.provider.notifier),
    );
  }

  // ... 생략
}


덧붙여서 IModelWithId의 코드도 첨부합니다. 이때 Dart 3의 새로운 Modifier인 interface를 사용했습니다. 이것이 문제가 될 것 같지는 않지만, 혹시 몰라서 알립니다.

 

abstract interface class IModelWithId {
  final String id;

  IModelWithId({
    required this.id,
  });
}

0

코드팩토리

class _PaginationListViewState<T extends IModelWithId> extends ConsumerState<PaginationListView> {

이 부분을

class _PaginationListViewState<T extends IModelWithId> extends ConsumerState<PaginationListView<T>> {

이렇게 해보세요!

0

limazero14

typedef PaginationWidgetBuilder<T extends IModelWithId> = Widget Function(
  BuildContext context,
  int index,
  T model,
);

class PaginationListView<T extends IModelWithId>
    extends ConsumerStatefulWidget {
  final StateNotifierProvider<PaginationProvider, CursorPaginationBase>
      provider;
  final PaginationWidgetBuilder<T> itemBuilder;

  const PaginationListView({
    required this.provider,
    required this.itemBuilder,
    super.key,
  });

  @override
  ConsumerState<PaginationListView> createState() =>
      _PaginationListViewState<T>();
}

class _PaginationListViewState<T extends IModelWithId>
    extends ConsumerState<PaginationListView<T>> {
  final controller = ScrollController();

빠른 답변에 감사드립니다. 그러나 여전히 같은 문제가 발생하고 있습니다.

 

위 코드와 같이, ConsumerState<PaginationListView<T>> 를 받게끔 하고 있음에도 여전히 같은 문제가 발생합니다. Dart 그자체의 문제일까요?

0

코드팩토리

Dart 자체 문제는 아닐겁니다. 레포지토리 공유해주시면 제가 봐볼게요

0

limazero14

직접 봐주신다니 감사합니다! 아래 레포지토리에 해당 강의를 수강하면서 따라한 결과물을 남기고 있습니다: 답변이 해결될 때까지 Public으로 바꿀게요!

 

https://github.com/nx006/delivery_app

1

코드팩토리

안녕하세요! 원인을 찾았습니다. itemBuilder의 generic을 제거해주시면 됩니다. 아래 예제 코드 복붙해드립니다.

return PaginationListView<ProductModel>(
  provider: productProvider,
  itemBuilder: (_, index, ProductModel model) => GestureDetector(
    onTap: () {
      context.goNamed(
        RestaurantDetailPage.routeName,
        pathParameters: {
          'rid': model.restaurant.id,
        },
      );
    },
    child: ProductCard.fromProductModel(
      model: model,
    ),
  ),
);

itemBuilder의 generic에 직접 ProductModel을 넣어줄 경우 PaginationListView로부터 T 타입을 내려받는게 아니라 강제로 ProductModel을 새로 넣어주는거라 인터프레터가 서로 연관성이 없다고 보는 것 같습니다!

0

limazero14

문제가 해결되었습니다! 감사합니다!

Isar 마지막 업데이트는 2년전입니다.

0

31

0

FlutterSecureStorage 질문

0

32

0

Dio onError Interceptor 만드는 부분에 질문이 있습니다.

0

80

2

관리자 기능에 대한 질문

0

100

2

part 'restaurant_model.g.dart';

0

92

1

36강. dio 인터셉터에 storage를 전달하는 코드가 이해 안되는데요. 도움 부탁드립니다.

0

56

2

2번 반환 상황 관련 질문

0

61

2

riverpod 3.0

0

140

2

Asset folder??

0

82

2

디자이너와 협업 시 프레임 크기 설정 관련 질문

0

114

2

FutureProvider, StateNotifierProvider 선택 기준

0

70

2

컴포넌트 모델화

0

64

2

쿼리 파라미터

0

84

2

화면 안보임

0

68

2

PaginationListView

0

54

1

강의중 37.Dio onErrorInterceptor 작업하기 dio 관련 질문입니다.

0

103

2

프로토타입이미지

0

62

2

여러 객체를 상태 관리하는 방법에 대한 질문

0

85

2

장바구니 결제하기 응답이 500이 옵니다.

0

105

2

removeFromBasket에서 await patchBasket()을 마지막에 하면 에러나는거 아닌가요?

0

67

2

이 두가지는 완전히 동일한 기능인가요?

0

106

3

내부 코드를 작성하지 않은 CursorPaginationLoading가 어떻게 로딩상태를 갖는지 잘 모르겠습니다...

0

77

2

_SplashScreenState에서 storage를 late로 호출해서 한번만 불러와도 되나요?

0

86

2

코딩 작성 순서 관련 질문

0

88

2