mirror of
https://github.com/tiennm99/goclaw.git
synced 2026-06-10 16:10:59 +00:00
7d211fa796
Pure cut-and-paste of functions/methods into separate files within the same package — no logic changes. Reduces file sizes for readability. - loop.go (1312→856) → loop_types.go, loop_compact.go, loop_media.go, loop_utils.go - delegate.go (687→171) → delegate_sync.go, delegate_async.go, delegate_prep.go - browser.go (605→154) → browser_tabs.go, browser_page.go, browser_remote.go - teams.go (602→170) → teams_crud.go, teams_members.go - web_fetch_convert.go (572→176) → web_fetch_convert_handlers.go, web_fetch_convert_utils.go - resolver.go (543→373) → resolver_helpers.go - sessions.go (536→157) → sessions_tokens.go, sessions_ops.go, sessions_list.go Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
130 lines
2.7 KiB
Go
130 lines
2.7 KiB
Go
package browser
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/go-rod/rod/lib/proto"
|
|
)
|
|
|
|
// ListTabs returns all open tabs.
|
|
func (m *Manager) ListTabs(ctx context.Context) ([]TabInfo, error) {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
if m.browser == nil {
|
|
return nil, fmt.Errorf("browser not running")
|
|
}
|
|
|
|
pages, err := m.browser.Pages()
|
|
if err != nil {
|
|
if m.remoteURL != "" {
|
|
if reconnErr := m.reconnectLocked(); reconnErr != nil {
|
|
return nil, fmt.Errorf("list pages: %w (reconnect also failed: %v)", err, reconnErr)
|
|
}
|
|
m.logger.Info("auto-reconnected to remote Chrome")
|
|
pages, err = m.browser.Pages()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("list pages after reconnect: %w", err)
|
|
}
|
|
} else {
|
|
return nil, fmt.Errorf("list pages: %w", err)
|
|
}
|
|
}
|
|
|
|
tabs := make([]TabInfo, 0, len(pages))
|
|
for _, p := range pages {
|
|
info, err := p.Info()
|
|
if err != nil || info == nil {
|
|
continue
|
|
}
|
|
tid := string(p.TargetID)
|
|
m.pages[tid] = p
|
|
tabs = append(tabs, TabInfo{
|
|
TargetID: tid,
|
|
URL: info.URL,
|
|
Title: info.Title,
|
|
})
|
|
}
|
|
return tabs, nil
|
|
}
|
|
|
|
// OpenTab opens a new tab with the given URL.
|
|
func (m *Manager) OpenTab(ctx context.Context, url string) (*TabInfo, error) {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
if m.browser == nil {
|
|
return nil, fmt.Errorf("browser not running")
|
|
}
|
|
|
|
page, err := m.browser.Page(proto.TargetCreateTarget{URL: url})
|
|
if err != nil {
|
|
return nil, fmt.Errorf("open tab: %w", err)
|
|
}
|
|
|
|
if err := page.WaitStable(300 * time.Millisecond); err != nil {
|
|
return nil, fmt.Errorf("wait stable: %w", err)
|
|
}
|
|
info, _ := page.Info()
|
|
tid := string(page.TargetID)
|
|
m.pages[tid] = page
|
|
|
|
// Set up console listener
|
|
m.setupConsoleListener(page, tid)
|
|
|
|
tab := &TabInfo{TargetID: tid, URL: url}
|
|
if info != nil {
|
|
tab.URL = info.URL
|
|
tab.Title = info.Title
|
|
}
|
|
return tab, nil
|
|
}
|
|
|
|
// FocusTab activates a tab.
|
|
func (m *Manager) FocusTab(ctx context.Context, targetID string) error {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
page, err := m.getPage(targetID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = page.Activate()
|
|
return err
|
|
}
|
|
|
|
// CloseTab closes a tab.
|
|
func (m *Manager) CloseTab(ctx context.Context, targetID string) error {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
page, err := m.getPage(targetID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
delete(m.pages, targetID)
|
|
delete(m.console, targetID)
|
|
return page.Close()
|
|
}
|
|
|
|
// ConsoleMessages returns captured console messages for a tab.
|
|
func (m *Manager) ConsoleMessages(targetID string) []ConsoleMessage {
|
|
m.mu.Lock()
|
|
defer m.mu.Unlock()
|
|
|
|
msgs := m.console[targetID]
|
|
if msgs == nil {
|
|
return []ConsoleMessage{}
|
|
}
|
|
|
|
// Return copy and clear
|
|
result := make([]ConsoleMessage, len(msgs))
|
|
copy(result, msgs)
|
|
m.console[targetID] = nil
|
|
return result
|
|
}
|