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

Flutter lint rule to prefer Text.rich over RichText #60334

Open
5 tasks done
IchordeDionysos opened this issue Mar 17, 2025 · 3 comments
Open
5 tasks done

Flutter lint rule to prefer Text.rich over RichText #60334

IchordeDionysos opened this issue Mar 17, 2025 · 3 comments
Labels
area-devexp For issues related to the analysis server, IDE support, linter, `dart fix`, and diagnostic messages. devexp-linter Issues with the analyzer's support for the linter package linter-lint-request P3 A lower priority bug or feature request

Comments

@IchordeDionysos
Copy link

IchordeDionysos commented Mar 17, 2025

prefer_text_over_rich_text

Description

Avoid using Text.rich() over RichText.

Details

RichText does not automatically scale it's font size based on device text scale settings.
Rich.text does adjust to the text scale factor settings on the device making it the preferred option.

Kind

Helps engineers build accessible Flutter apps.

Bad Examples

Widget build(BuildContext context) {
  return RichText(
    text: TextSpan(
      text: "Here is some text.",
      style: Theme.of(context).textTheme.bodyText2,
    ),
  );
}
Widget build(BuildContext context) {
  return RichText(
    text: TextSpan(
      text: "Here is some text.",
      style: Theme.of(context).textTheme.bodyText2,
      children: [
        TextSpan(
          text: " (some more text)",
          style: Theme.of(context).textTheme.bodyText1,
        ),
      ],
    ),
  );
}

Good Examples

Widget build(BuildContext context) {
  return Text.rich(
    TextSpan(
      text: "Here is some text.",
      style: Theme.of(context).textTheme.bodyText2,
    ),
  );
}
Widget build(BuildContext context) {
  return Text.rich(
    TextSpan(
      text: "Here is some text.",
      style: Theme.of(context).textTheme.bodyText2,
      children: [
        TextSpan(
          text: " (some more text)",
          style: Theme.of(context).textTheme.bodyText1,
        ),
      ],
    ),
  );
}

Discussion

From a accessibility-by-design/accessibility-by-default perspective, this lint would improve the developers and users experiences greatly.

There may be good reasons to use RichText if you need full control over it. However, in the majority of the cases Text.rich is the preferred option.
Engineers may not be aware of this fact, making this lint a great option to help engineers build accessible apps.

We have faced this issues in an internal app, and instead of having to educate people not to use RichText or to keep an eye out for it in PRs.
Having this automatically checked would be super helpful.

Reference: flutter/flutter#61452 (comment)

Discussion checklist

  • List any existing rules this proposal modifies, complements, overlaps or conflicts with.
  • List any relevant issues (reported here, the SDK Tracker, or elsewhere).
  • If there's any prior art (e.g., in other linters), please add references here.
  • If this proposal corresponds to Effective Dart or Flutter Style Guide advice, please call it out. (If there isn't any corresponding advice, should there be?)
  • If this proposal is motivated by real-world examples, please provide as many details as you can. Demonstrating potential impact is especially valuable.
@IchordeDionysos IchordeDionysos added the area-analyzer Use area-analyzer for Dart analyzer issues, including the analysis server and code completion. label Mar 17, 2025
@johnniwinther johnniwinther added area-devexp For issues related to the analysis server, IDE support, linter, `dart fix`, and diagnostic messages. and removed area-analyzer Use area-analyzer for Dart analyzer issues, including the analysis server and code completion. labels Mar 17, 2025
@bwilkerson bwilkerson added devexp-linter Issues with the analyzer's support for the linter package linter-lint-request labels Mar 17, 2025
@chunhtai
Copy link

yeah, I can see the this lint to be useful, but we shouldn't add this to the flutter_lints since the Text.rich isn't a complete replacement for RichText.

@bwilkerson
Copy link
Member

I assume that you're rightly concerned about false positives.

Could we statically determine when it's necessary to use RichText in order to remove false positives? If not, then I suspect that we won't be adding this lint at this time.

@pq pq added the P3 A lower priority bug or feature request label Mar 17, 2025
@IchordeDionysos
Copy link
Author

@chunhtai correct me if I'm wrong but there is only one property unique to RichText and that's selectionRegistrar.

If we would detect whether someone is using that property and then not suggesting to use Text.rich would you be happy to add that lint to flutter_lints?

The other differences is that RichText provides defaults and it actually provides defaults like TextScaler.noScaling breaking accessibility if not considered with care, which is exactly why I'm proposing this lint rule.

Are there other legitimate use cases besides selectionRegistrar?
Could an option be that a user of RichText explicitly acknowledges the drawbacks of RichText by ignoring the lint?


RichText({
  required InlineSpan text,
  Key? key,
  TextAlign textAlign = TextAlign.start,
  TextDirection? textDirection,
  bool softWrap = true,
  TextOverflow overflow = TextOverflow.clip,
  double textScaleFactor = 1.0,
  TextScaler textScaler = TextScaler.noScaling,
  int? maxLines,
  Locale? locale,
  StrutStyle? strutStyle,
  TextWidthBasis textWidthBasis = TextWidthBasis.parent,
  TextHeightBehavior? textHeightBehavior,
  Color? selectionColor,

  // Unique for RichText
  SelectionRegistrar? selectionRegistrar,
})
const Text.rich(
  InlineSpan textSpan, {
  Key? key,
  TextAlign? textAlign, // different default
  TextDirection? textDirection,
  bool? softWrap, // different default
  TextOverflow? overflow, // different default
  double? textScaleFactor, // different default
  TextScaler? textScaler, // different default
  int? maxLines,
  Locale? locale,
  StrutStyle? strutStyle,
  TextWidthBasis? textWidthBasis,  // different default
  TextHeightBehavior? textHeightBehavior,
  Color? selectionColor,

  // Unique for Text.rich
  TextStyle? style,
  String? semanticsLabel,
})

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area-devexp For issues related to the analysis server, IDE support, linter, `dart fix`, and diagnostic messages. devexp-linter Issues with the analyzer's support for the linter package linter-lint-request P3 A lower priority bug or feature request
Projects
None yet
Development

No branches or pull requests

5 participants