We’re really excited for you to try Lip Gloss v2! Keep in mind that this is an early alpha release and things may change.
[!NOTE]
We take API changes seriously and strive to make the upgrade process as simple as possible. We believe the changes bring necessary improvements as well as pave the way for the future. If something feels way off, let us know.
The big changes are that Styles are now deterministic (λipgloss!) and you can be much more intentional with your inputs and outputs. Why does this matter?
Playing nicely with others
v2 gives you precise control over I/O. One of the issues we saw with the Lip Gloss and Bubble Tea v1s is that they could fight over the same inputs and outputs, producing lock-ups. The v2s now operate in lockstep.
Querying the right inputs and outputs
In v1, Lip Gloss defaulted to looking at stdin and when downsampling colors and querying for the background color. This was not always necessarily what you wanted. For example, if your application was writing to while redirecting to a file, the program would erroneously think output was not a TTY and strip colors. Lip Gloss v2 gives you control and intentionality over this.
stdout
stderr
stdout
Going beyond localhost
Did you know TUIs and CLIs can be served over the network? For example, Wish allows you to serve Bubble Tea and Lip Gloss over SSH. In these cases, you need to work with the input and output of the connected clients as opposed to stdin and stdout, which belong to the server. Lip Gloss v2 gives you flexibility around this in a more natural way.
🧋 Using Lip Gloss with Bubble Tea?
Make sure you get all the latest v2s as they’ve been designed to work together.
go get github.com/charmbracelet/bubbletea/v2@v2.0.0-alpha.2
go get github.com/charmbracelet/bubbles/v2@v2.0.0-alpha.2
go get github.com/charmbracelet/lipgloss/v2@v2.0.0-alpha.2
🐇 Quick upgrade
If you don't have time for changes and just want to upgrade to Lip Gloss v2 as fast as possible, do the following:
Use the compat package
The compat package provides adaptive colors, complete colors, and complete adaptive colors:
import "github.com/charmbracelet/lipgloss/v2/compat"
// Before
color := lipgloss.AdaptiveColor{Light: "#f1f1f1", Dark: "#cccccc"}
// After
color := compat.AdaptiveColor{Light: "#f1f1f1", Dark: "#cccccc"}
compat works by looking at stdin and stdout on a global basis. Want to change the inputs and outputs? Knock yourself out:
If you’re using Bubble Tea with Lip Gloss you can skip this step. If you're using Lip Gloss in a standalone fashion, use lipgloss.Println (and lipgloss.Printf and so on) when printing your output:
s := someStyle.Render("Fancy Lip Gloss Output")
// Before
fmt.Println(s)
// After
lipgloss.Println(s)
That’s it!
All this said, we encourage you to read on to get the full benefit of v2.
👀 What’s changing?
Only a couple main things that are changing in Lip Gloss v2:
Color downsampling in non-Bubble-Tea uses cases is now a manual proccess (don't worry, it's easy)
Background color detection and adaptive colors are manual, and intentional (but optional)
🪄 Downsampling colors with a writer
One of the best things about Lip Gloss is that it can automatically downsample colors to the best available profile, stripping colors (and ANSI) entirely when output is not a TTY.
If you're using Lip Gloss with Bubble Tea there's nothing to do here: downsampling is built into Bubble Tea v2. If you're not using Bubble Tea you now need to use a writer to downsample colors. Lip Gloss writers are a drop-in replacement for the usual functions found in the fmt package:
s := someStyle.Render("Hello!")
// Downsample and print to stdout.
lipgloss.Println(s)
// Render to a variable.
downsampled := lipgloss.Sprint(s)
// Print to stderr.
lipgloss.Fprint(os.Stderr, s)
🌛 Background color detection and adaptive colors
Rendering different colors depending on whether the terminal has a light or dark background is an awesome power. Lip Gloss v2 gives you more control over this progress. This especially matters when input and output are not stdin and stdout.
If that doesn’t matter to you and you're only working with stdout you skip this via compat above, though encourage you to explore this new functionality.
With Bubble Tea
In Bubble Tea, request the background color, listen for a BackgroundColorMsg in your update, and respond accordingly.
// Query for the background color.
func (m model) Init() (tea.Model, tea.Cmd) {
return m, tea.RequestBackgroundColor
}
// Listen for the response and initialize your styles accordigly.
func (m model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
switch msg := msg.(type) {
case tea.BackgroundColorMsg:
// Initialize your styles now that you know the background color.
m.styles = newStyles(msg.IsDark())
return m, nil
}
}
type styles {
myHotStyle lipgloss.Style
}
func newStyles(bgIsDark bool) (s styles) {
lightDark := lipgloss.LightDark(bgIsDark) // just a helper function
return styles{
myHotStyle := lipgloss.NewStyle().Foreground(lightDark("#f1f1f1", "#333333"))
}
}
Standalone
If you're not using Bubble Tea you simply can perform the query manually:
// Detect the background color. Notice we're writing to stderr.
hasDarkBG, err := lipgloss.HasDarkBackground(os.Stdin, os.Stderr)
if err != nil {
log.Fatal("Oof:", err)
}
// Create a helper for choosing the appropriate color.
lightDark := lipgloss.LightDark(hasDarkBG)
// Declare some colors.
thisColor := lightDark("#C5ADF9", "#864EFF")
thatColor := lightDark("#37CD96", "#22C78A")
// Render some styles.
a := lipgloss.NewStyle().Foreground(thisColor).Render("this")
b := lipgloss.NewStyle().Foreground(thatColor).Render("that")
// Print to stderr.
lipgloss.Fprintf(os.Stderr, "my fave colors are %s and %s...for now.", a, b)
🥕 Other stuff
Colors are now color.Color
lipgloss.Color() now produces an idomatic color.Color, whereas before colors were type lipgloss.TerminalColor. Generally speaking, this is more of an implementation detail, but it’s worth noting the structural differences.
// Before
type TerminalColor interface{/* ... */}
type Color string
// After
func Color(any) color.Color
type ANSIColor uint
type RGBColor struct { R, G, B uint8 }
Quotes are now optional in colors
There are also some quality-of-life niceties around color UX:
a := lipgloss.Color("#f1f1f1") // This still works
b := lipgloss.Color(0xf1f1f1) // But this also works
c := lipgloss.Color("212") // You can still do this
d := lipgloss.Color(212) // But you can also do this too
Changelog
(v2) adaptive colors + writers by @meowgorithm in https://github.com/charmbracelet/lipgloss/pull/397
(v2) feat: add adaptive color package by @aymanbagabas in https://github.com/charmbracelet/lipgloss/pull/359
chore: rename LightDark to Adapt per @bashbunni's acute suggestion by @meowgorithm in https://github.com/charmbracelet/lipgloss/pull/392
refactor: unexport isDarkColor helper by @bashbunni in https://github.com/charmbracelet/lipgloss/pull/410
(v2) fix: query both stdin and stdout for background color on non-Windows … by @aymanbagabas in https://github.com/charmbracelet/lipgloss/pull/416
Sync golangci-lint config by @github-actions in https://github.com/charmbracelet/lipgloss/pull/421
Sync golangci-lint config by @github-actions in https://github.com/charmbracelet/lipgloss/pull/422
chore(lint): update soft lint directives; fix soft lint issues by @meowgorithm in https://github.com/charmbracelet/lipgloss/pull/423
V2 examples by @meowgorithm in https://github.com/charmbracelet/lipgloss/pull/426
fix: manually query terminal for background color by @aymanbagabas in https://github.com/charmbracelet/lipgloss/pull/429
(v2) feat: complete color support by @aymanbagabas in https://github.com/charmbracelet/lipgloss/pull/420
(v2) feat: add compat package by @aymanbagabas in https://github.com/charmbracelet/lipgloss/pull/419
Full Changelog: https://github.com/charmbracelet/lipgloss/compare/v1.0.0...v2.0.0-alpha.2
🌈 Feedback
That's a wrap! Feel free to reach out, ask questions, and let us know how it's going. We'd love to know what you think.