feat: derive AI model list from Raycast SDK enum

Generate MODEL_OPTIONS at runtime from AI.Model (dedupe by value, drop
@deprecated aliases), so the AI-view dropdowns show every model the
installed @raycast/api ships — no more hand-maintained six-entry list.

Switch the summaryModel preference from a static dropdown to a textfield
holding a model ID. The discoverable picker lives in the AI views (now
fully dynamic), and the preference just stores the chosen default.

Drop duplicate MODEL_OPTIONS / CREATIVITY_OPTIONS constants from
summarize-email.tsx and reply-email.tsx; both pull from ./ai now like
the briefing and extract views already do.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
muena
2026-05-20 07:08:58 +02:00
parent c8b3cf44e3
commit c0078ce339
5 changed files with 25 additions and 60 deletions

View File

@@ -17,7 +17,7 @@ Alle Werte sind in den Raycast-Einstellungen pro Extension einstellbar.
**Verhalten:** **Verhalten:**
- `Standard-Sitzungsmodus``Aktive Sitzung wiederverwenden` (Default), `Neue Sitzung pro Anfrage`, `Tagessitzung` - `Standard-Sitzungsmodus``Aktive Sitzung wiederverwenden` (Default), `Neue Sitzung pro Anfrage`, `Tagessitzung`
- `Ausgabe der Schnellbefehle``In die Zwischenablage kopieren` (Default) oder `Am Cursor einfügen` - `Ausgabe der Schnellbefehle``In die Zwischenablage kopieren` (Default) oder `Am Cursor einfügen`
- `Standard-Modell für Zusammenfassungen` — Default-Raycast-KI-Modell für alle AI-Befehle (benötigt Raycast Pro); pro AI-View überschreibbar - `Standard-Modell für AI-Befehle` — Modell-ID, die in den AI-Views vorausgewählt ist (Default `anthropic-claude-sonnet-4-6`, benötigt Raycast Pro). Die vollständige Modellliste leiten die Views dynamisch aus `AI.Model` der Raycast-API ab — sobald ein neues Modell verfügbar ist, taucht es ohne Code-Änderung im View-Dropdown auf.
- `Eigener Name` — Default-Signatur für AI-generierte Email-Antworten, im Antwort-Befehl pro Aufruf überschreibbar - `Eigener Name` — Default-Signatur für AI-generierte Email-Antworten, im Antwort-Befehl pro Aufruf überschreibbar
- `Maximale Anzahl gespeicherter Sitzungen` — älteste werden geprunt (Default 20) - `Maximale Anzahl gespeicherter Sitzungen` — älteste werden geprunt (Default 20)
- `Raycast nach Kopieren/Einfügen schließen` — Auto-Close und Pop-To-Root nach AI-Workflow-Abschluss (Default an) - `Raycast nach Kopieren/Einfügen schließen` — Auto-Close und Pop-To-Root nach AI-Workflow-Abschluss (Default an)

View File

