import (
"flag"
"fmt"
- "io/ioutil"
"os"
- "path/filepath"
- "runtime"
- "strings"
"time"
"github.com/go-errors/errors"
- "github.com/layeh/gopher-luar"
- "github.com/mattn/go-isatty"
- "github.com/mitchellh/go-homedir"
- "github.com/yuin/gopher-lua"
- "github.com/zyedidia/clipboard"
+ homedir "github.com/mitchellh/go-homedir"
+ "github.com/zyedidia/micro/cmd/micro/terminfo"
"github.com/zyedidia/tcell"
- "github.com/zyedidia/tcell/encoding"
)
const (
- synLinesUp = 75 // How many lines up to look to do syntax highlighting
- synLinesDown = 75 // How many lines down to look to do syntax highlighting
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
// 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
// Version is the version number or commit hash
// These variables should be set by the linker when compiling
- Version = "0.0.0-unknown"
- CommitHash = "Unknown"
+ 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"
+ // Debug logging
+ Debug = "ON"
- // L is the lua state
- // This is the VM that runs the plugins
- L *lua.LState
-
- // 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
-)
-
-// LoadInput determines which files should be loaded into buffers
-// based on the input stored in flag.Args()
-func LoadInput() []*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
+ // How many redraws have happened
+ numRedraw uint
- // 2. If there is no input file and the input is not a terminal, that means
- // something is being piped in and the stdin should be opened in an
- // empty buffer
-
- // 3. If there is no input file and the input is a terminal, an empty buffer
- // should be opened
-
- var filename string
- var input []byte
- var err error
- var buffers []*Buffer
-
- if len(flag.Args()) > 0 {
- // Option 1
- // We go through each file and load it
- for i := 0; i < len(flag.Args()); i++ {
- filename = flag.Args()[i]
-
- // Check that the file exists
- if _, e := os.Stat(filename); e == nil {
- // If it exists we load it into a buffer
- input, err = ioutil.ReadFile(filename)
- if err != nil {
- TermMessage(err)
- input = []byte{}
- filename = ""
- }
- }
- // If the file didn't exist, input will be empty, and we'll open an empty buffer
- buffers = append(buffers, NewBuffer(input, filename))
- }
- } else if !isatty.IsTerminal(os.Stdin.Fd()) {
- // Option 2
- // The input is not a terminal, so something is being piped in
- // and we should read from stdin
- input, err = ioutil.ReadAll(os.Stdin)
- if err != nil {
- TermMessage("Error reading from stdin: ", err)
- input = []byte{}
- }
- buffers = append(buffers, NewBuffer(input, filename))
- } else {
- // Option 3, just open an empty buffer
- buffers = append(buffers, NewBuffer(input, filename))
- }
-
- return buffers
-}
+ // Command line flags
+ flagVersion = flag.Bool("version", false, "Show the version number and information")
+ flagStartPos = flag.String("startpos", "", "LINE,COL to start the cursor at when opening a buffer.")
+ flagConfigDir = flag.String("config-dir", "", "Specify a custom location for the configuration directory")
+ flagOptions = flag.Bool("options", false, "Show all option help")
+)
// InitConfigDir finds the configuration directory for micro according to the XDG spec.
// If no directory is found, it creates one.
}
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)
// Should we enable true color?
truecolor := os.Getenv("MICRO_TRUECOLOR") == "1"
+ tcelldb := os.Getenv("TCELLDB")
+ os.Setenv("TCELLDB", configDir+"/.tcelldb")
+
// 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")
var err error
screen, err = tcell.NewScreen()
if err != nil {
- fmt.Println(err)
- os.Exit(1)
+ if err == tcell.ErrTermNotFound {
+ err = terminfo.WriteDB(configDir + "/.tcelldb")
+ if err != nil {
+ fmt.Println(err)
+ fmt.Println("Fatal: Micro could not create tcelldb")
+ os.Exit(1)
+ }
+ 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.Setenv("TERM", oldTerm)
}
- screen.SetStyle(defStyle)
- screen.EnableMouse()
-}
-
-// RedrawAll redraws everything -- all the views and the messenger
-func RedrawAll() {
- messenger.Clear()
- for _, v := range tabs[curTab].views {
- v.Display()
+ if GetGlobalOption("mouse").(bool) {
+ screen.EnableMouse()
}
- DisplayTabs()
- messenger.Display()
- screen.Show()
-}
-func LoadAll() {
- // Find the user's configuration directory (probably $XDG_CONFIG_HOME/micro)
- InitConfigDir()
+ os.Setenv("TCELLDB", tcelldb)
- // Build a list of available Extensions (Syntax, Colorscheme etc.)
- InitRuntimeFiles()
-
- // Load the user's settings
- InitGlobalSettings()
-
- InitCommands()
- InitBindings()
-
- LoadSyntaxFiles()
-
- for _, tab := range tabs {
- for _, v := range tab.views {
- v.Buf.UpdateRules()
- if v.Buf.Settings["syntax"].(bool) {
- v.matches = Match(v)
- }
- }
- }
+ // screen.SetStyle(defStyle)
}
-// Passing -version as a flag will have micro print out the version number
-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.")
-
-func main() {
+func InitFlags() {
flag.Usage = func() {
fmt.Println("Usage: micro [OPTIONS] [FILE]...")
- fmt.Print("Micro's options can be set via command line arguments for quick adjustments. For real configuration, please use the bindings.json file (see 'help options').\n\n")
- flag.PrintDefaults()
+ 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")
}
optionFlags := make(map[string]*string)
os.Exit(0)
}
- // Start the Lua VM for running plugins
- L = lua.NewState()
- defer L.Close()
+ if *flagOptions {
+ // If -options was passed
+ for k, v := range DefaultGlobalSettings() {
+ fmt.Printf("-%s value\n", k)
+ fmt.Printf(" \tDefault value: '%v'\n", v)
+ }
+ os.Exit(0)
+ }
+}
- // Some encoding stuff in case the user isn't using UTF-8
- encoding.Register()
- tcell.SetEncodingFallback(tcell.EncodingFallbackASCII)
+func main() {
+ var err error
- LoadAll()
+ InitLog()
+ InitFlags()
+ InitConfigDir()
+ InitRuntimeFiles()
+ err = ReadSettings()
+ if err != nil {
+ TermMessage(err)
+ }
+ InitGlobalSettings()
+ err = InitColorscheme()
+ if err != nil {
+ TermMessage(err)
+ }
- // 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() {
}
}()
- // Create a new messenger
- // This is used for sending the user messages in the bottom of the editor
- messenger = new(Messenger)
- messenger.history = make(map[string][]string)
-
- // Now we load the input
- buffers := LoadInput()
- 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)
- if globalSettings["syntax"].(bool) {
- v.matches = Match(v)
- }
- }
-
- t.Resize()
- }
- }
-
- for k, v := range optionFlags {
- if *v != "" {
- SetOption(k, *v)
- }
- }
+ b, err := NewBufferFromFile(os.Args[1])
- // 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("GetLeadingWhitespace", luar.New(L, GetLeadingWhitespace))
- L.SetGlobal("MakeCompletion", luar.New(L, MakeCompletion))
- L.SetGlobal("NewBuffer", luar.New(L, NewBuffer))
- 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("JoinPaths", luar.New(L, filepath.Join))
- 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))
-
- jobs = make(chan JobFunction, 100)
- events = make(chan tcell.Event, 100)
- autosave = make(chan bool)
-
- LoadPlugins()
-
- // Load the syntax files, including the colorscheme
- LoadSyntaxFiles()
-
- for _, t := range tabs {
- for _, v := range t.views {
- v.Buf.FindFileType()
- v.Buf.UpdateRules()
- for _, pl := range loadedPlugins {
- _, err := Call(pl+".onViewOpen", v)
- if err != nil && !strings.HasPrefix(err.Error(), "function does not exist") {
- TermMessage(err)
- continue
- }
- }
- if v.Buf.Settings["syntax"].(bool) {
- v.matches = Match(v)
- }
- }
+ if err != nil {
+ TermMessage(err)
}
- // Here is the event loop which runs in a separate thread
- go func() {
- for {
- events <- screen.PollEvent()
- }
- }()
-
- go func() {
- for {
- time.Sleep(autosaveTime * time.Second)
- if globalSettings["autosave"].(bool) {
- autosave <- true
- }
- }
- }()
-
- for {
- // Display everything
- RedrawAll()
-
- var event tcell.Event
-
- // Check for new events
- select {
- case f := <-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:
- CurView().Save(true)
- case event = <-events:
- }
+ width, height := screen.Size()
- for event != nil {
- switch e := event.(type) {
- case *tcell.EventMouse:
- 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
- }
- }
- }
- }
- }
-
- // 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
- }
+ w := NewWindow(0, 0, width/2, height/2, b)
- 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)
- }
+ for i := 0; i < 5; i++ {
+ screen.Clear()
+ w.DisplayBuffer()
+ w.DisplayStatusLine()
+ screen.Show()
+ time.Sleep(200 * time.Millisecond)
+ w.StartLine++
+ }
- select {
- case event = <-events:
- default:
- event = nil
- }
+ // time.Sleep(2 * time.Second)
- }
- }
+ screen.Fini()
}