| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492 |
- package widget
- import (
- "image/color"
- "net/url"
- "strconv"
- "fyne.io/fyne/v2"
- "fyne.io/fyne/v2/canvas"
- "fyne.io/fyne/v2/internal"
- "fyne.io/fyne/v2/theme"
- )
- var (
- // RichTextStyleBlockquote represents a quote presented in an indented block.
- //
- // Since: 2.1
- RichTextStyleBlockquote = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: false,
- SizeName: theme.SizeNameText,
- TextStyle: fyne.TextStyle{Italic: true},
- }
- // RichTextStyleCodeBlock represents a code blog segment.
- //
- // Since: 2.1
- RichTextStyleCodeBlock = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: false,
- SizeName: theme.SizeNameText,
- TextStyle: fyne.TextStyle{Monospace: true},
- }
- // RichTextStyleCodeInline represents an inline code segment.
- //
- // Since: 2.1
- RichTextStyleCodeInline = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: true,
- SizeName: theme.SizeNameText,
- TextStyle: fyne.TextStyle{Monospace: true},
- }
- // RichTextStyleEmphasis represents regular text with emphasis.
- //
- // Since: 2.1
- RichTextStyleEmphasis = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: true,
- SizeName: theme.SizeNameText,
- TextStyle: fyne.TextStyle{Italic: true},
- }
- // RichTextStyleHeading represents a heading text that stands on its own line.
- //
- // Since: 2.1
- RichTextStyleHeading = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: false,
- SizeName: theme.SizeNameHeadingText,
- TextStyle: fyne.TextStyle{Bold: true},
- }
- // RichTextStyleInline represents standard text that can be surrounded by other elements.
- //
- // Since: 2.1
- RichTextStyleInline = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: true,
- SizeName: theme.SizeNameText,
- }
- // RichTextStyleParagraph represents standard text that should appear separate from other text.
- //
- // Since: 2.1
- RichTextStyleParagraph = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: false,
- SizeName: theme.SizeNameText,
- }
- // RichTextStylePassword represents standard sized text where the characters are obscured.
- //
- // Since: 2.1
- RichTextStylePassword = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: true,
- SizeName: theme.SizeNameText,
- concealed: true,
- }
- // RichTextStyleStrong represents regular text with a strong emphasis.
- //
- // Since: 2.1
- RichTextStyleStrong = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: true,
- SizeName: theme.SizeNameText,
- TextStyle: fyne.TextStyle{Bold: true},
- }
- // RichTextStyleSubHeading represents a sub-heading text that stands on its own line.
- //
- // Since: 2.1
- RichTextStyleSubHeading = RichTextStyle{
- ColorName: theme.ColorNameForeground,
- Inline: false,
- SizeName: theme.SizeNameSubHeadingText,
- TextStyle: fyne.TextStyle{Bold: true},
- }
- )
- // HyperlinkSegment represents a hyperlink within a rich text widget.
- //
- // Since: 2.1
- type HyperlinkSegment struct {
- Alignment fyne.TextAlign
- Text string
- URL *url.URL
- }
- // Inline returns true as hyperlinks are inside other elements.
- func (h *HyperlinkSegment) Inline() bool {
- return true
- }
- // Textual returns the content of this segment rendered to plain text.
- func (h *HyperlinkSegment) Textual() string {
- return h.Text
- }
- // Visual returns the hyperlink widget required to render this segment.
- func (h *HyperlinkSegment) Visual() fyne.CanvasObject {
- link := NewHyperlink(h.Text, h.URL)
- link.Alignment = h.Alignment
- return &fyne.Container{Layout: &unpadTextWidgetLayout{}, Objects: []fyne.CanvasObject{link}}
- }
- // Update applies the current state of this hyperlink segment to an existing visual.
- func (h *HyperlinkSegment) Update(o fyne.CanvasObject) {
- link := o.(*fyne.Container).Objects[0].(*Hyperlink)
- link.Text = h.Text
- link.URL = h.URL
- link.Alignment = h.Alignment
- link.Refresh()
- }
- // Select tells the segment that the user is selecting the content between the two positions.
- func (h *HyperlinkSegment) Select(begin, end fyne.Position) {
- // no-op: this will be added when we progress to editor
- }
- // SelectedText should return the text representation of any content currently selected through the Select call.
- func (h *HyperlinkSegment) SelectedText() string {
- // no-op: this will be added when we progress to editor
- return ""
- }
- // Unselect tells the segment that the user is has cancelled the previous selection.
- func (h *HyperlinkSegment) Unselect() {
- // no-op: this will be added when we progress to editor
- }
- // ImageSegment represents an image within a rich text widget.
- //
- // Since: 2.3
- type ImageSegment struct {
- Source fyne.URI
- Title string
- }
- // Inline returns false as images in rich text are blocks.
- func (i *ImageSegment) Inline() bool {
- return false
- }
- // Textual returns the content of this segment rendered to plain text.
- func (i *ImageSegment) Textual() string {
- return "Image " + i.Title
- }
- // Visual returns the image widget required to render this segment.
- func (i *ImageSegment) Visual() fyne.CanvasObject {
- return newRichImage(i.Source)
- }
- // Update applies the current state of this image segment to an existing visual.
- func (i *ImageSegment) Update(o fyne.CanvasObject) {
- newer := canvas.NewImageFromURI(i.Source)
- img := o.(*richImage)
- // one of the following will be used
- img.img.File = newer.File
- img.img.Resource = newer.Resource
- img.Refresh()
- }
- // Select tells the segment that the user is selecting the content between the two positions.
- func (i *ImageSegment) Select(begin, end fyne.Position) {
- // no-op: this will be added when we progress to editor
- }
- // SelectedText should return the text representation of any content currently selected through the Select call.
- func (i *ImageSegment) SelectedText() string {
- // no-op: images have no text rendering
- return ""
- }
- // Unselect tells the segment that the user is has cancelled the previous selection.
- func (i *ImageSegment) Unselect() {
- // no-op: this will be added when we progress to editor
- }
- // ListSegment includes an itemised list with the content set using the Items field.
- //
- // Since: 2.1
- type ListSegment struct {
- Items []RichTextSegment
- Ordered bool
- }
- // Inline returns false as a list should be in a block.
- func (l *ListSegment) Inline() bool {
- return false
- }
- // Segments returns the segments required to draw bullets before each item
- func (l *ListSegment) Segments() []RichTextSegment {
- out := make([]RichTextSegment, len(l.Items))
- for i, in := range l.Items {
- txt := "• "
- if l.Ordered {
- txt = strconv.Itoa(i+1) + "."
- }
- bullet := &TextSegment{Text: txt + " ", Style: RichTextStyleStrong}
- out[i] = &ParagraphSegment{Texts: []RichTextSegment{
- bullet,
- in,
- }}
- }
- return out
- }
- // Textual returns no content for a list as the content is in sub-segments.
- func (l *ListSegment) Textual() string {
- return ""
- }
- // Visual returns no additional elements for this segment.
- func (l *ListSegment) Visual() fyne.CanvasObject {
- return nil
- }
- // Update doesnt need to change a list visual.
- func (l *ListSegment) Update(fyne.CanvasObject) {
- }
- // Select does nothing for a list container.
- func (l *ListSegment) Select(_, _ fyne.Position) {
- }
- // SelectedText returns the empty string for this list.
- func (l *ListSegment) SelectedText() string {
- return ""
- }
- // Unselect does nothing for a list container.
- func (l *ListSegment) Unselect() {
- }
- // ParagraphSegment wraps a number of text elements in a paragraph.
- // It is similar to using a list of text elements when the final style is RichTextStyleParagraph.
- //
- // Since: 2.1
- type ParagraphSegment struct {
- Texts []RichTextSegment
- }
- // Inline returns false as a paragraph should be in a block.
- func (p *ParagraphSegment) Inline() bool {
- return false
- }
- // Segments returns the list of text elements in this paragraph.
- func (p *ParagraphSegment) Segments() []RichTextSegment {
- return p.Texts
- }
- // Textual returns no content for a paragraph container.
- func (p *ParagraphSegment) Textual() string {
- return ""
- }
- // Visual returns the no extra elements.
- func (p *ParagraphSegment) Visual() fyne.CanvasObject {
- return nil
- }
- // Update doesnt need to change a paragraph container.
- func (p *ParagraphSegment) Update(fyne.CanvasObject) {
- }
- // Select does nothing for a paragraph container.
- func (p *ParagraphSegment) Select(_, _ fyne.Position) {
- }
- // SelectedText returns the empty string for this paragraph container.
- func (p *ParagraphSegment) SelectedText() string {
- return ""
- }
- // Unselect does nothing for a paragraph container.
- func (p *ParagraphSegment) Unselect() {
- }
- // SeparatorSegment includes a horizontal separator in a rich text widget.
- //
- // Since: 2.1
- type SeparatorSegment struct {
- //lint:ignore U1000 This is required due to language design.
- dummy uint8 // without this a pointer to SeparatorSegment will always be the same
- }
- // Inline returns false as a separator should be full width.
- func (s *SeparatorSegment) Inline() bool {
- return false
- }
- // Textual returns no content for a separator element.
- func (s *SeparatorSegment) Textual() string {
- return ""
- }
- // Visual returns the separator element for this segment.
- func (s *SeparatorSegment) Visual() fyne.CanvasObject {
- return NewSeparator()
- }
- // Update doesnt need to change a separator visual.
- func (s *SeparatorSegment) Update(fyne.CanvasObject) {
- }
- // Select does nothing for a separator.
- func (s *SeparatorSegment) Select(_, _ fyne.Position) {
- }
- // SelectedText returns the empty string for this separator.
- func (s *SeparatorSegment) SelectedText() string {
- return "" // TODO maybe return "---\n"?
- }
- // Unselect does nothing for a separator.
- func (s *SeparatorSegment) Unselect() {
- }
- // RichTextStyle describes the details of a text object inside a RichText widget.
- //
- // Since: 2.1
- type RichTextStyle struct {
- Alignment fyne.TextAlign
- ColorName fyne.ThemeColorName
- Inline bool
- SizeName fyne.ThemeSizeName
- TextStyle fyne.TextStyle
- // an internal detail where we obscure password fields
- concealed bool
- }
- // RichTextSegment describes any element that can be rendered in a RichText widget.
- //
- // Since: 2.1
- type RichTextSegment interface {
- Inline() bool
- Textual() string
- Update(fyne.CanvasObject)
- Visual() fyne.CanvasObject
- Select(pos1, pos2 fyne.Position)
- SelectedText() string
- Unselect()
- }
- // TextSegment represents the styling for a segment of rich text.
- //
- // Since: 2.1
- type TextSegment struct {
- Style RichTextStyle
- Text string
- }
- // Inline should return true if this text can be included within other elements, or false if it creates a new block.
- func (t *TextSegment) Inline() bool {
- return t.Style.Inline
- }
- // Textual returns the content of this segment rendered to plain text.
- func (t *TextSegment) Textual() string {
- return t.Text
- }
- // Visual returns the graphical elements required to render this segment.
- func (t *TextSegment) Visual() fyne.CanvasObject {
- obj := canvas.NewText(t.Text, t.color())
- t.Update(obj)
- return obj
- }
- // Update applies the current state of this text segment to an existing visual.
- func (t *TextSegment) Update(o fyne.CanvasObject) {
- obj := o.(*canvas.Text)
- obj.Text = t.Text
- obj.Color = t.color()
- obj.Alignment = t.Style.Alignment
- obj.TextStyle = t.Style.TextStyle
- obj.TextSize = t.size()
- obj.Refresh()
- }
- // Select tells the segment that the user is selecting the content between the two positions.
- func (t *TextSegment) Select(begin, end fyne.Position) {
- // no-op: this will be added when we progress to editor
- }
- // SelectedText should return the text representation of any content currently selected through the Select call.
- func (t *TextSegment) SelectedText() string {
- // no-op: this will be added when we progress to editor
- return ""
- }
- // Unselect tells the segment that the user is has cancelled the previous selection.
- func (t *TextSegment) Unselect() {
- // no-op: this will be added when we progress to editor
- }
- func (t *TextSegment) color() color.Color {
- if t.Style.ColorName != "" {
- return fyne.CurrentApp().Settings().Theme().Color(t.Style.ColorName, fyne.CurrentApp().Settings().ThemeVariant())
- }
- return theme.ForegroundColor()
- }
- func (t *TextSegment) size() float32 {
- if t.Style.SizeName != "" {
- return fyne.CurrentApp().Settings().Theme().Size(t.Style.SizeName)
- }
- return theme.TextSize()
- }
- type richImage struct {
- BaseWidget
- img *canvas.Image
- oldMin fyne.Size
- }
- func newRichImage(u fyne.URI) *richImage {
- img := canvas.NewImageFromURI(u)
- img.FillMode = canvas.ImageFillOriginal
- i := &richImage{img: img}
- i.ExtendBaseWidget(i)
- return i
- }
- func (r *richImage) CreateRenderer() fyne.WidgetRenderer {
- return NewSimpleRenderer(r.img)
- }
- func (r *richImage) MinSize() fyne.Size {
- orig := r.img.MinSize()
- c := fyne.CurrentApp().Driver().CanvasForObject(r)
- if c == nil {
- return r.oldMin // not yet rendered
- }
- // unscale the image so it is not varying based on canvas
- w := internal.ScaleInt(c, orig.Width)
- h := internal.ScaleInt(c, orig.Height)
- // we return size / 2 as this assumes a HiDPI / 2x image scaling
- return fyne.NewSize(float32(w)/2, float32(h)/2)
- }
- type unpadTextWidgetLayout struct {
- }
- func (u *unpadTextWidgetLayout) Layout(o []fyne.CanvasObject, s fyne.Size) {
- pad := theme.InnerPadding() * -1
- pad2 := pad * -2
- o[0].Move(fyne.NewPos(pad, pad))
- o[0].Resize(s.Add(fyne.NewSize(pad2, pad2)))
- }
- func (u *unpadTextWidgetLayout) MinSize(o []fyne.CanvasObject) fyne.Size {
- pad := theme.InnerPadding() * 2
- return o[0].MinSize().Subtract(fyne.NewSize(pad, pad))
- }
|