@@ -180,37 +180,12 @@
}, },
{ {
"name": "summaryModel", "name": "summaryModel",
"title": "Standard-Modell für Zusammenfassungen", "title": "Standard-Modell für AI-Befehle",
"description": "Raycast-KI-Modell für „Email-Konversation zusammenfassen“. Benötigt Raycast Pro.", "description": "Raycast-KI-Modell-ID, vorausgewählt in den AI-Views. Die volle Modellliste ist dort als Dropdown verfügbar. Benötigt Raycast Pro.",
"type": "dropdown", "type": "textfield",
"required": true, "required": true,
"default": "anthropic-claude-sonnet-4-6", "default": "anthropic-claude-sonnet-4-6",
"data": [ "placeholder": "anthropic-claude-sonnet-4-6"
{
"title": "Claude 4.6 Sonnet",
"value": "anthropic-claude-sonnet-4-6"
},
{
"title": "Claude 4.7 Opus",
"value": "anthropic-claude-opus-4-7"
},
{
"title": "Claude 4.5 Haiku",
"value": "anthropic-claude-4-5-haiku"
},
{
"title": "OpenAI GPT-5.3 Instant",
"value": "openai-gpt-5.3-instant"
},
{
"title": "OpenAI GPT-4.1",
"value": "openai-gpt-4.1"
},
{
"title": "OpenAI GPT-4o mini",
"value": "openai-gpt-4o-mini"
}
]
}, },
{ {
"name": "maxSessions", "name": "maxSessions",

View File

@@ -1,3 +1,5 @@
import { AI } from "@raycast/api";
export type Creativity = "none" | "low" | "medium" | "high"; export type Creativity = "none" | "low" | "medium" | "high";
export const CREATIVITY_OPTIONS: Creativity[] = [ export const CREATIVITY_OPTIONS: Creativity[] = [
@@ -7,14 +9,22 @@ export const CREATIVITY_OPTIONS: Creativity[] = [
"high", "high",
]; ];
export const MODEL_OPTIONS: Array<{ value: string; title: string }> = [ export type ModelOption = { value: string; title: string };
{ value: "anthropic-claude-sonnet-4-6", title: "Claude 4.6 Sonnet" },
{ value: "anthropic-claude-opus-4-7", title: "Claude 4.7 Opus" }, function buildModelOptions(): ModelOption[] {
{ value: "anthropic-claude-4-5-haiku", title: "Claude 4.5 Haiku" }, const seen = new Set<string>();
{ value: "openai-gpt-5.3-instant", title: "OpenAI GPT-5.3 Instant" }, const options: ModelOption[] = [];
{ value: "openai-gpt-4.1", title: "OpenAI GPT-4.1" }, for (const [key, value] of Object.entries(AI.Model) as Array<
{ value: "openai-gpt-4o-mini", title: "OpenAI GPT-4o mini" }, [string, string]
]; >) {
if (seen.has(value)) continue;
seen.add(value);
options.push({ value, title: key.replace(/_/g, " ") });
}
return options;
}
export const MODEL_OPTIONS: ModelOption[] = buildModelOptions();
export const STRICT_PLACEHOLDER_RULE = [ export const STRICT_PLACEHOLDER_RULE = [
"STRENGE REGEL: Gib jeden Platzhalter zeichengetreu (inklusive spitzer Klammern, Großbuchstaben und Unterstrich + Nummer) zurück.", "STRENGE REGEL: Gib jeden Platzhalter zeichengetreu (inklusive spitzer Klammern, Großbuchstaben und Unterstrich + Nummer) zurück.",

View File

@@ -12,6 +12,7 @@ import {
} from "@raycast/api"; } from "@raycast/api";
import { getSelectedTextSafely } from "./selection"; import { getSelectedTextSafely } from "./selection";
import { useEffect, useMemo, useRef, useState } from "react"; import { useEffect, useMemo, useRef, useState } from "react";
import { CREATIVITY_OPTIONS, MODEL_OPTIONS } from "./ai";
import { getPreferences } from "./preferences"; import { getPreferences } from "./preferences";
import { import {
buildDisclosureContent, buildDisclosureContent,
@@ -54,17 +55,6 @@ type FormValues = {
instructions: string; instructions: string;
}; };
const CREATIVITY_OPTIONS: Creativity[] = ["none", "low", "medium", "high"];
const MODEL_OPTIONS: Array<{ value: string; title: string }> = [
{ value: "anthropic-claude-sonnet-4-6", title: "Claude 4.6 Sonnet" },
{ value: "anthropic-claude-opus-4-7", title: "Claude 4.7 Opus" },
{ value: "anthropic-claude-4-5-haiku", title: "Claude 4.5 Haiku" },
{ value: "openai-gpt-5.3-instant", title: "OpenAI GPT-5.3 Instant" },
{ value: "openai-gpt-4.1", title: "OpenAI GPT-4.1" },
{ value: "openai-gpt-4o-mini", title: "OpenAI GPT-4o mini" },
];
export default function Command() { export default function Command() {
const preferences = getPreferences(); const preferences = getPreferences();
const [text, setText] = useState(""); const [text, setText] = useState("");

View File

@@ -23,6 +23,7 @@ import {
setActiveSession, setActiveSession,
updateSessionMapping, updateSessionMapping,
} from "./sessions"; } from "./sessions";
import { CREATIVITY_OPTIONS, MODEL_OPTIONS } from "./ai";
import { buildSummaryPrompt, type Creativity } from "./summarize"; import { buildSummaryPrompt, type Creativity } from "./summarize";
import type { EntityType, VelumSession } from "./types"; import type { EntityType, VelumSession } from "./types";
import { import {
@@ -47,17 +48,6 @@ type FormValues = {
instructions: string; instructions: string;
}; };
const CREATIVITY_OPTIONS: Creativity[] = ["none", "low", "medium", "high"];
const MODEL_OPTIONS: Array<{ value: string; title: string }> = [
{ value: "anthropic-claude-sonnet-4-6", title: "Claude 4.6 Sonnet" },
{ value: "anthropic-claude-opus-4-7", title: "Claude 4.7 Opus" },
{ value: "anthropic-claude-4-5-haiku", title: "Claude 4.5 Haiku" },
{ value: "openai-gpt-5.3-instant", title: "OpenAI GPT-5.3 Instant" },
{ value: "openai-gpt-4.1", title: "OpenAI GPT-4.1" },
{ value: "openai-gpt-4o-mini", title: "OpenAI GPT-4o mini" },
];
export default function Command() { export default function Command() {
const preferences = getPreferences(); const preferences = getPreferences();
const [text, setText] = useState(""); const [text, setText] = useState("");