"fmt"
"io/ioutil"
"os"
- "path/filepath"
"runtime"
- "strings"
+ "sort"
"time"
"github.com/go-errors/errors"
- "github.com/mattn/go-isatty"
- "github.com/mitchellh/go-homedir"
- "github.com/yuin/gopher-lua"
- "github.com/zyedidia/clipboard"
- "github.com/zyedidia/micro/cmd/micro/terminfo"
+ isatty "github.com/mattn/go-isatty"
+ "github.com/zyedidia/micro/internal/action"
+ "github.com/zyedidia/micro/internal/buffer"
+ "github.com/zyedidia/micro/internal/config"
+ "github.com/zyedidia/micro/internal/screen"
+ "github.com/zyedidia/micro/internal/shell"
+ "github.com/zyedidia/micro/internal/util"
"github.com/zyedidia/tcell"
- "github.com/zyedidia/tcell/encoding"
- "layeh.com/gopher-luar"
-)
-
-const (
- doubleClickThreshold = 400 // How many milliseconds to wait before a second click is not a double click
- undoThreshold = 500 // If two events are less than n milliseconds apart, undo both of them
- autosaveTime = 8 // Number of seconds to wait before autosaving
)
var (
- // The main screen
- screen tcell.Screen
-
- // Object to send messages and prompts to the user
- messenger *Messenger
-
- // The default highlighting style
- // This simply defines the default foreground and background colors
- defStyle tcell.Style
-
- // Where the user's configuration is
- // This should be $XDG_CONFIG_HOME/micro
- // If $XDG_CONFIG_HOME is not set, it is ~/.config/micro
- configDir string
-
- // Version is the version number or commit hash
- // These variables should be set by the linker when compiling
- Version = "0.0.0-unknown"
- // CommitHash is the commit this version was built on
- CommitHash = "Unknown"
- // CompileDate is the date this binary was compiled on
- CompileDate = "Unknown"
-
- // The list of views
- tabs []*Tab
- // This is the currently open tab
- // It's just an index to the tab in the tabs array
- curTab int
-
- // Channel of jobs running in the background
- jobs chan JobFunction
-
// Event channel
events chan tcell.Event
autosave chan bool
- // Channels for the terminal emulator
- updateterm chan bool
- closeterm chan int
-
- // How many redraws have happened
- numRedraw uint
+ // Command line flags
+ flagVersion = flag.Bool("version", false, "Show the version number and information")
+ flagConfigDir = flag.String("config-dir", "", "Specify a custom location for the configuration directory")
+ flagOptions = flag.Bool("options", false, "Show all option help")
+ flagDebug = flag.Bool("debug", false, "Enable debug mode (prints debug info to ./log.txt)")
+ flagPlugin = flag.String("plugin", "", "Plugin command")
+ flagClean = flag.Bool("clean", false, "Clean configuration directory")
+ optionFlags map[string]*string
)
+func InitFlags() {
+ flag.Usage = func() {
+ fmt.Println("Usage: micro [OPTIONS] [FILE]...")
+ fmt.Println("-clean")
+ fmt.Println(" \tCleans the configuration directory")
+ fmt.Println("-config-dir dir")
+ fmt.Println(" \tSpecify a custom location for the configuration directory")
+ fmt.Println("[FILE]:LINE:COL")
+ fmt.Println(" \tSpecify a line and column to start the cursor at when opening a buffer")
+ fmt.Println("-options")
+ fmt.Println(" \tShow all option help")
+ fmt.Println("-debug")
+ fmt.Println(" \tEnable debug mode (enables logging to ./log.txt)")
+ fmt.Println("-version")
+ fmt.Println(" \tShow the version number and information")
+
+ fmt.Print("\nMicro's plugin's can be managed at the command line with the following commands.\n")
+ fmt.Println("-plugin install [PLUGIN]...")
+ fmt.Println(" \tInstall plugin(s)")
+ fmt.Println("-plugin remove [PLUGIN]...")
+ fmt.Println(" \tRemove plugin(s)")
+ fmt.Println("-plugin update [PLUGIN]...")
+ fmt.Println(" \tUpdate plugin(s) (if no argument is given, updates all plugins)")
+ fmt.Println("-plugin search [PLUGIN]...")
+ fmt.Println(" \tSearch for a plugin")
+ fmt.Println("-plugin list")
+ fmt.Println(" \tList installed plugins")
+ fmt.Println("-plugin available")
+ fmt.Println(" \tList available plugins")
+
+ fmt.Print("\nMicro's options can also be set via command line arguments for quick\nadjustments. For real configuration, please use the settings.json\nfile (see 'help options').\n\n")
+ fmt.Println("-option value")
+ fmt.Println(" \tSet `option` to `value` for this session")
+ fmt.Println(" \tFor example: `micro -syntax off file.c`")
+ fmt.Println("\nUse `micro -options` to see the full list of configuration options")
+ }
+
+ optionFlags = make(map[string]*string)
+
+ for k, v := range config.DefaultAllSettings() {
+ optionFlags[k] = flag.String(k, "", fmt.Sprintf("The %s option. Default value: '%v'.", k, v))
+ }
+
+ flag.Parse()
+
+ if *flagVersion {
+ // If -version was passed
+ fmt.Println("Version:", util.Version)
+ fmt.Println("Commit hash:", util.CommitHash)
+ fmt.Println("Compiled on", util.CompileDate)
+ os.Exit(0)
+ }
+
+ if *flagOptions {
+ // If -options was passed
+ var keys []string
+ m := config.DefaultAllSettings()
+ for k := range m {
+ keys = append(keys, k)
+ }
+ sort.Strings(keys)
+ for _, k := range keys {
+ v := m[k]
+ fmt.Printf("-%s value\n", k)
+ fmt.Printf(" \tDefault value: '%v'\n", v)
+ }
+ os.Exit(0)
+ }
+
+ if util.Debug == "OFF" && *flagDebug {
+ util.Debug = "ON"
+ }
+}
+
+// DoPluginFlags parses and executes any flags that require LoadAllPlugins (-plugin and -clean)
+func DoPluginFlags() {
+ if *flagClean || *flagPlugin != "" {
+ config.LoadAllPlugins()
+
+ if *flagPlugin != "" {
+ args := flag.Args()
+
+ config.PluginCommand(os.Stdout, *flagPlugin, args)
+ } else if *flagClean {
+ CleanConfig()
+ }
+
+ os.Exit(0)
+ }
+}
+
// LoadInput determines which files should be loaded into buffers
// based on the input stored in flag.Args()
-func LoadInput() []*Buffer {
+func LoadInput() []*buffer.Buffer {
// There are a number of ways micro should start given its input
// 1. If it is given a files in flag.Args(), it should open those
var input []byte
var err error
args := flag.Args()
- buffers := make([]*Buffer, 0, len(args))
+ buffers := make([]*buffer.Buffer, 0, len(args))
if len(args) > 0 {
// Option 1
// We go through each file and load it
for i := 0; i < len(args); i++ {
- if strings.HasPrefix(args[i], "+") {
- if strings.Contains(args[i], ":") {
- split := strings.Split(args[i], ":")
- *flagStartPos = split[0][1:] + "," + split[1]
- } else {
- *flagStartPos = args[i][1:] + ",0"
- }
- continue
- }
-
- buf, err := NewBufferFromFile(args[i])
+ buf, err := buffer.NewBufferFromFile(args[i], buffer.BTDefault)
if err != nil {
- TermMessage(err)
+ screen.TermMessage(err)
continue
}
// If the file didn't exist, input will be empty, and we'll open an empty buffer
// and we should read from stdin
input, err = ioutil.ReadAll(os.Stdin)
if err != nil {
- TermMessage("Error reading from stdin: ", err)
+ screen.TermMessage("Error reading from stdin: ", err)
input = []byte{}
}
- buffers = append(buffers, NewBufferFromString(string(input), filename))
+ buffers = append(buffers, buffer.NewBufferFromString(string(input), filename, buffer.BTDefault))
} else {
// Option 3, just open an empty buffer
- buffers = append(buffers, NewBufferFromString(string(input), filename))
+ buffers = append(buffers, buffer.NewBufferFromString(string(input), filename, buffer.BTDefault))
}
return buffers
}
-// InitConfigDir finds the configuration directory for micro according to the XDG spec.
-// If no directory is found, it creates one.
-func InitConfigDir() {
- xdgHome := os.Getenv("XDG_CONFIG_HOME")
- if xdgHome == "" {
- // The user has not set $XDG_CONFIG_HOME so we should act like it was set to ~/.config
- home, err := homedir.Dir()
- if err != nil {
- TermMessage("Error finding your home directory\nCan't load config files")
- return
- }
- xdgHome = home + "/.config"
- }
- configDir = xdgHome + "/micro"
-
- if len(*flagConfigDir) > 0 {
- if _, err := os.Stat(*flagConfigDir); os.IsNotExist(err) {
- TermMessage("Error: " + *flagConfigDir + " does not exist. Defaulting to " + configDir + ".")
- } else {
- configDir = *flagConfigDir
- return
- }
- }
-
- if _, err := os.Stat(xdgHome); os.IsNotExist(err) {
- // If the xdgHome doesn't exist we should create it
- err = os.Mkdir(xdgHome, os.ModePerm)
- if err != nil {
- TermMessage("Error creating XDG_CONFIG_HOME directory: " + err.Error())
- }
- }
-
- if _, err := os.Stat(configDir); os.IsNotExist(err) {
- // If the micro specific config directory doesn't exist we should create that too
- err = os.Mkdir(configDir, os.ModePerm)
- if err != nil {
- TermMessage("Error creating configuration directory: " + err.Error())
- }
- }
-}
-
-// InitScreen creates and initializes the tcell screen
-func InitScreen() {
- // Should we enable true color?
- truecolor := os.Getenv("MICRO_TRUECOLOR") == "1"
-
- tcelldb := os.Getenv("TCELLDB")
- os.Setenv("TCELLDB", configDir+"/.tcelldb")
+func main() {
+ defer os.Exit(0)
- // In order to enable true color, we have to set the TERM to `xterm-truecolor` when
- // initializing tcell, but after that, we can set the TERM back to whatever it was
- oldTerm := os.Getenv("TERM")
- if truecolor {
- os.Setenv("TERM", "xterm-truecolor")
- }
+ // runtime.SetCPUProfileRate(400)
+ // f, _ := os.Create("micro.prof")
+ // pprof.StartCPUProfile(f)
+ // defer pprof.StopCPUProfile()
- // Initilize tcell
var err error
- screen, err = tcell.NewScreen()
- if err != nil {
- if err == tcell.ErrTermNotFound {
- terminfo.WriteDB(configDir + "/.tcelldb")
- screen, err = tcell.NewScreen()
- if err != nil {
- fmt.Println(err)
- fmt.Println("Fatal: Micro could not initialize a screen.")
- os.Exit(1)
- }
- } else {
- fmt.Println(err)
- fmt.Println("Fatal: Micro could not initialize a screen.")
- os.Exit(1)
- }
- }
- if err = screen.Init(); err != nil {
- fmt.Println(err)
- os.Exit(1)
- }
-
- // Now we can put the TERM back to what it was before
- if truecolor {
- os.Setenv("TERM", oldTerm)
- }
-
- if GetGlobalOption("mouse").(bool) {
- screen.EnableMouse()
- }
-
- os.Setenv("TCELLDB", tcelldb)
-
- // screen.SetStyle(defStyle)
-}
-
-// RedrawAll redraws everything -- all the views and the messenger
-func RedrawAll() {
- messenger.Clear()
-
- w, h := screen.Size()
- for x := 0; x < w; x++ {
- for y := 0; y < h; y++ {
- screen.SetContent(x, y, ' ', nil, defStyle)
- }
- }
-
- for _, v := range tabs[curTab].Views {
- v.Display()
- }
- DisplayTabs()
- messenger.Display()
- if globalSettings["keymenu"].(bool) {
- DisplayKeyMenu()
- }
- screen.Show()
-
- if numRedraw%50 == 0 {
- runtime.GC()
- }
- numRedraw++
-}
-
-func LoadAll() {
- // Find the user's configuration directory (probably $XDG_CONFIG_HOME/micro)
- InitConfigDir()
-
- // Build a list of available Extensions (Syntax, Colorscheme etc.)
- InitRuntimeFiles()
-
- // Load the user's settings
- InitGlobalSettings()
-
- InitCommands()
- InitBindings()
-
- InitColorscheme()
-
- for _, tab := range tabs {
- for _, v := range tab.Views {
- v.Buf.UpdateRules()
- }
- }
-}
-
-// Command line flags
-var flagVersion = flag.Bool("version", false, "Show the version number and information")
-var flagStartPos = flag.String("startpos", "", "LINE,COL to start the cursor at when opening a buffer.")
-var flagConfigDir = flag.String("config-dir", "", "Specify a custom location for the configuration directory")
-var flagOptions = flag.Bool("options", false, "Show all option help")
-
-func main() {
- flag.Usage = func() {
- fmt.Println("Usage: micro [OPTIONS] [FILE]...")
- fmt.Println("-config-dir dir")
- fmt.Println(" \tSpecify a custom location for the configuration directory")
- fmt.Println("-startpos LINE,COL")
- fmt.Println("+LINE:COL")
- fmt.Println(" \tSpecify a line and column to start the cursor at when opening a buffer")
- fmt.Println(" \tThis can also be done by opening file:LINE:COL")
- fmt.Println("-options")
- fmt.Println(" \tShow all option help")
- fmt.Println("-version")
- fmt.Println(" \tShow the version number and information")
- fmt.Print("\nMicro's options can also be set via command line arguments for quick\nadjustments. For real configuration, please use the settings.json\nfile (see 'help options').\n\n")
- fmt.Println("-option value")
- fmt.Println(" \tSet `option` to `value` for this session")
- fmt.Println(" \tFor example: `micro -syntax off file.c`")
- fmt.Println("\nUse `micro -options` to see the full list of configuration options")
- }
+ InitFlags()
- optionFlags := make(map[string]*string)
+ InitLog()
- for k, v := range DefaultGlobalSettings() {
- optionFlags[k] = flag.String(k, "", fmt.Sprintf("The %s option. Default value: '%v'", k, v))
+ err = config.InitConfigDir(*flagConfigDir)
+ if err != nil {
+ screen.TermMessage(err)
}
- flag.Parse()
-
- if *flagVersion {
- // If -version was passed
- fmt.Println("Version:", Version)
- fmt.Println("Commit hash:", CommitHash)
- fmt.Println("Compiled on", CompileDate)
- os.Exit(0)
+ config.InitRuntimeFiles()
+ err = config.ReadSettings()
+ if err != nil {
+ screen.TermMessage(err)
}
+ config.InitGlobalSettings()
- if *flagOptions {
- // If -options was passed
- for k, v := range DefaultGlobalSettings() {
- fmt.Printf("-%s value\n", k)
- fmt.Printf(" \tThe %s option. Default value: '%v'\n", k, v)
+ // flag options
+ for k, v := range optionFlags {
+ if *v != "" {
+ nativeValue, err := config.GetNativeValue(k, config.DefaultAllSettings()[k], *v)
+ if err != nil {
+ screen.TermMessage(err)
+ continue
+ }
+ config.GlobalSettings[k] = nativeValue
}
- os.Exit(0)
}
- // Start the Lua VM for running plugins
- L = lua.NewState()
- defer L.Close()
-
- // Some encoding stuff in case the user isn't using UTF-8
- encoding.Register()
- tcell.SetEncodingFallback(tcell.EncodingFallbackASCII)
+ DoPluginFlags()
- // Find the user's configuration directory (probably $XDG_CONFIG_HOME/micro)
- InitConfigDir()
+ screen.Init()
- // Build a list of available Extensions (Syntax, Colorscheme etc.)
- InitRuntimeFiles()
-
- // Load the user's settings
- InitGlobalSettings()
-
- InitCommands()
- InitBindings()
-
- // Start the screen
- InitScreen()
-
- // This is just so if we have an error, we can exit cleanly and not completely
+ // If we have an error, we can exit cleanly and not completely
// mess up the terminal being worked in
// In other words we need to shut down tcell before the program crashes
defer func() {
if err := recover(); err != nil {
- screen.Fini()
+ screen.Screen.Fini()
fmt.Println("Micro encountered an error:", err)
+ // backup all open buffers
+ for _, b := range buffer.OpenBuffers {
+ b.Backup(false)
+ }
// Print the stack trace too
fmt.Print(errors.Wrap(err, 2).ErrorStack())
os.Exit(1)
}
}()
- // Create a new messenger
- // This is used for sending the user messages in the bottom of the editor
- messenger = new(Messenger)
- messenger.LoadHistory()
-
- // Now we load the input
- buffers := LoadInput()
- if len(buffers) == 0 {
- screen.Fini()
- os.Exit(1)
+ err = config.LoadAllPlugins()
+ if err != nil {
+ screen.TermMessage(err)
}
- for _, buf := range buffers {
- // For each buffer we create a new tab and place the view in that tab
- tab := NewTabFromView(NewView(buf))
- tab.SetNum(len(tabs))
- tabs = append(tabs, tab)
- for _, t := range tabs {
- for _, v := range t.Views {
- v.Center(false)
- }
+ action.InitBindings()
+ action.InitCommands()
- t.Resize()
- }
+ err = config.InitColorscheme()
+ if err != nil {
+ screen.TermMessage(err)
}
- for k, v := range optionFlags {
- if *v != "" {
- SetOption(k, *v)
- }
+ b := LoadInput()
+
+ if len(b) == 0 {
+ // No buffers to open
+ screen.Screen.Fini()
+ runtime.Goexit()
}
- // Load all the plugin stuff
- // We give plugins access to a bunch of variables here which could be useful to them
- L.SetGlobal("OS", luar.New(L, runtime.GOOS))
- L.SetGlobal("tabs", luar.New(L, tabs))
- L.SetGlobal("curTab", luar.New(L, curTab))
- L.SetGlobal("messenger", luar.New(L, messenger))
- L.SetGlobal("GetOption", luar.New(L, GetOption))
- L.SetGlobal("AddOption", luar.New(L, AddOption))
- L.SetGlobal("SetOption", luar.New(L, SetOption))
- L.SetGlobal("SetLocalOption", luar.New(L, SetLocalOption))
- L.SetGlobal("BindKey", luar.New(L, BindKey))
- L.SetGlobal("MakeCommand", luar.New(L, MakeCommand))
- L.SetGlobal("CurView", luar.New(L, CurView))
- L.SetGlobal("IsWordChar", luar.New(L, IsWordChar))
- L.SetGlobal("HandleCommand", luar.New(L, HandleCommand))
- L.SetGlobal("HandleShellCommand", luar.New(L, HandleShellCommand))
- L.SetGlobal("ExecCommand", luar.New(L, ExecCommand))
- L.SetGlobal("RunShellCommand", luar.New(L, RunShellCommand))
- L.SetGlobal("RunBackgroundShell", luar.New(L, RunBackgroundShell))
- L.SetGlobal("RunInteractiveShell", luar.New(L, RunInteractiveShell))
- L.SetGlobal("TermEmuSupported", luar.New(L, TermEmuSupported))
- L.SetGlobal("RunTermEmulator", luar.New(L, RunTermEmulator))
- L.SetGlobal("GetLeadingWhitespace", luar.New(L, GetLeadingWhitespace))
- L.SetGlobal("MakeCompletion", luar.New(L, MakeCompletion))
- L.SetGlobal("NewBuffer", luar.New(L, NewBufferFromString))
- L.SetGlobal("NewBufferFromFile", luar.New(L, NewBufferFromFile))
- L.SetGlobal("RuneStr", luar.New(L, func(r rune) string {
- return string(r)
- }))
- L.SetGlobal("Loc", luar.New(L, func(x, y int) Loc {
- return Loc{x, y}
- }))
- L.SetGlobal("WorkingDirectory", luar.New(L, os.Getwd))
- L.SetGlobal("JoinPaths", luar.New(L, filepath.Join))
- L.SetGlobal("DirectoryName", luar.New(L, filepath.Dir))
- L.SetGlobal("configDir", luar.New(L, configDir))
- L.SetGlobal("Reload", luar.New(L, LoadAll))
- L.SetGlobal("ByteOffset", luar.New(L, ByteOffset))
- L.SetGlobal("ToCharPos", luar.New(L, ToCharPos))
-
- // Used for asynchronous jobs
- L.SetGlobal("JobStart", luar.New(L, JobStart))
- L.SetGlobal("JobSpawn", luar.New(L, JobSpawn))
- L.SetGlobal("JobSend", luar.New(L, JobSend))
- L.SetGlobal("JobStop", luar.New(L, JobStop))
-
- // Extension Files
- L.SetGlobal("ReadRuntimeFile", luar.New(L, PluginReadRuntimeFile))
- L.SetGlobal("ListRuntimeFiles", luar.New(L, PluginListRuntimeFiles))
- L.SetGlobal("AddRuntimeFile", luar.New(L, PluginAddRuntimeFile))
- L.SetGlobal("AddRuntimeFilesFromDirectory", luar.New(L, PluginAddRuntimeFilesFromDirectory))
- L.SetGlobal("AddRuntimeFileFromMemory", luar.New(L, PluginAddRuntimeFileFromMemory))
-
- // Access to Go stdlib
- L.SetGlobal("import", luar.New(L, Import))
-
- jobs = make(chan JobFunction, 100)
- events = make(chan tcell.Event, 100)
- autosave = make(chan bool)
- updateterm = make(chan bool)
- closeterm = make(chan int)
-
- LoadPlugins()
-
- for _, t := range tabs {
- for _, v := range t.Views {
- GlobalPluginCall("onViewOpen", v)
- GlobalPluginCall("onBufferOpen", v.Buf)
- }
+ action.InitTabs(b)
+ action.InitGlobals()
+
+ err = config.RunPluginFn("init")
+ if err != nil {
+ screen.TermMessage(err)
}
- InitColorscheme()
- messenger.style = defStyle
+ events = make(chan tcell.Event)
// Here is the event loop which runs in a separate thread
go func() {
for {
- if screen != nil {
- events <- screen.PollEvent()
+ screen.Lock()
+ e := screen.Screen.PollEvent()
+ screen.Unlock()
+ if e != nil {
+ events <- e
}
}
}()
- go func() {
- for {
- time.Sleep(autosaveTime * time.Second)
- if globalSettings["autosave"].(bool) {
- autosave <- true
- }
- }
- }()
+ // clear the drawchan so we don't redraw excessively
+ // if someone requested a redraw before we started displaying
+ for len(screen.DrawChan) > 0 {
+ <-screen.DrawChan
+ }
+
+ var event tcell.Event
+
+ // wait for initial resize event
+ select {
+ case event = <-events:
+ action.Tabs.HandleEvent(event)
+ case <-time.After(10 * time.Millisecond):
+ // time out after 10ms
+ }
for {
// Display everything
- RedrawAll()
+ screen.Screen.Fill(' ', config.DefStyle)
+ screen.Screen.HideCursor()
+ action.Tabs.Display()
+ for _, ep := range action.MainTab().Panes {
+ ep.Display()
+ }
+ action.MainTab().Display()
+ action.InfoBar.Display()
+ screen.Screen.Show()
- var event tcell.Event
+ event = nil
// Check for new events
select {
- case f := <-jobs:
+ case f := <-shell.Jobs:
// If a new job has finished while running in the background we should execute the callback
- f.function(f.output, f.args...)
- continue
- case <-autosave:
- if CurView().Buf.Path != "" {
- CurView().Save(true)
+ f.Function(f.Output, f.Args...)
+ case <-config.Autosave:
+ for _, b := range buffer.OpenBuffers {
+ b.Save()
}
- case <-updateterm:
- continue
- case vnum := <-closeterm:
- tabs[curTab].Views[vnum].CloseTerminal()
+ case <-shell.CloseTerms:
case event = <-events:
+ case <-screen.DrawChan:
}
- for event != nil {
- didAction := false
-
- switch e := event.(type) {
- case *tcell.EventResize:
- for _, t := range tabs {
- t.Resize()
- }
- case *tcell.EventMouse:
- if !searching {
- if e.Buttons() == tcell.Button1 {
- // If the user left clicked we check a couple things
- _, h := screen.Size()
- x, y := e.Position()
- if y == h-1 && messenger.message != "" && globalSettings["infobar"].(bool) {
- // If the user clicked in the bottom bar, and there is a message down there
- // we copy it to the clipboard.
- // Often error messages are displayed down there so it can be useful to easily
- // copy the message
- clipboard.WriteAll(messenger.message, "primary")
- break
- }
-
- if CurView().mouseReleased {
- // We loop through each view in the current tab and make sure the current view
- // is the one being clicked in
- for _, v := range tabs[curTab].Views {
- if x >= v.x && x < v.x+v.Width && y >= v.y && y < v.y+v.Height {
- tabs[curTab].CurView = v.Num
- }
- }
- }
- } else if e.Buttons() == tcell.WheelUp || e.Buttons() == tcell.WheelDown {
- var view *View
- x, y := e.Position()
- for _, v := range tabs[curTab].Views {
- if x >= v.x && x < v.x+v.Width && y >= v.y && y < v.y+v.Height {
- view = tabs[curTab].Views[v.Num]
- }
- }
- if view != nil {
- view.HandleEvent(e)
- didAction = true
- }
- }
- }
- }
-
- if !didAction {
- // This function checks the mouse event for the possibility of changing the current tab
- // If the tab was changed it returns true
- if TabbarHandleMouseEvent(event) {
- break
- }
-
- if searching {
- // Since searching is done in real time, we need to redraw every time
- // there is a new event in the search bar so we need a special function
- // to run instead of the standard HandleEvent.
- HandleSearchEvent(event, CurView())
- } else {
- // Send it to the view
- CurView().HandleEvent(event)
- }
- }
-
- select {
- case event = <-events:
- default:
- event = nil
- }
+ if action.InfoBar.HasPrompt {
+ action.InfoBar.HandleEvent(event)
+ } else {
+ action.Tabs.HandleEvent(event)
}
}
}