1
0
Fork 0
mirror of https://github.com/diamondburned/cchat-gtk.git synced 2025-01-23 10:26:42 +00:00
cchat-gtk/internal/ui/service/service.go

285 lines
7.7 KiB
Go
Raw Normal View History

2020-05-26 06:51:06 +00:00
package service
import (
"fmt"
2020-05-26 06:51:06 +00:00
"github.com/diamondburned/cchat"
2020-06-07 04:27:28 +00:00
"github.com/diamondburned/cchat-gtk/internal/keyring"
2020-06-07 07:06:13 +00:00
"github.com/diamondburned/cchat-gtk/internal/log"
2020-05-28 19:26:55 +00:00
"github.com/diamondburned/cchat-gtk/internal/ui/primitives"
"github.com/diamondburned/cchat-gtk/internal/ui/primitives/actions"
"github.com/diamondburned/cchat-gtk/internal/ui/primitives/drag"
"github.com/diamondburned/cchat-gtk/internal/ui/primitives/roundimage"
"github.com/diamondburned/cchat-gtk/internal/ui/rich"
"github.com/diamondburned/cchat-gtk/internal/ui/rich/parser/markup"
"github.com/diamondburned/cchat-gtk/internal/ui/service/config"
2020-05-26 06:51:06 +00:00
"github.com/diamondburned/cchat-gtk/internal/ui/service/session"
2020-06-07 04:27:28 +00:00
"github.com/diamondburned/cchat-gtk/internal/ui/service/session/server"
"github.com/diamondburned/cchat-gtk/internal/ui/service/session/server/traverse"
2020-05-26 06:51:06 +00:00
"github.com/gotk3/gotk3/gtk"
)
const IconSize = 48
2020-05-26 06:51:06 +00:00
type ListController interface {
// ClearMessenger is called when a nil slice of servers is set.
ClearMessenger(*session.Row)
2020-10-15 06:32:11 +00:00
// MessengerSelected is called when a server message row is clicked.
MessengerSelected(*session.Row, *server.ServerRow)
// SessionSelected tells the view to change the session view.
SessionSelected(*Service, *session.Row)
// AuthenticateSession tells View to call to the parent's authenticator.
AuthenticateSession(*Service)
// MoveService tells the view to shift the service to before the target.
MoveService(id, targetID string)
2020-07-16 05:41:21 +00:00
OnSessionRemove(*Service, *session.Row)
OnSessionDisconnect(*Service, *session.Row)
}
// Service holds everything that a single service has.
type Service struct {
ListController
*gtk.Box
Button *gtk.ToggleButton
Icon *rich.Icon
Menu *actions.Menu
2020-05-26 06:51:06 +00:00
BodyRev *gtk.Revealer // revealed
BodyList *session.List // not really supposed to be here
2020-05-28 19:26:55 +00:00
service cchat.Service // state
Configurator cchat.Configurator
}
2020-05-26 06:51:06 +00:00
var serviceCSS = primitives.PrepareClassCSS("service", `
.service {
box-shadow: 0 0 2px 0 alpha(@theme_bg_color, 0.75);
margin: 6px 8px;
margin-bottom: 0;
border-radius: 14px;
2020-05-26 06:51:06 +00:00
}
.service:first-child { margin-top: 8px; }
.service:last-child { margin-bottom: 8px; }
`)
2020-06-07 07:06:13 +00:00
var serviceButtonCSS = primitives.PrepareClassCSS("service-button", `
.service-button {
padding: 2px;
margin: 0;
}
2020-06-07 07:06:13 +00:00
.service-button:not(:checked) {
border-radius: 14px;
transition: linear 80ms border-radius; /* TODO add delay */
}
2020-05-26 06:51:06 +00:00
.service-button:checked {
border-radius: 14px 14px 0 0;
background-color: alpha(@theme_fg_color, 0.2);
}
`)
2020-06-07 04:27:28 +00:00
var serviceIconCSS = primitives.PrepareClassCSS("service-icon", `
.service-icon { padding: 4px }
`)
2020-06-07 04:27:28 +00:00
func NewService(svc cchat.Service, svclctrl ListController) *Service {
service := &Service{
service: svc,
ListController: svclctrl,
}
2020-05-26 06:51:06 +00:00
service.BodyList = session.NewList(service)
service.BodyList.Show()
service.BodyRev, _ = gtk.RevealerNew()
service.BodyRev.SetRevealChild(false) // TODO persistent state
service.BodyRev.SetTransitionDuration(50)
service.BodyRev.SetTransitionType(gtk.REVEALER_TRANSITION_TYPE_SLIDE_DOWN)
service.BodyRev.Add(service.BodyList)
service.BodyRev.Show()
// TODO: have it so the button changes to the session avatar when collapsed
avatar := roundimage.NewAvatar(IconSize)
avatar.SetText(svc.Name().String())
avatar.Show()
service.Icon = rich.NewCustomIcon(avatar, IconSize)
service.Icon.Show()
// potentially nonstandard
service.Icon.SetPlaceholderIcon("text-html-symbolic", IconSize)
// TODO: hover for name. We use tooltip for now.
service.Icon.SetTooltipMarkup(markup.Render(svc.Name()))
serviceIconCSS(service.Icon)
2020-10-15 06:32:11 +00:00
if iconer := svc.AsIconer(); iconer != nil {
service.Icon.AsyncSetIconer(iconer, "Failed to set service icon")
}
2020-06-07 07:06:13 +00:00
service.Button, _ = gtk.ToggleButtonNew()
service.Button.Add(service.Icon)
service.Button.SetRelief(gtk.RELIEF_NONE)
service.Button.Show()
service.Button.Connect("clicked", func(tb *gtk.ToggleButton) {
revealed := !service.GetRevealChild()
service.SetRevealChild(revealed)
tb.SetActive(revealed)
})
serviceButtonCSS(service.Button)
2020-05-28 19:26:55 +00:00
// Bind session.* actions into row.
service.Menu = actions.NewMenu("service")
// Bind right clicks and show a popover menu on such event.
service.Menu.BindRightClick(service.Button)
if configurator := svc.AsConfigurator(); configurator != nil {
cfg := config.Configurator{
Service: svc,
Configurator: configurator,
}
config.Restore(cfg)
service.Menu.AddAction("Configure", func() { config.Spawn(cfg) })
}
// Intermediary box to contain both the icon and the revealer.
service.Box, _ = gtk.BoxNew(gtk.ORIENTATION_VERTICAL, 0)
service.Box.PackStart(service.Button, false, false, 0)
service.Box.PackStart(service.BodyRev, false, false, 0)
service.Box.Show()
serviceCSS(service.Box)
2020-05-26 06:51:06 +00:00
// Bind a drag and drop on the button instead of the entire box.
drag.BindDraggable(service, "network-workgroup", svclctrl.MoveService, service.Button)
return service
}
2020-06-04 23:00:41 +00:00
// SetRevealChild sets whether or not the service should reveal all sessions.
func (s *Service) SetRevealChild(reveal bool) {
s.BodyRev.SetRevealChild(reveal)
}
2020-05-26 06:51:06 +00:00
// GetRevealChild gets whether or not the service is revealing all sessions.
func (s *Service) GetRevealChild() bool {
return s.BodyRev.GetRevealChild()
}
2020-05-26 06:51:06 +00:00
func (s *Service) SessionSelected(srow *session.Row) {
s.ListController.SessionSelected(s, srow)
}
2020-05-28 19:26:55 +00:00
func (s *Service) AuthenticateSession() {
s.ListController.AuthenticateSession(s)
}
2020-05-28 19:26:55 +00:00
func (s *Service) AddLoadingSession(id, name string) *session.Row {
2020-10-15 06:32:11 +00:00
if srow := s.BodyList.Session(id); srow != nil {
return srow
}
srow := session.NewLoading(s, id, name, s)
srow.Show()
2020-05-26 06:51:06 +00:00
s.BodyList.AddSessionRow(id, srow)
return srow
}
2020-10-15 06:32:11 +00:00
// AddSession adds the given session. It returns nil if the session already
// exists with the given ID.
func (s *Service) AddSession(ses cchat.Session) *session.Row {
2020-10-15 06:32:11 +00:00
if srow := s.BodyList.Session(ses.ID()); srow != nil {
return srow
}
srow := session.New(s, ses, s)
srow.Show()
2020-06-07 04:27:28 +00:00
s.BodyList.AddSessionRow(ses.ID(), srow)
s.SaveAllSessions()
return srow
2020-05-26 06:51:06 +00:00
}
func (s *Service) ID() string {
return s.service.Name().Content
}
func (s *Service) Service() cchat.Service {
return s.service
}
func (s *Service) OnSessionDisconnect(row *session.Row) {
2020-07-16 05:41:21 +00:00
// Unselect if selected.
if cur := s.BodyList.GetSelectedRow(); cur.GetIndex() == row.GetIndex() {
s.BodyList.UnselectAll()
}
s.ListController.OnSessionDisconnect(s, row)
2020-06-04 23:00:41 +00:00
}
func (s *Service) RemoveSession(row *session.Row) {
s.ListController.OnSessionRemove(s, row)
2020-10-15 06:32:11 +00:00
s.BodyList.RemoveSessionRow(row.ID())
s.SaveAllSessions()
2020-05-26 06:51:06 +00:00
}
func (s *Service) MoveSession(id, movingID string) {
s.BodyList.MoveSession(id, movingID)
s.SaveAllSessions()
}
func (s *Service) Breadcrumb() string {
return s.service.Name().Content
}
func (s *Service) ParentBreadcrumb() traverse.Breadcrumber {
return nil
2020-06-07 07:06:13 +00:00
}
func (s *Service) SaveAllSessions() {
var sessions = s.BodyList.Sessions()
var keyrings = make([]keyring.Session, 0, len(sessions))
for _, s := range sessions {
if k := keyring.ConvertSession(s.Session); k != nil {
keyrings = append(keyrings, *k)
}
}
keyring.SaveSessions(s.service, keyrings)
}
func (s *Service) RestoreSession(row *session.Row, id string) {
2020-10-15 06:32:11 +00:00
rs := s.service.AsSessionRestorer()
if rs == nil {
2020-06-07 07:06:13 +00:00
return
}
if k := keyring.RestoreSession(s.service, id); k != nil {
2020-10-15 06:32:11 +00:00
row.RestoreSession(rs, *k)
return
}
log.Error(fmt.Errorf(
"Missing keyring for service %s, session ID %s",
s.service.Name().Content, id,
))
2020-06-07 07:06:13 +00:00
}
// restoreAll restores all sessions.
func (s *Service) restoreAll() {
2020-10-15 06:32:11 +00:00
rs := s.service.AsSessionRestorer()
if rs == nil {
2020-06-07 07:06:13 +00:00
return
}
// Session is not a pointer, so we can pass it into arguments safely.
for _, ses := range keyring.RestoreSessions(s.service) {
row := s.AddLoadingSession(ses.ID, ses.Name)
2020-10-15 06:32:11 +00:00
row.RestoreSession(rs, ses)
2020-06-07 07:06:13 +00:00
}
}