Files
voidraft/internal/services/keybinding_service.go

300 lines
8.1 KiB
Go

package services
import (
"context"
"fmt"
"voidraft/internal/models"
"voidraft/internal/models/ent"
"voidraft/internal/models/ent/extension"
"voidraft/internal/models/ent/keybinding"
"voidraft/internal/models/schema/mixin"
"github.com/wailsapp/wails/v3/pkg/application"
"github.com/wailsapp/wails/v3/pkg/services/log"
)
// KeyBindingService 快捷键服务
type KeyBindingService struct {
db *DatabaseService
logger *log.LogService
}
// NewKeyBindingService 创建快捷键服务
func NewKeyBindingService(db *DatabaseService, logger *log.LogService) *KeyBindingService {
if logger == nil {
logger = log.New()
}
return &KeyBindingService{db: db, logger: logger}
}
// ServiceStartup 服务启动
func (s *KeyBindingService) ServiceStartup(ctx context.Context, options application.ServiceOptions) error {
return s.SyncKeyBindings(ctx)
}
// SyncKeyBindings 同步快捷键配置
func (s *KeyBindingService) SyncKeyBindings(ctx context.Context) error {
defaults := models.NewDefaultKeyBindings()
// 使用 type + name 作为唯一键
definedKeys := make(map[string]models.KeyBinding)
for _, kb := range defaults {
key := string(kb.Type) + ":" + string(kb.Name)
definedKeys[key] = kb
}
// 获取数据库中已有的快捷键
existing, err := s.db.Client.KeyBinding.Query().All(ctx)
if err != nil {
return fmt.Errorf("find key bindings error: %w", err)
}
existingKeys := make(map[string]bool)
for _, kb := range existing {
key := kb.Type + ":" + kb.Name
existingKeys[key] = true
}
// 批量添加缺失的快捷键
var builders []*ent.KeyBindingCreate
for key, kb := range definedKeys {
if !existingKeys[key] {
create := s.db.Client.KeyBinding.Create().
SetName(string(kb.Name)).
SetType(string(kb.Type)).
SetExtension(string(kb.Extension)).
SetEnabled(kb.Enabled).
SetPreventDefault(kb.PreventDefault)
// 设置快捷键字段
if kb.Key != "" {
create.SetKey(kb.Key)
}
if kb.Macos != "" {
create.SetMacos(kb.Macos)
}
if kb.Windows != "" {
create.SetWindows(kb.Windows)
}
if kb.Linux != "" {
create.SetLinux(kb.Linux)
}
if kb.Scope != "" {
create.SetScope(kb.Scope)
}
builders = append(builders, create)
}
}
if len(builders) > 0 {
if _, err := s.db.Client.KeyBinding.CreateBulk(builders...).Save(ctx); err != nil {
return fmt.Errorf("bulk insert key bindings error: %w", err)
}
}
// 批量删除废弃的快捷键(硬删除)
var deleteIDs []int
for _, kb := range existing {
key := kb.Type + ":" + kb.Name
if _, ok := definedKeys[key]; !ok {
deleteIDs = append(deleteIDs, kb.ID)
}
}
if len(deleteIDs) > 0 {
if _, err := s.db.Client.KeyBinding.Delete().
Where(keybinding.IDIn(deleteIDs...)).
Exec(mixin.SkipSoftDelete(ctx)); err != nil {
return fmt.Errorf("bulk delete key bindings error: %w", err)
}
}
return nil
}
// GetKeyBindings 根据类型获取快捷键
func (s *KeyBindingService) GetKeyBindings(ctx context.Context, kbType models.KeyBindingType) ([]*ent.KeyBinding, error) {
// 获取启用的扩展名称集合
enabledExts, err := s.db.Client.Extension.Query().
Where(extension.Enabled(true)).
All(ctx)
if err != nil {
return nil, fmt.Errorf("query enabled extensions error: %w", err)
}
enabledExtMap := make(map[string]bool, len(enabledExts))
for _, ext := range enabledExts {
enabledExtMap[ext.Name] = true
}
if kbType == models.Standard {
// Standard 模式:返回扩展已启用的快捷键
bindings, err := s.db.Client.KeyBinding.Query().
Where(keybinding.Type(string(kbType))).
Order(ent.Asc(keybinding.FieldID)).
All(ctx)
if err != nil {
return nil, err
}
return filterByExtension(bindings, enabledExtMap), nil
}
// Emacs 模式:获取所有快捷键
allBindings, err := s.db.Client.KeyBinding.Query().
Order(ent.Asc(keybinding.FieldID)).
All(ctx)
if err != nil {
return nil, fmt.Errorf("query key bindings error: %w", err)
}
// 构建 emacs 快捷键的 name 集合
emacsNames := make(map[string]bool)
for _, kb := range allBindings {
if kb.Type == string(models.Emacs) {
emacsNames[kb.Name] = true
}
}
// 过滤:去掉与 emacs 冲突的 standard 快捷键,并过滤扩展未启用的
var result []*ent.KeyBinding
for _, kb := range allBindings {
if kb.Type == string(models.Standard) && emacsNames[kb.Name] {
continue
}
// editor 扩展始终包含,不检查启用状态
if kb.Extension != models.DefaultExtension && !enabledExtMap[kb.Extension] {
continue
}
result = append(result, kb)
}
return result, nil
}
// filterByExtension 过滤出扩展已启用的快捷键
func filterByExtension(bindings []*ent.KeyBinding, enabledExtMap map[string]bool) []*ent.KeyBinding {
result := make([]*ent.KeyBinding, 0, len(bindings))
for _, kb := range bindings {
// editor 扩展始终包含,不检查启用状态
if kb.Extension == models.DefaultExtension || enabledExtMap[kb.Extension] {
result = append(result, kb)
}
}
return result
}
// GetKeyBindingByID 根据ID获取快捷键
func (s *KeyBindingService) GetKeyBindingByID(ctx context.Context, id int) (*ent.KeyBinding, error) {
kb, err := s.db.Client.KeyBinding.Get(ctx, id)
if err != nil {
if ent.IsNotFound(err) {
return nil, nil
}
return nil, fmt.Errorf("get key binding error: %w", err)
}
return kb, nil
}
// UpdateKeyBindingKeys 更新快捷键绑定
func (s *KeyBindingService) UpdateKeyBindingKeys(ctx context.Context, id int, key string) error {
kb, err := s.GetKeyBindingByID(ctx, id)
if err != nil {
return err
}
if kb == nil {
return fmt.Errorf("key binding not found: id=%d", id)
}
update := s.db.Client.KeyBinding.UpdateOneID(kb.ID)
os := application.Get().Env.Info().OS
switch os {
case "darwin":
update.SetMacos(key)
case "windows":
update.SetWindows(key)
case "linux":
update.SetLinux(key)
default:
s.logger.Error("unknown os: %s", os)
}
return update.SetKey(key).Exec(ctx)
}
// UpdateKeyBindingEnabled 更新快捷键启用状态
func (s *KeyBindingService) UpdateKeyBindingEnabled(ctx context.Context, id int, enabled bool) error {
kb, err := s.GetKeyBindingByID(ctx, id)
if err != nil {
return err
}
if kb == nil {
return fmt.Errorf("key binding not found: id=%d", id)
}
return s.db.Client.KeyBinding.UpdateOneID(kb.ID).
SetEnabled(enabled).
Exec(ctx)
}
// UpdateKeyBindingPreventDefault 更新快捷键 PreventDefault 状态
func (s *KeyBindingService) UpdateKeyBindingPreventDefault(ctx context.Context, id int, preventDefault bool) error {
kb, err := s.GetKeyBindingByID(ctx, id)
if err != nil {
return err
}
if kb == nil {
return fmt.Errorf("key binding not found: id=%d", id)
}
return s.db.Client.KeyBinding.UpdateOneID(kb.ID).
SetPreventDefault(preventDefault).
Exec(ctx)
}
// GetDefaultKeyBindings 获取默认快捷键配置
func (s *KeyBindingService) GetDefaultKeyBindings() []models.KeyBinding {
return models.NewDefaultKeyBindings()
}
// ResetKeyBindings 重置所有快捷键到默认值
func (s *KeyBindingService) ResetKeyBindings(ctx context.Context) error {
// 获取默认快捷键
defaults := models.NewDefaultKeyBindings()
// 构建默认快捷键映射 (type:name -> KeyBinding)
defaultsMap := make(map[string]models.KeyBinding)
for _, kb := range defaults {
key := string(kb.Type) + ":" + string(kb.Name)
defaultsMap[key] = kb
}
// 获取数据库中所有快捷键
existing, err := s.db.Client.KeyBinding.Query().All(ctx)
if err != nil {
return fmt.Errorf("query key bindings error: %w", err)
}
// 更新所有快捷键到默认值
for _, existingKb := range existing {
key := existingKb.Type + ":" + existingKb.Name
defaultKb, ok := defaultsMap[key]
if !ok {
// 如果默认配置中没有这个快捷键,跳过
continue
}
// 无条件更新所有字段到默认值
update := s.db.Client.KeyBinding.UpdateOneID(existingKb.ID).
SetKey(defaultKb.Key).
SetMacos(defaultKb.Macos).
SetWindows(defaultKb.Windows).
SetLinux(defaultKb.Linux).
SetScope(defaultKb.Scope).
SetEnabled(defaultKb.Enabled).
SetPreventDefault(defaultKb.PreventDefault)
if err := update.Exec(ctx); err != nil {
return fmt.Errorf("update key binding error: %w", err)
}
}
return nil
}