Apply declarative window specs across display, MCP, and window service paths; route layout/window controls through IPC tasks; and add a local Wails stub so the workspace builds cleanly here. Co-Authored-By: Virgil <virgil@lethean.io>
150 lines
4.9 KiB
Go
150 lines
4.9 KiB
Go
// pkg/mcp/tools_screen.go
|
|
package mcp
|
|
|
|
import (
|
|
"context"
|
|
|
|
coreerr "forge.lthn.ai/core/go-log"
|
|
"forge.lthn.ai/core/gui/pkg/screen"
|
|
"forge.lthn.ai/core/gui/pkg/window"
|
|
"github.com/modelcontextprotocol/go-sdk/mcp"
|
|
)
|
|
|
|
// --- screen_list ---
|
|
|
|
type ScreenListInput struct{}
|
|
type ScreenListOutput struct {
|
|
Screens []screen.Screen `json:"screens"`
|
|
}
|
|
|
|
func (s *Subsystem) screenList(_ context.Context, _ *mcp.CallToolRequest, _ ScreenListInput) (*mcp.CallToolResult, ScreenListOutput, error) {
|
|
result, _, err := s.core.QUERY(screen.QueryAll{})
|
|
if err != nil {
|
|
return nil, ScreenListOutput{}, err
|
|
}
|
|
screens, ok := result.([]screen.Screen)
|
|
if !ok {
|
|
return nil, ScreenListOutput{}, coreerr.E("mcp.screenList", "unexpected result type", nil)
|
|
}
|
|
return nil, ScreenListOutput{Screens: screens}, nil
|
|
}
|
|
|
|
// --- screen_get ---
|
|
|
|
type ScreenGetInput struct {
|
|
ID string `json:"id"`
|
|
}
|
|
type ScreenGetOutput struct {
|
|
Screen *screen.Screen `json:"screen"`
|
|
}
|
|
|
|
func (s *Subsystem) screenGet(_ context.Context, _ *mcp.CallToolRequest, input ScreenGetInput) (*mcp.CallToolResult, ScreenGetOutput, error) {
|
|
result, _, err := s.core.QUERY(screen.QueryByID{ID: input.ID})
|
|
if err != nil {
|
|
return nil, ScreenGetOutput{}, err
|
|
}
|
|
scr, ok := result.(*screen.Screen)
|
|
if !ok {
|
|
return nil, ScreenGetOutput{}, coreerr.E("mcp.screenGet", "unexpected result type", nil)
|
|
}
|
|
return nil, ScreenGetOutput{Screen: scr}, nil
|
|
}
|
|
|
|
// --- screen_primary ---
|
|
|
|
type ScreenPrimaryInput struct{}
|
|
type ScreenPrimaryOutput struct {
|
|
Screen *screen.Screen `json:"screen"`
|
|
}
|
|
|
|
func (s *Subsystem) screenPrimary(_ context.Context, _ *mcp.CallToolRequest, _ ScreenPrimaryInput) (*mcp.CallToolResult, ScreenPrimaryOutput, error) {
|
|
result, _, err := s.core.QUERY(screen.QueryPrimary{})
|
|
if err != nil {
|
|
return nil, ScreenPrimaryOutput{}, err
|
|
}
|
|
scr, ok := result.(*screen.Screen)
|
|
if !ok {
|
|
return nil, ScreenPrimaryOutput{}, coreerr.E("mcp.screenPrimary", "unexpected result type", nil)
|
|
}
|
|
return nil, ScreenPrimaryOutput{Screen: scr}, nil
|
|
}
|
|
|
|
// --- screen_at_point ---
|
|
|
|
type ScreenAtPointInput struct {
|
|
X int `json:"x"`
|
|
Y int `json:"y"`
|
|
}
|
|
type ScreenAtPointOutput struct {
|
|
Screen *screen.Screen `json:"screen"`
|
|
}
|
|
|
|
func (s *Subsystem) screenAtPoint(_ context.Context, _ *mcp.CallToolRequest, input ScreenAtPointInput) (*mcp.CallToolResult, ScreenAtPointOutput, error) {
|
|
result, _, err := s.core.QUERY(screen.QueryAtPoint{X: input.X, Y: input.Y})
|
|
if err != nil {
|
|
return nil, ScreenAtPointOutput{}, err
|
|
}
|
|
scr, ok := result.(*screen.Screen)
|
|
if !ok {
|
|
return nil, ScreenAtPointOutput{}, coreerr.E("mcp.screenAtPoint", "unexpected result type", nil)
|
|
}
|
|
return nil, ScreenAtPointOutput{Screen: scr}, nil
|
|
}
|
|
|
|
// --- screen_work_areas ---
|
|
|
|
type ScreenWorkAreasInput struct{}
|
|
type ScreenWorkAreasOutput struct {
|
|
WorkAreas []screen.Rect `json:"workAreas"`
|
|
}
|
|
|
|
func (s *Subsystem) screenWorkAreas(_ context.Context, _ *mcp.CallToolRequest, _ ScreenWorkAreasInput) (*mcp.CallToolResult, ScreenWorkAreasOutput, error) {
|
|
result, _, err := s.core.QUERY(screen.QueryWorkAreas{})
|
|
if err != nil {
|
|
return nil, ScreenWorkAreasOutput{}, err
|
|
}
|
|
areas, ok := result.([]screen.Rect)
|
|
if !ok {
|
|
return nil, ScreenWorkAreasOutput{}, coreerr.E("mcp.screenWorkAreas", "unexpected result type", nil)
|
|
}
|
|
return nil, ScreenWorkAreasOutput{WorkAreas: areas}, nil
|
|
}
|
|
|
|
// --- screen_for_window ---
|
|
|
|
type ScreenForWindowInput struct {
|
|
Name string `json:"name"`
|
|
}
|
|
type ScreenForWindowOutput struct {
|
|
Screen *screen.Screen `json:"screen"`
|
|
}
|
|
|
|
func (s *Subsystem) screenForWindow(_ context.Context, _ *mcp.CallToolRequest, input ScreenForWindowInput) (*mcp.CallToolResult, ScreenForWindowOutput, error) {
|
|
result, _, err := s.core.QUERY(window.QueryWindowByName{Name: input.Name})
|
|
if err != nil {
|
|
return nil, ScreenForWindowOutput{}, err
|
|
}
|
|
info, _ := result.(*window.WindowInfo)
|
|
if info == nil {
|
|
return nil, ScreenForWindowOutput{}, nil
|
|
}
|
|
centerX := info.X + info.Width/2
|
|
centerY := info.Y + info.Height/2
|
|
screenResult, _, err := s.core.QUERY(screen.QueryAtPoint{X: centerX, Y: centerY})
|
|
if err != nil {
|
|
return nil, ScreenForWindowOutput{}, err
|
|
}
|
|
scr, _ := screenResult.(*screen.Screen)
|
|
return nil, ScreenForWindowOutput{Screen: scr}, nil
|
|
}
|
|
|
|
// --- Registration ---
|
|
|
|
func (s *Subsystem) registerScreenTools(server *mcp.Server) {
|
|
mcp.AddTool(server, &mcp.Tool{Name: "screen_list", Description: "List all connected displays/screens"}, s.screenList)
|
|
mcp.AddTool(server, &mcp.Tool{Name: "screen_get", Description: "Get information about a specific screen"}, s.screenGet)
|
|
mcp.AddTool(server, &mcp.Tool{Name: "screen_primary", Description: "Get the primary screen"}, s.screenPrimary)
|
|
mcp.AddTool(server, &mcp.Tool{Name: "screen_at_point", Description: "Get the screen at a specific point"}, s.screenAtPoint)
|
|
mcp.AddTool(server, &mcp.Tool{Name: "screen_work_areas", Description: "Get work areas for all screens"}, s.screenWorkAreas)
|
|
mcp.AddTool(server, &mcp.Tool{Name: "screen_for_window", Description: "Get the screen containing a window"}, s.screenForWindow)
|
|
}
|