mirror of
https://github.com/diamondburned/cchat-gtk.git
synced 2024-11-01 03:54:16 +00:00
219 lines
5.6 KiB
Go
219 lines
5.6 KiB
Go
package message
|
|
|
|
import (
|
|
"time"
|
|
|
|
"github.com/diamondburned/cchat"
|
|
"github.com/diamondburned/cchat-gtk/internal/humanize"
|
|
"github.com/diamondburned/cchat-gtk/internal/ui/primitives"
|
|
"github.com/diamondburned/cchat-gtk/internal/ui/primitives/menu"
|
|
"github.com/diamondburned/cchat-gtk/internal/ui/rich"
|
|
"github.com/diamondburned/cchat-gtk/internal/ui/rich/labeluri"
|
|
"github.com/diamondburned/cchat-gtk/internal/ui/rich/parser/markup"
|
|
"github.com/diamondburned/cchat/text"
|
|
"github.com/gotk3/gotk3/gtk"
|
|
"github.com/gotk3/gotk3/pango"
|
|
)
|
|
|
|
type Container interface {
|
|
ID() string
|
|
AuthorID() string
|
|
AvatarURL() string // avatar
|
|
Nonce() string
|
|
|
|
UpdateAuthor(cchat.MessageAuthor)
|
|
UpdateAuthorName(text.Rich)
|
|
UpdateContent(c text.Rich, edited bool)
|
|
UpdateTimestamp(time.Time)
|
|
}
|
|
|
|
// FillContainer sets the container's contents to the one from MessageCreate.
|
|
func FillContainer(c Container, msg cchat.MessageCreate) {
|
|
c.UpdateAuthor(msg.Author())
|
|
c.UpdateContent(msg.Content(), false)
|
|
c.UpdateTimestamp(msg.Time())
|
|
}
|
|
|
|
// RefreshContainer sets the container's contents to the one from
|
|
// GenericContainer. This is mainly used for transferring between different
|
|
// containers.
|
|
//
|
|
// Right now, this only works with Timestamp, as that's the only state tracked.
|
|
func RefreshContainer(c Container, gc *GenericContainer) {
|
|
c.UpdateTimestamp(gc.time)
|
|
}
|
|
|
|
// GenericContainer provides a single generic message container for subpackages
|
|
// to use.
|
|
type GenericContainer struct {
|
|
id string
|
|
time time.Time
|
|
authorID string
|
|
avatarURL string // avatar
|
|
nonce string
|
|
|
|
Timestamp *gtk.Label
|
|
Username *labeluri.Label
|
|
Content gtk.IWidget // conceal widget implementation
|
|
|
|
contentBox *gtk.Box // basically what is in Content
|
|
ContentBody *labeluri.Label
|
|
|
|
MenuItems []menu.Item
|
|
}
|
|
|
|
var _ Container = (*GenericContainer)(nil)
|
|
|
|
var timestampCSS = primitives.PrepareCSS(`
|
|
.message-time {
|
|
opacity: 0.3;
|
|
font-size: 0.8em;
|
|
margin-top: 0.2em;
|
|
margin-bottom: 0.2em;
|
|
}
|
|
`)
|
|
|
|
// NewContainer creates a new message container with the given ID and nonce. It
|
|
// does not update the widgets, so FillContainer should be called afterwards.
|
|
func NewContainer(msg cchat.MessageCreate) *GenericContainer {
|
|
c := NewEmptyContainer()
|
|
c.id = msg.ID()
|
|
c.time = msg.Time()
|
|
c.authorID = msg.Author().ID()
|
|
|
|
if noncer, ok := msg.(cchat.MessageNonce); ok {
|
|
c.nonce = noncer.Nonce()
|
|
}
|
|
|
|
return c
|
|
}
|
|
|
|
func NewEmptyContainer() *GenericContainer {
|
|
ts, _ := gtk.LabelNew("")
|
|
ts.SetEllipsize(pango.ELLIPSIZE_MIDDLE)
|
|
ts.SetXAlign(1) // right align
|
|
ts.SetVAlign(gtk.ALIGN_END)
|
|
ts.Show()
|
|
|
|
user := labeluri.NewLabel(text.Rich{})
|
|
user.SetMaxWidthChars(35)
|
|
user.SetLineWrap(true)
|
|
user.SetLineWrapMode(pango.WRAP_WORD_CHAR)
|
|
user.SetXAlign(1) // right align
|
|
user.SetVAlign(gtk.ALIGN_START)
|
|
user.SetTrackVisitedLinks(false)
|
|
user.Show()
|
|
|
|
ctbody := labeluri.NewLabel(text.Rich{})
|
|
ctbody.SetEllipsize(pango.ELLIPSIZE_NONE)
|
|
ctbody.SetLineWrap(true)
|
|
ctbody.SetLineWrapMode(pango.WRAP_WORD_CHAR)
|
|
ctbody.SetXAlign(0) // left align
|
|
ctbody.SetSelectable(true)
|
|
ctbody.SetTrackVisitedLinks(false)
|
|
ctbody.Show()
|
|
|
|
// Wrap the content label inside a content box.
|
|
ctbox, _ := gtk.BoxNew(gtk.ORIENTATION_VERTICAL, 0)
|
|
ctbox.PackStart(ctbody, false, false, 0)
|
|
ctbox.Show()
|
|
|
|
// Causes bugs with selections.
|
|
|
|
// ctbody.Connect("grab-notify", func(l *gtk.Label, grabbed bool) {
|
|
// if grabbed {
|
|
// // Hack to stop the label from selecting everything after being
|
|
// // refocused.
|
|
// ctbody.SetSelectable(false)
|
|
// gts.ExecAsync(func() { ctbody.SetSelectable(true) })
|
|
// }
|
|
// })
|
|
|
|
// Add CSS classes.
|
|
primitives.AddClass(ts, "message-time")
|
|
primitives.AddClass(user, "message-author")
|
|
primitives.AddClass(ctbody, "message-content")
|
|
|
|
// Attach the timestamp CSS.
|
|
primitives.AttachCSS(ts, timestampCSS)
|
|
|
|
gc := &GenericContainer{
|
|
Timestamp: ts,
|
|
Username: user,
|
|
Content: ctbox,
|
|
contentBox: ctbox,
|
|
ContentBody: ctbody,
|
|
}
|
|
|
|
// Bind the custom popup menu to the content label.
|
|
gc.ContentBody.Connect("populate-popup", func(l *gtk.Label, m *gtk.Menu) {
|
|
menu.MenuSeparator(m)
|
|
menu.MenuItems(m, gc.MenuItems)
|
|
})
|
|
|
|
return gc
|
|
}
|
|
|
|
func (m *GenericContainer) ID() string {
|
|
return m.id
|
|
}
|
|
|
|
func (m *GenericContainer) Time() time.Time {
|
|
return m.time
|
|
}
|
|
|
|
func (m *GenericContainer) AuthorID() string {
|
|
return m.authorID
|
|
}
|
|
|
|
func (m *GenericContainer) AvatarURL() string {
|
|
return m.avatarURL
|
|
}
|
|
|
|
func (m *GenericContainer) Nonce() string {
|
|
return m.nonce
|
|
}
|
|
|
|
func (m *GenericContainer) UpdateTimestamp(t time.Time) {
|
|
m.time = t
|
|
m.Timestamp.SetText(humanize.TimeAgo(t))
|
|
m.Timestamp.SetTooltipText(t.Format(time.Stamp))
|
|
}
|
|
|
|
func (m *GenericContainer) UpdateAuthor(author cchat.MessageAuthor) {
|
|
m.authorID = author.ID()
|
|
m.UpdateAuthorName(author.Name())
|
|
|
|
// Set the avatar URL for future access on-demand.
|
|
if avatarer, ok := author.(cchat.MessageAuthorAvatar); ok {
|
|
m.avatarURL = avatarer.Avatar()
|
|
}
|
|
}
|
|
|
|
// authorRenderCfg is the config to render author names. It disables author
|
|
// mention links, as there's no way to make normal names not appear blue.
|
|
var authorRenderCfg = markup.RenderConfig{
|
|
NoMentionLinks: true,
|
|
}
|
|
|
|
func (m *GenericContainer) UpdateAuthorName(name text.Rich) {
|
|
var out = markup.RenderCmplxWithConfig(name, authorRenderCfg)
|
|
m.Username.SetOutput(out)
|
|
}
|
|
|
|
func (m *GenericContainer) UpdateContent(content text.Rich, edited bool) {
|
|
m.ContentBody.SetLabelUnsafe(content)
|
|
|
|
if edited {
|
|
markup := m.ContentBody.Output().Markup
|
|
markup += " " + rich.Small("(edited)")
|
|
m.ContentBody.SetMarkup(markup)
|
|
}
|
|
}
|
|
|
|
// AttachMenu connects signal handlers to handle a list of menu items from
|
|
// the container.
|
|
func (m *GenericContainer) AttachMenu(newItems []menu.Item) {
|
|
m.MenuItems = newItems
|
|
}
|