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 functions to style text (colors, weight, etc) #140

Open
wants to merge 4 commits into
base: main
Choose a base branch
from

Conversation

gustavothecoder
Copy link

@gustavothecoder gustavothecoder commented Dec 2, 2024

Inspired by the issue #121, I added a module with functions to change text style. The doc docsite/source/styling-your-output.html.md demonstrate how it works. I used the thor implementation as a basis, but I changed the API to make combinations shorter:

# Thor API
set_color "Hi!", :red, :on_white, :bold
# Implemented API
red(on_white(bold("Hi!")))

Resolve #121

@gustavothecoder
Copy link
Author

Maybe this API is a better idea because it looks more like Ruby:

stylize("Hi!").red.on_white.bold

In this case, stylize will return an object with methods to style the text.

@cllns
Copy link
Member

cllns commented Dec 5, 2024

Maybe this API is a better idea because it looks more like Ruby:

stylize("Hi!").red.on_white.bold

In this case, stylize will return an object with methods to style the text.

First of all, thanks for doing this!

When I first read this PR, I also thought of this API :)

Want to go for it?

Also, can you extract some constants like BLUE = 34 and refactor from based on that?

@gustavothecoder
Copy link
Author

Maybe this API is a better idea because it looks more like Ruby:

stylize("Hi!").red.on_white.bold

In this case, stylize will return an object with methods to style the text.

First of all, thanks for doing this!

When I first read this PR, I also thought of this API :)

Want to go for it?

Also, can you extract some constants like BLUE = 34 and refactor from based on that?

Yeah, I'll let you know when I'm done refactoring.

@gustavothecoder gustavothecoder force-pushed the add-colors branch 2 times, most recently from c5934e3 to ad75bc1 Compare December 8, 2024 12:58
@gustavothecoder
Copy link
Author

Hey @cllns, I'm done refactoring and I think that the solution is better now, what you think?

image

Copy link
Member

@cllns cllns left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for that refactoring! In reviewing it, I found a few more we can do as well

lib/dry/cli/styles.rb Show resolved Hide resolved
lib/dry/cli/styles.rb Outdated Show resolved Hide resolved
lib/dry/cli/styles.rb Outdated Show resolved Hide resolved
@gustavothecoder
Copy link
Author

Thanks for that refactoring! In reviewing it, I found a few more we can do as well

Thanks for the feedback! Can you review it again, please?

@cllns
Copy link
Member

cllns commented Jan 3, 2025

Sweet, thanks!

Can you try to use recursion instead of mutation? In dry-rb, we prefer to leverage immutability, as we find it easier to reason about.

To spell it out, calling style("hello") would return a Dry::CLI::Styles::StyledText instance, then each call will wrap that in another Dry::CLI::Styles::StyledText instance, and so on. Then when .to_s is called, it adds the shell escape code and calls .to_s on its contents.

The StyledText can also hold the @escape_code. For the first call to style, that value will be nil and we can use that as a convention to throw the clear/reset at the end.

Does that make sense?

- Changed the API to `stylize("string").bold.blue.on_white`
- Moved magic numbers to constants
- Improve code reuse and legibility
- Change `StyledText` to be immutable
@gustavothecoder
Copy link
Author

Sweet, thanks!

Can you try to use recursion instead of mutation? In dry-rb, we prefer to leverage immutability, as we find it easier to reason about.

To spell it out, calling style("hello") would return a Dry::CLI::Styles::StyledText instance, then each call will wrap that in another Dry::CLI::Styles::StyledText instance, and so on. Then when .to_s is called, it adds the shell escape code and calls .to_s on its contents.

The StyledText can also hold the @escape_code. For the first call to style, that value will be nil and we can use that as a convention to throw the clear/reset at the end.

Does that make sense?

It does, the code is simpler now:

def chainable_update!(style, new_text)
  StyledText.new(
    select_graphic_rendition(style) + new_text,
    select_graphic_rendition(RESET)
  )
end

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Feature request: colorized output
2 participants