// pkg/mcp/tools_screen.go package mcp import ( "context" coreerr "dappco.re/go/core/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) { r := s.core.QUERY(screen.QueryAll{}) if !r.OK { if e, ok := r.Value.(error); ok { return nil, ScreenListOutput{}, e } return nil, ScreenListOutput{}, nil } screens, ok := r.Value.([]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) { r := s.core.QUERY(screen.QueryByID{ID: input.ID}) if !r.OK { if e, ok := r.Value.(error); ok { return nil, ScreenGetOutput{}, e } return nil, ScreenGetOutput{}, nil } scr, ok := r.Value.(*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) { r := s.core.QUERY(screen.QueryPrimary{}) if !r.OK { if e, ok := r.Value.(error); ok { return nil, ScreenPrimaryOutput{}, e } return nil, ScreenPrimaryOutput{}, nil } scr, ok := r.Value.(*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) { r := s.core.QUERY(screen.QueryAtPoint{X: input.X, Y: input.Y}) if !r.OK { if e, ok := r.Value.(error); ok { return nil, ScreenAtPointOutput{}, e } return nil, ScreenAtPointOutput{}, nil } scr, ok := r.Value.(*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) { r := s.core.QUERY(screen.QueryWorkAreas{}) if !r.OK { if e, ok := r.Value.(error); ok { return nil, ScreenWorkAreasOutput{}, e } return nil, ScreenWorkAreasOutput{}, nil } areas, ok := r.Value.([]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) { r := s.core.QUERY(window.QueryWindowByName{Name: input.Name}) if !r.OK { return nil, ScreenForWindowOutput{}, nil } info, _ := r.Value.(*window.WindowInfo) if info == nil { return nil, ScreenForWindowOutput{}, nil } centerX := info.X + info.Width/2 centerY := info.Y + info.Height/2 r2 := s.core.QUERY(screen.QueryAtPoint{X: centerX, Y: centerY}) if !r2.OK { return nil, ScreenForWindowOutput{}, nil } scr, _ := r2.Value.(*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) }