]> git.lizzy.rs Git - micro.git/blobdiff - internal/action/bufpane.go
Search and replace fixes
[micro.git] / internal / action / bufpane.go
index ac6c674fe14c63a22bcf3cc9cf81df996d06928d..5941a509081569a197e2cf807f20fb005f09d340 100644 (file)
@@ -4,8 +4,13 @@ import (
        "strings"
        "time"
 
+       luar "layeh.com/gopher-luar"
+
+       lua "github.com/yuin/gopher-lua"
        "github.com/zyedidia/micro/internal/buffer"
+       "github.com/zyedidia/micro/internal/config"
        "github.com/zyedidia/micro/internal/display"
+       ulua "github.com/zyedidia/micro/internal/lua"
        "github.com/zyedidia/micro/internal/screen"
        "github.com/zyedidia/tcell"
 )
@@ -23,21 +28,92 @@ func init() {
        BufMouseBindings = make(map[MouseEvent]BufMouseAction)
 }
 
+func LuaAction(fn string) func(*BufPane) bool {
+       luaFn := strings.Split(fn, ".")
+       if len(luaFn) <= 1 {
+               return nil
+       }
+       plName, plFn := luaFn[0], luaFn[1]
+       pl := config.FindPlugin(plName)
+       if pl == nil {
+               return nil
+       }
+       return func(h *BufPane) bool {
+               val, err := pl.Call(plFn, luar.New(ulua.L, h))
+               if err != nil {
+                       screen.TermMessage(err)
+               }
+               if v, ok := val.(lua.LBool); !ok {
+                       return false
+               } else {
+                       return bool(v)
+               }
+       }
+}
+
 // BufMapKey maps a key event to an action
 func BufMapKey(k Event, action string) {
-       if strings.HasPrefix(action, "command:") {
-               action = strings.SplitN(action, ":", 2)[1]
-               BufKeyStrings[k] = action
-               BufKeyBindings[k] = CommandAction(action)
-       } else if strings.HasPrefix(action, "command-edit:") {
-               action = strings.SplitN(action, ":", 2)[1]
-               BufKeyStrings[k] = action
-               BufKeyBindings[k] = CommandEditAction(action)
-       } else if f, ok := BufKeyActions[action]; ok {
-               BufKeyStrings[k] = action
-               BufKeyBindings[k] = f
-       } else {
-               screen.TermMessage("Error:", action, "does not exist")
+       BufKeyStrings[k] = action
+       var actionfns []func(*BufPane) bool
+       var names []string
+       var types []byte
+       for i := 0; ; i++ {
+               if action == "" {
+                       break
+               }
+
+               idx := strings.IndexAny(action, "&|,")
+               a := action
+               if idx >= 0 {
+                       a = action[:idx]
+                       types = append(types, action[idx])
+                       action = action[idx+1:]
+               } else {
+                       types = append(types, ' ')
+                       action = ""
+               }
+
+               var afn func(*BufPane) bool
+               if strings.HasPrefix(action, "command:") {
+                       a = strings.SplitN(a, ":", 2)[1]
+                       afn = CommandAction(a)
+                       names = append(names, "")
+               } else if strings.HasPrefix(a, "command-edit:") {
+                       a = strings.SplitN(a, ":", 2)[1]
+                       afn = CommandEditAction(a)
+                       names = append(names, "")
+               } else if strings.HasPrefix(a, "lua:") {
+                       a = strings.SplitN(a, ":", 2)[1]
+                       afn = LuaAction(a)
+                       if afn == nil {
+                               screen.TermMessage("Lua Error:", action, "does not exist")
+                               continue
+                       }
+                       names = append(names, "")
+               } else if f, ok := BufKeyActions[a]; ok {
+                       afn = f
+                       names = append(names, a)
+               } else {
+                       screen.TermMessage("Error:", action, "does not exist")
+                       continue
+               }
+               actionfns = append(actionfns, afn)
+       }
+       BufKeyBindings[k] = func(h *BufPane) bool {
+               cursors := h.Buf.GetCursors()
+               success := true
+               for i, a := range actionfns {
+                       for j, c := range cursors {
+                               h.Buf.SetCurCursor(c.Num)
+                               h.Cursor = c
+                               if i == 0 || (success && types[i-1] == '&') || (!success && types[i-1] == '|') || (types[i-1] == ',') {
+                                       success = h.execAction(a, names[i], j)
+                               } else {
+                                       break
+                               }
+                       }
+               }
+               return true
        }
 }
 
@@ -63,9 +139,6 @@ type BufPane struct {
 
        Cursor *buffer.Cursor // the active cursor
 
-       StartLine int // Vertical scrolling
-       StartCol  int // Horizontal scrolling
-
        // Since tcell doesn't differentiate between a mouse release event
        // and a mouse move event with no keys pressed, we need to keep
        // track of whether or not the mouse was pressed (or not released) last event to determine
@@ -100,6 +173,9 @@ type BufPane struct {
        multiWord bool
 
        splitID uint64
+
+       // remember original location of a search in case the search is canceled
+       searchOrig buffer.Loc
 }
 
 func NewBufPane(buf *buffer.Buffer, win display.BWindow) *BufPane {
@@ -110,6 +186,8 @@ func NewBufPane(buf *buffer.Buffer, win display.BWindow) *BufPane {
        h.Cursor = h.Buf.GetActiveCursor()
        h.mouseReleased = true
 
+       config.RunPluginFn("onBufPaneOpen", luar.New(ulua.L, h))
+
        return h
 }
 
@@ -118,11 +196,33 @@ func NewBufPaneFromBuf(buf *buffer.Buffer) *BufPane {
        return NewBufPane(buf, w)
 }
 
+// PluginCB calls all plugin callbacks with a certain name and
+// displays an error if there is one and returns the aggregrate
+// boolean response
+func (h *BufPane) PluginCB(cb string) bool {
+       b, err := config.RunPluginFnBool(cb, luar.New(ulua.L, h))
+       if err != nil {
+               screen.TermMessage(err)
+       }
+       return b
+}
+
+// PluginCBRune is the same as PluginCB but also passes a rune to
+// the plugins
+func (h *BufPane) PluginCBRune(cb string, r rune) bool {
+       b, err := config.RunPluginFnBool(cb, luar.New(ulua.L, h), luar.New(ulua.L, string(r)))
+       if err != nil {
+               screen.TermMessage(err)
+       }
+       return b
+}
+
 func (h *BufPane) OpenBuffer(b *buffer.Buffer) {
        h.Buf.Close()
        h.Buf = b
        h.BWindow.SetBuffer(b)
        h.Cursor = b.GetActiveCursor()
+       h.Resize(h.GetView().Width, h.GetView().Height)
        v := new(display.View)
        h.SetView(v)
        h.Relocate()
@@ -148,8 +248,18 @@ func (h *BufPane) Name() string {
 }
 
 // HandleEvent executes the tcell event properly
-// TODO: multiple actions bound to one key
 func (h *BufPane) HandleEvent(event tcell.Event) {
+       if h.Buf.ExternallyModified() {
+               InfoBar.YNPrompt("The file on disk has changed. Reload file? (y,n)", func(yes, canceled bool) {
+                       if !yes || canceled {
+                               h.Buf.UpdateModTime()
+                       } else {
+                               h.Buf.ReOpen()
+                       }
+               })
+
+       }
+
        switch e := event.(type) {
        case *tcell.EventRaw:
                re := RawEvent{
@@ -175,7 +285,7 @@ func (h *BufPane) HandleEvent(event tcell.Event) {
                                // Mouse was just released
 
                                mx, my := e.Position()
-                               mouseLoc := h.GetMouseLoc(buffer.Loc{X: mx, Y: my})
+                               mouseLoc := h.LocFromVisual(buffer.Loc{X: mx, Y: my})
 
                                // Relocating here isn't really necessary because the cursor will
                                // be in the right place from the last mouse event
@@ -200,18 +310,20 @@ func (h *BufPane) HandleEvent(event tcell.Event) {
        }
        h.Buf.MergeCursors()
 
-       // Display any gutter messages for this line
-       c := h.Buf.GetActiveCursor()
-       none := true
-       for _, m := range h.Buf.Messages {
-               if c.Y == m.Start.Y || c.Y == m.End.Y {
-                       InfoBar.GutterMessage(m.Msg)
-                       none = false
-                       break
+       if h.IsActive() {
+               // Display any gutter messages for this line
+               c := h.Buf.GetActiveCursor()
+               none := true
+               for _, m := range h.Buf.Messages {
+                       if c.Y == m.Start.Y || c.Y == m.End.Y {
+                               InfoBar.GutterMessage(m.Msg)
+                               none = false
+                               break
+                       }
+               }
+               if none && InfoBar.HasGutter {
+                       InfoBar.ClearGutter()
                }
-       }
-       if none && InfoBar.HasGutter {
-               InfoBar.ClearGutter()
        }
 }
 
@@ -219,25 +331,34 @@ func (h *BufPane) HandleEvent(event tcell.Event) {
 // to and executing it (possibly multiple times for multiple cursors)
 func (h *BufPane) DoKeyEvent(e Event) bool {
        if action, ok := BufKeyBindings[e]; ok {
-               estr := BufKeyStrings[e]
-               for _, s := range MultiActions {
-                       if s == estr {
-                               cursors := h.Buf.GetCursors()
-                               for _, c := range cursors {
-                                       h.Buf.SetCurCursor(c.Num)
-                                       h.Cursor = c
-                                       if action(h) {
-                                               h.Relocate()
+               return action(h)
+       }
+       return false
+}
+
+func (h *BufPane) execAction(action func(*BufPane) bool, name string, cursor int) bool {
+       if name != "Autocomplete" {
+               h.Buf.HasSuggestions = false
+       }
+
+       _, isMulti := MultiActions[name]
+       if (!isMulti && cursor == 0) || isMulti {
+               if h.PluginCB("pre" + name) {
+                       asuccess := action(h)
+                       psuccess := h.PluginCB("on" + name)
+
+                       if isMulti {
+                               if recording_macro {
+                                       if name != "ToggleMacro" && name != "PlayMacro" {
+                                               curmacro = append(curmacro, action)
                                        }
                                }
-                               return true
                        }
+
+                       return asuccess && psuccess
                }
-               if action(h) {
-                       h.Relocate()
-               }
-               return true
        }
+
        return false
 }
 
@@ -267,6 +388,10 @@ func (h *BufPane) DoRuneInsert(r rune) {
        for _, c := range cursors {
                // Insert a character
                h.Buf.SetCurCursor(c.Num)
+               h.Cursor = c
+               if !h.PluginCBRune("preRune", r) {
+                       continue
+               }
                if c.HasSelection() {
                        c.DeleteSelection()
                        c.ResetSelection()
@@ -279,27 +404,53 @@ func (h *BufPane) DoRuneInsert(r rune) {
                } else {
                        h.Buf.Insert(c.Loc, string(r))
                }
+               if recording_macro {
+                       curmacro = append(curmacro, r)
+               }
+               h.PluginCBRune("onRune", r)
        }
 }
 
-func (h *BufPane) VSplitBuf(buf *buffer.Buffer) {
+func (h *BufPane) VSplitBuf(buf *buffer.Buffer) *BufPane {
        e := NewBufPaneFromBuf(buf)
        e.splitID = MainTab().GetNode(h.splitID).VSplit(h.Buf.Settings["splitright"].(bool))
        MainTab().Panes = append(MainTab().Panes, e)
        MainTab().Resize()
        MainTab().SetActive(len(MainTab().Panes) - 1)
+       return e
 }
-func (h *BufPane) HSplitBuf(buf *buffer.Buffer) {
+func (h *BufPane) HSplitBuf(buf *buffer.Buffer) *BufPane {
        e := NewBufPaneFromBuf(buf)
        e.splitID = MainTab().GetNode(h.splitID).HSplit(h.Buf.Settings["splitbottom"].(bool))
        MainTab().Panes = append(MainTab().Panes, e)
        MainTab().Resize()
        MainTab().SetActive(len(MainTab().Panes) - 1)
+       return e
 }
 func (h *BufPane) Close() {
        h.Buf.Close()
 }
 
+func (h *BufPane) SetActive(b bool) {
+       h.BWindow.SetActive(b)
+       if b {
+               // Display any gutter messages for this line
+               c := h.Buf.GetActiveCursor()
+               none := true
+               for _, m := range h.Buf.Messages {
+                       if c.Y == m.Start.Y || c.Y == m.End.Y {
+                               InfoBar.GutterMessage(m.Msg)
+                               none = false
+                               break
+                       }
+               }
+               if none && InfoBar.HasGutter {
+                       InfoBar.ClearGutter()
+               }
+       }
+
+}
+
 // BufKeyActions contains the list of all possible key actions the bufhandler could execute
 var BufKeyActions = map[string]BufKeyAction{
        "CursorUp":               (*BufPane).CursorUp,
@@ -349,6 +500,7 @@ var BufKeyActions = map[string]BufKeyAction{
        "MoveLinesDown":          (*BufPane).MoveLinesDown,
        "IndentSelection":        (*BufPane).IndentSelection,
        "OutdentSelection":       (*BufPane).OutdentSelection,
+       "Autocomplete":           (*BufPane).Autocomplete,
        "OutdentLine":            (*BufPane).OutdentLine,
        "Paste":                  (*BufPane).Paste,
        "PastePrimary":           (*BufPane).PastePrimary,
@@ -367,7 +519,6 @@ var BufKeyActions = map[string]BufKeyAction{
        "ToggleHelp":             (*BufPane).ToggleHelp,
        "ToggleKeyMenu":          (*BufPane).ToggleKeyMenu,
        "ToggleRuler":            (*BufPane).ToggleRuler,
-       "JumpLine":               (*BufPane).JumpLine,
        "ClearStatus":            (*BufPane).ClearStatus,
        "ShellMode":              (*BufPane).ShellMode,
        "CommandMode":            (*BufPane).CommandMode,
@@ -394,6 +545,7 @@ var BufKeyActions = map[string]BufKeyAction{
        "RemoveAllMultiCursors":  (*BufPane).RemoveAllMultiCursors,
        "SkipMultiCursor":        (*BufPane).SkipMultiCursor,
        "JumpToMatchingBrace":    (*BufPane).JumpToMatchingBrace,
+       "None":                   (*BufPane).None,
 
        // This was changed to InsertNewline but I don't want to break backwards compatibility
        "InsertEnter": (*BufPane).InsertNewline,
@@ -409,52 +561,52 @@ var BufMouseActions = map[string]BufMouseAction{
 // times if there are multiple cursors (one per cursor)
 // Generally actions that modify global editor state like quitting or
 // saving should not be included in this list
-var MultiActions = []string{
-       "CursorUp",
-       "CursorDown",
-       "CursorPageUp",
-       "CursorPageDown",
-       "CursorLeft",
-       "CursorRight",
-       "CursorStart",
-       "CursorEnd",
-       "SelectToStart",
-       "SelectToEnd",
-       "SelectUp",
-       "SelectDown",
-       "SelectLeft",
-       "SelectRight",
-       "WordRight",
-       "WordLeft",
-       "SelectWordRight",
-       "SelectWordLeft",
-       "DeleteWordRight",
-       "DeleteWordLeft",
-       "SelectLine",
-       "SelectToStartOfLine",
-       "SelectToEndOfLine",
-       "ParagraphPrevious",
-       "ParagraphNext",
-       "InsertNewline",
-       "Backspace",
-       "Delete",
-       "InsertTab",
-       "FindNext",
-       "FindPrevious",
-       "Cut",
-       "CutLine",
-       "DuplicateLine",
-       "DeleteLine",
-       "MoveLinesUp",
-       "MoveLinesDown",
-       "IndentSelection",
-       "OutdentSelection",
-       "OutdentLine",
-       "Paste",
-       "PastePrimary",
-       "SelectPageUp",
-       "SelectPageDown",
-       "StartOfLine",
-       "EndOfLine",
-       "JumpToMatchingBrace",
+var MultiActions = map[string]bool{
+       "CursorUp":            true,
+       "CursorDown":          true,
+       "CursorPageUp":        true,
+       "CursorPageDown":      true,
+       "CursorLeft":          true,
+       "CursorRight":         true,
+       "CursorStart":         true,
+       "CursorEnd":           true,
+       "SelectToStart":       true,
+       "SelectToEnd":         true,
+       "SelectUp":            true,
+       "SelectDown":          true,
+       "SelectLeft":          true,
+       "SelectRight":         true,
+       "WordRight":           true,
+       "WordLeft":            true,
+       "SelectWordRight":     true,
+       "SelectWordLeft":      true,
+       "DeleteWordRight":     true,
+       "DeleteWordLeft":      true,
+       "SelectLine":          true,
+       "SelectToStartOfLine": true,
+       "SelectToEndOfLine":   true,
+       "ParagraphPrevious":   true,
+       "ParagraphNext":       true,
+       "InsertNewline":       true,
+       "Backspace":           true,
+       "Delete":              true,
+       "InsertTab":           true,
+       "FindNext":            true,
+       "FindPrevious":        true,
+       "Cut":                 true,
+       "CutLine":             true,
+       "DuplicateLine":       true,
+       "DeleteLine":          true,
+       "MoveLinesUp":         true,
+       "MoveLinesDown":       true,
+       "IndentSelection":     true,
+       "OutdentSelection":    true,
+       "OutdentLine":         true,
+       "Paste":               true,
+       "PastePrimary":        true,
+       "SelectPageUp":        true,
+       "SelectPageDown":      true,
+       "StartOfLine":         true,
+       "EndOfLine":           true,
+       "JumpToMatchingBrace": true,
 }