How can I get started with Emacs + fcsh + Flex3? - apache-flex

I've got some reasonable emacs-fu, and I know how to use the Flash Builder IDE to work in Flex/Actionscript, but I'd like to move to using Emacs to do most of my coding instead of the flash builder. This poses some problems, though, namely that I'm not sure how to go about compiling my flex project from the fcsh-mode command line.
Here's an outline of my setup:
My projects routinely depend on a combination of compiled third-party .swc files.
I use some internal libraries that I keep as Actionscript Library Projects in FB.
I occasionally recompile dependencies, but I can probably continue to do this in FB the rare time it's necessary.
Each Flex application project consists of a mix of MXML and ActionScript source.
I've never used the command line compilers in this way, so I'm a bit stumped on how to start. I keep reading about compile 1 when people talk about fcsh on the 'net, but that just gives me fcsh: Target 1 not found which I take to mean that I have to have run the 'right' compilation beforehand. But not a single web site touches on what exact mystical incantation that would be.
So, the basic question here is: What is the minimum complete set of steps starting with opening an actionscript or MXML file to a successful compilation of the file?

Here's the thing - fcsh isn't really for building projects, it's for incremental development.
The best way to figure out fcsh is to first learn how to use the mxmlc and compc on the command line. The verbosity necessary to compile your project using these tools will irritate you at some point, so you'll want to make an ant task or the like as weiji commented above.
So, after you get your ant project set up, you'll start to realize that even when you have everything set up nicely, it still takes a lot longer to recompile than it does when you use Flex Builder. The reason is incremental compilation - FB isn't actually recompiling everything, it's only recompiling the closure of the dependency tree that hangs off whatever files you changed.
Enter fcsh. It's basically a hacked version of mxmlc that includes the same incremental compilation code that's used by FB. It stores sets of partial compilation results into indexed result sets, so that's what the "compile 1" refers to; it means "rebuild the same thing as before, but only recompile the changed stuff".
Ok, that was the fact part, so now onto opinion: don't bother. Although fcsh can greatly speed up recompilation of a big fluffy source tree, the better answer is "don't organize your code that way". I personally believe that more carefully partitioning your source to build SWCs or modules, and setting up multiple independent separately compiled libraries, with interface-biased dependencies between them is a much better route. Then, when you recompile, you're generally either only rebuilding the library, or rebuilding the app, but not all other stuff as well. To me, fcsh takes too much "thinking" mid-compile, whereas a library-based project hierarchy is nearly as fast and I can be more "dumb" about invoking M-x compile.
OH! And don't forget "fdb". It's mostly gdb compatible, so M-x gdb actually works pretty well, or at least it used to when I was still involved. :-D

See
http://nisheet.wordpress.com/category/emacs/
or
http://anirudhs.chaosnet.org/code/ani-fcsh.el.html to create your .emacs file and then (optional) add actionscript-mode with the following:
To compile see: http://anirudhs.chaosnet.org/blog/2008.03.26.html
;;; as3-mode.el --- A simple mode for editing Actionscript 3 files
;; Copyright (C) 2008 Austin Haas
;; Author: Austin Haas
;; Keywords: language modes
;; Tabbing stolen from: Aemon Cannon, http://github.com/aemoncannon/as3-mode/tree/master/as3-mode.el
;; Copyright (C) 2007 Aemon Cannon
;; This file is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation; either version 2, or (at your option)
;; any later version.
;; This file is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs; see the file COPYING. If not, write to
;; the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
;; Boston, MA 02111-1307, USA.
;;; Commentary:
;;
;;------------------
;;; TODO
;; Imenu (imenu-generic-expression or imenu-create-index-function)
;;------------------
;;; Code:
(require 'font-lock)
(eval-when-compile
(require 'regexp-opt))
(defconst actionscript-mode-version "7.0"
"Actionscript Mode version number.")
(defgroup actionscript nil
"Major mode for editing Actionscript code."
:group 'languages)
(defvar actionscript-mode-syntax-table nil
"Syntax table used in actionscript-mode buffers.")
(if actionscript-mode-syntax-table
()
(setq actionscript-mode-syntax-table (make-syntax-table))
(modify-syntax-entry ?_ "w" actionscript-mode-syntax-table)
;; (modify-syntax-entry ?_ "_" actionscript-mode-syntax-table)
(modify-syntax-entry ?$ "w" actionscript-mode-syntax-table)
(modify-syntax-entry ?\\ "\\" actionscript-mode-syntax-table)
(modify-syntax-entry ?+ "." actionscript-mode-syntax-table)
(modify-syntax-entry ?- "." actionscript-mode-syntax-table)
(modify-syntax-entry ?= "." actionscript-mode-syntax-table)
(modify-syntax-entry ?% "." actionscript-mode-syntax-table)
(modify-syntax-entry ?< "." actionscript-mode-syntax-table)
(modify-syntax-entry ?> "." actionscript-mode-syntax-table)
(modify-syntax-entry ?& "." actionscript-mode-syntax-table)
(modify-syntax-entry ?| "." actionscript-mode-syntax-table)
(modify-syntax-entry ?\' "\"" actionscript-mode-syntax-table)
(modify-syntax-entry ?\240 "." actionscript-mode-syntax-table)
(modify-syntax-entry ?/ ". 124b" actionscript-mode-syntax-table)
(modify-syntax-entry ?* ". 23" actionscript-mode-syntax-table)
(modify-syntax-entry ?\n "> b" actionscript-mode-syntax-table)
(modify-syntax-entry ?\^m "> b" actionscript-mode-syntax-table))
(defvar actionscript-mode-map ()
"Keymap used in actionscript-mode buffers.")
(if actionscript-mode-map
()
(setq actionscript-mode-map (make-sparse-keymap))
(define-key actionscript-mode-map [(control meta a)] 'as-beginning-of-defun)
(define-key actionscript-mode-map [(control meta e)] 'as-end-of-defun)
(define-key actionscript-mode-map [(control meta h)] 'as-mark-defun)
(define-key actionscript-mode-map "\C-c\C-c" 'comment-region)
(define-key actionscript-mode-map "\C-c\C-u" 'uncomment-region))
(defcustom actionscript-mode-hook nil
"*Hook called by `actionscript-mode'."
:type 'hook
:group 'actionscript)
;; Preprocessor directives (for cpp, not Actionscript).
(defconst preprocessor-kwds
'("#include" "#define" "#else" "#endif" "#ifdef" "#ifndef"))
;; Constants
(defconst actionscript-constant-kwds
'("true" "false" "null" "undefined" "NaN" "Infinity" "-Infinity"))
;; Global funcs
(defconst actionscript-global-funcs
'("Array" "Boolean" "decodeURI" "decodeURIComponent" "encodeURI"
"encodeURIComponent" "escape" "int" "isFinite" "isNaN" "isXMLName"
"Number" "Object" "parseFloat" "parseInt" "String" "trace" "uint"
"unescape" "XML" "XMLList"))
;; Top Level Classes
(defconst actionscript-global-classes
'("ArgumentError" "arguments" "Array" "Boolean" "Class" "Date"
"DefinitionError" "Error" "EvalError" "Function" "int" "Math"
"Namespace" "Number" "Object" "QName" "RangeError" "ReferenceError"
"RegExp" "SecurityError" "String" "SyntaxError" "TypeError" "uint"
"URIError" "VerifyError" "XML" "XMLList"))
;; Global props
(defconst actionscript-global-props
'("this"))
;; Operators
(defconst actionscript-symbol-operators
'("+" "+=" "[]" "=" "&" "&=" "<<" "<<="
"~" "|" "|=" ">>" ">>=" ">>>" ">>>="
"^" "^=" "/*" "*/" "," "?:" "--" "/"
"/=" "." "==" ">" ">=" "++" "!=" "<>"
"<" "<=" "//" "&&" "!" "||" "%" "%="
"*" "*=" "{}" "()" "===" "!==" "\""
"-" "-=" ":"))
(defconst actionscript-word-operators
'("as" "is" "instanceof" "new" "typeof" "void"))
;; Declaration specifier keywords.
(defconst actionscript-specifier-kwds
'("override" "instrinsic" "private" "protected" "public" "static" "dynamic"))
;; Class/struct declaration keywords.
(defconst actionscript-class-kwds
'("class" "interface"))
(defconst actionscript-package-kwds
'("package"))
;; Keywords introducing other declaration-level constructs.
(defconst actionscript-other-decl-kwds
'("import"))
;; Variable and Function declarations
(defconst actionscript-other-decl-2-kwds
'("var" "function" "const"))
;; Keywords that occur in declaration-level constructs.
(defconst actionscript-decl-level-kwds
'("extends" "implements"))
;; Conditionals
;; *Not sure if "catch" should be here,
;; but it's assumed to be here for the actionscript-conditional-key.
(defconst actionscript-conditional-kwds
'("for" "for each" "if" "while" "switch" "catch"))
;; Statement keywords followed directly by a block.
(defconst actionscript-block-stmt-1-kwds
'("do" "else" "finally" "try"))
;; Statement keywords followed by an expression or nothing.
(defconst actionscript-simple-stmt-kwds
'("break" "continue" "return" "throw"))
;; Keywords introducing labels in blocks.
(defconst actionscript-label-kwds
'("case" "default"))
;; Keywords that can occur anywhere in expressions.
(defconst actionscript-expr-kwds
'("super"))
;; Other keywords that we haven't grouped properly.
(defconst actionscript-other-kwds
'("delete" "get" "set" "with"))
;; (defconst actionscript-builtin-props
;; (regexp-opt
;; (append actionscript-object-props
;; actionscript-function-props
;; actionscript-array-props
;; actionscript-string-props
;; actionscript-date-props
;; actionscript-number-props
;; actionscript-math-props)))
;; (defconst actionscript-builtin-funcs(regexp-opt
;; (append actionscript-global-funcs
;; actionscript-math-funcs
;; actionscript-array-funcs
;; actionscript-date-funcs)))
(defconst actionscript-keywords
(regexp-opt
(append actionscript-constant-kwds
actionscript-global-funcs
actionscript-global-classes
actionscript-global-props
actionscript-specifier-kwds
actionscript-class-kwds
actionscript-package-kwds
actionscript-other-decl-kwds
actionscript-other-decl-2-kwds
actionscript-decl-level-kwds
actionscript-conditional-kwds
actionscript-block-stmt-1-kwds
actionscript-simple-stmt-kwds
actionscript-label-kwds
actionscript-expr-kwds
actionscript-other-kwds) 'words))
(defconst actionscript-identifier-re "[a-zA-Z_$][a-zA-Z0-9_$]*"
"Regexp to match any valid identifier in actionscript.")
(defcustom actionscript-mode-hook nil
"*Hook called by `actionscript-mode'."
:type 'hook
:group 'actionscript)
(defcustom actionscript-font-lock-level 2
"*What level of syntax highlighting do we want. 1-3"
:type '(radio (const :tag "Only keywords." 1)
(const :tag "Keywords and contextual tags." 2)
(const :tag "All of the above plus all of Actionscript's builtin classes. (not implemented)" 3))
:group 'actionscript)
;;;; Faces -------------------------------------------------------------------
(defvar actionscript-font-lock-default-face 'actionscript-font-lock-default-face)
(let ((red "#a35757")
(green "#7ac470")
(yellow "#dfe14e")
(orange "#ef6d22")
(blue "#5083b2")
(magenta "#b781ac")
(cyan "#b0b5d2")
(white "#f0f0f0"))
(defface actionscript-preprocessor-kwds-face
`((t (:foreground ,yellow)))
"*Face for preprocesor directives."
:group 'actionscript-faces :group 'faces)
(defface actionscript-constant-kwds-face
`((t (:foreground ,cyan)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-global-funcs-face
`((t (:foreground ,red)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-global-classes
`((t (:foreground ,blue)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-global-props-face
`((t (:foreground ,blue)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-operators-face
`((t (:foreground ,yellow)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-specifier-kwds-face
`((t (:foreground ,magenta)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-package-kwds-face
`((t (:foreground ,yellow)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-class-kwds-face
`((t (:foreground ,yellow)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-other-decl-kwds-face
`((t (:foreground ,yellow)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-other-decl-2-kwds-face
`((t (:foreground ,blue)))
"* function, var"
:group 'actionscript-faces :group 'faces)
(defface actionscript-decl-level-kwds-face
`((t (:foreground ,yellow)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-conditional-kwds-face
`((t (:foreground ,yellow)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-block-stmt-1-kwds-face
`((t (:foreground ,yellow)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-simple-stmt-kwds-face
`((t (:foreground ,yellow)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-label-kwds-face
`((t (:foreground ,yellow)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-expr-kwds-face
`((t (:foreground ,red)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-other-kwds-face
`((t (:foreground ,red)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-package-name-face
`((t (:foreground ,green)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-class-name-face
`((t (:foreground ,cyan)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-function-name-face
`((t (:foreground ,green)))
"*"
:group 'actionscript-faces :group 'faces)
(defface actionscript-variable-name-face
`((t (:foreground ,cyan)))
"*"
:group 'actionscript-faces :group 'faces)
)
(defconst actionscript-font-lock-keywords-1
;; The following only highlight specific words that exist in the language.
(list
`(,(regexp-opt preprocessor-kwds 'words) 0 'actionscript-preprocessor-kwds-face)
`(,(regexp-opt actionscript-constant-kwds 'words) 0 'actionscript-constant-kwds-face)
`(,(regexp-opt actionscript-global-funcs 'words) 0 'actionscript-global-funcs-face)
`(,(regexp-opt actionscript-global-props 'words) 0 'actionscript-global-props-face)
;; `(,(regexp-opt actionscript-symbol-operators) 0 'actionscript-operators-face)
`(,(regexp-opt actionscript-word-operators 'words) 0 'actionscript-operators-face)
`(,(regexp-opt actionscript-specifier-kwds 'words) 0 'actionscript-specifier-kwds-face)
`(,(regexp-opt actionscript-class-kwds 'words) 0 'actionscript-class-kwds-face)
`(,(regexp-opt actionscript-package-kwds 'words) 0 'actionscript-package-kwds-face)
`(,(regexp-opt actionscript-other-decl-kwds 'words) 0 'actionscript-other-decl-kwds-face)
`(,(regexp-opt actionscript-other-decl-2-kwds 'words) 0 'actionscript-other-decl-2-kwds-face)
`(,(regexp-opt actionscript-decl-level-kwds 'words) 0 'actionscript-decl-level-kwds-face)
`(,(regexp-opt actionscript-conditional-kwds 'words) 0 'actionscript-conditional-kwds-face)
`(,(regexp-opt actionscript-block-stmt-1-kwds 'words) 0 'actionscript-block-stmt-1-kwds-face)
`(,(regexp-opt actionscript-simple-stmt-kwds 'words) 0 'actionscript-simple-stmt-kwds-face)
`(,(regexp-opt actionscript-label-kwds 'words) 0 'actionscript-label-kwds-face)
`(,(regexp-opt actionscript-expr-kwds 'words) 0 'actionscript-expr-kwds-face)
`(,(regexp-opt actionscript-other-kwds 'words) 0 'actionscript-other-kwds-face))
"Subdued level highlighting for Actionscript mode.")
(defconst actionscript-font-lock-keywords-2
(append
actionscript-font-lock-keywords-1
;;;; The rules in this section highlight words in the buffer by determining their context.
(list
;; Fontify package names in import directives.
;; TODO: support '*' as the last symbol in the package name.
(list (concat (regexp-opt actionscript-other-decl-kwds 'words) "[ \t]*\\(?:" actionscript-identifier-re "\\.\\)*\\(" actionscript-identifier-re "\\)?")
'(2 'actionscript-class-name-face nil t)
(list (concat "[ \t]*\\(" actionscript-identifier-re "\\)\\.") '(goto-char (match-end 1)) '(goto-char (match-end 0)) '(1 'actionscript-package-name-face nil t)))
;; Fontify package names.
(list (concat (regexp-opt (append actionscript-package-kwds) 'words) "[ \t]*\\(" actionscript-identifier-re "\\)?")
'(2 'actionscript-package-name-face nil t))
;; Fontify class names.
(list (concat (regexp-opt (append actionscript-class-kwds actionscript-decl-level-kwds) 'words) "[ \t]*\\(" actionscript-identifier-re "\\)?")
'(2 'actionscript-class-name-face nil t))
;; Function names.
(list (concat "\\<function\\>[ \t\n]+\\(?:\\(?:get\\|set\\)[ \t\n]+\\)?\\(?:\\(" actionscript-identifier-re "\\)\\)?")
'(1 'actionscript-function-name-face nil t))
;; The 'in' in 'for..in.'
(list (concat "\\<for\\>[ \t\n]*([ \t\n]*\\(?:var[ \t\n]+\\)?" actionscript-identifier-re "[ \t\n]*\\(?::[ \t\n]*\\([a-zA-Z0-9_$*]*\\)\\)?[ \t\n]+\\(in\\)[ \t\n]+")
'(2 'actionscript-other-kwds-face nil t))
;; The 'each' and the 'in' in 'for each..in.'
;; (list (concat "\\<for\\>[ \t\n]+\\(?:\\(each\\)[ \t\n]*\\)([ \t\n]*\\(?:var[ \t\n]+\\)?" actionscript-identifier-re "[ \t\n]*\\(?::[ \t\n]*\\([a-zA-Z0-9_$*]*\\)\\)?[ \t\n]+\\(in\\)[ \t\n]+")
;; '(1 'actionscript-other-kwds-face nil t)
;; '(3 'actionscript-other-kwds-face nil t))
;; Local variables.
(list (concat "\\<var\\>"
"\\([ \t]*"
actionscript-identifier-re
"\\)")
;; Fontify each declaration item.
'(font-lock-match-c-style-declaration-item-and-skip-to-next
;; Start and finish with point after the type specifier.
(goto-char (match-beginning 1))
(goto-char (match-beginning 1))
(1 'actionscript-variable-name-face)))
;; Objects and their functions
;; package(s) class property
;; TODO: We'd like to be able to distinguish true class names from a variable, so
;; that String.parse() would look different than mystring.parse().
;; -We might also want to distinguish local function calls, like parse().
;; (list (concat "\\<" "\\(?:[A-Za-z_]\\sw*\\.\\)*" "\\(?:[A-Za-z_]\\sw*\\.\\)" "\\([A-Za-z_]*\\)")
;; '(1 'actionscript-function-name-face))
))
"Medium level highlighting for Actionscript mode.")
(defconst actionscript-font-lock-keywords-3
(append
actionscript-font-lock-keywords-2
(list
;; TODO: Add all the builtin objects in Actionscript.
;; Builtin props.
;; (list actionscript-builtin-props 1 'actionscript-builtin-props-face)
;; Builtin funcs.
;; (list actionscript-builtin-funcs 1 'actionscript-builtin-funcs-face)
))
"Gaudy level highlighting for Actionscript mode.")
;; --------------------------------------------------------------------------------
(defun as-get-function-re(&optional function-name)
"Returns a regular expression that will match the function signature
containing the supplied function-name. If function-name is omitted,
the regexp will match any function."
(unless function-name
(setq function-name actionscript-identifier-re))
(let ((visibility-kwd (regexp-opt '("public" "protected" "private") nil))
(other-kwd (regexp-opt '("final" "static" "override") nil))
(get-set-kwd (regexp-opt '("get" "set") nil)))
(concat "\\(?:^[ \t\n]*\\)" ; Start of line, followed by any amount of whitespace.
"\\(?:\\(" visibility-kwd "\\|" other-kwd "\\)[ \t\n]+\\)?" ; (1) Any of the optional keywords, which must be followed by whitespace.
"\\(?:\\(" visibility-kwd "\\|" other-kwd "\\)[ \t\n]+\\)?" ; (2) A second optional keyword, which must be followed by whitespace.
"\\<function\\>" ; 'function' keyword.
"[ \t\n]+" ; Mandatory whitespace.
"\\(?:" get-set-kwd "[ \t\n]+\\)?" ; Optional get/set keyword, which must be followed by whitespace.
"\\(" function-name "\\)" ; (3) Function name.
"[ \t\n]*" ; Optional whitespace.
"(" ; Opening paren for function parameters.
"[ \t\n]*" ; Optional whitespace.
"\\([\"a-zA-Z\-0-9_$*,:= \t\n]*?\\(?:\\.\\.\\.[a-zA-Z\-0-9_$]+\\)?\\)" ; (4) Function parameters, including any trailing '...args'.
")" ; Closing paren for function parameters.
"[ \t\n]*" ; Optional whitespace.
"\\(?::[ \t\n]*\\(" actionscript-identifier-re "\\|*\\)\\)?" ; (5) Optional return value type specifier.
"[ \t\n]*" ; Optional whitespace.
"{"))) ; Opening brace for function body.
(defconst as-function-re (as-get-function-re)
"A regexp that matches a function signature in Actionscript 3.0.")
(defun as-get-attribute-re(&optional attribute-name)
"Returns a regular expression that will match the class attribute
definition containing the supplied attribute-name. If attribute-name
is omitted, the regexp will match any class attribute."
(unless attribute-name
(setq attribute-name actionscript-identifier-re))
(let ((visibility-kwd (regexp-opt '("public" "protected" "private") nil))
(var-kwd (regexp-opt '("var" "const") nil))
(other-kwd "static"))
(concat "\\(?:^[ \t\n]*\\)" ; Start of line, followed by any amount of whitespace.
"\\(?:\\(" visibility-kwd "\\|" other-kwd "\\)[ \t\n]+\\)?" ; (1) Any of the optional keywords, which must be followed by whitespace.
"\\(?:\\(" visibility-kwd "\\|" other-kwd "\\)[ \t\n]+\\)?" ; (2) A second optional keyword, which must be followed by whitespace.
"\\<\\(" var-kwd "\\)\\>" ; (3) var keyword.
"[ \t\n]+" ; Mandatory whitespace.
"\\(" attribute-name "\\)" ; (4) Attribute name.
"[ \t\n]*" ; Optional whitespace.
"\\(?::[ \t\n]*\\(" actionscript-identifier-re "*\\)\\)?" ; (5) Optional type specifier.
)))
(defconst as-attribute-re (as-get-attribute-re)
"A regexp that matches a class attribute definition in Actionscript 3.0.")
;; Support for imenu
(defvar as-imenu-generic-expression
`((nil ,as-function-re 3)))
(defun as-imenu-init (mode-generic-expression)
(setq imenu-generic-expression mode-generic-expression
imenu-case-fold-search nil))
(defun as-get-beginning-of-defun()
;; Returns the position.
(save-excursion
(when (re-search-backward as-function-re nil t)
(match-beginning 1))))
(defun as-get-end-of-defun()
;; This only works if we are inside a defun.
(save-excursion
(when (re-search-backward as-function-re nil t)
(goto-char (match-end 0))
;; Move back a char, so that point is right on the opening
;; brace.
(backward-char)
(forward-list)
(point))))
(defun as-get-end-of-defun2()
;; This should work if we are not inside any defuns.
(save-excursion
(beginning-of-line) ; hack, in case point is currently inside a function sig.
(when (re-search-forward as-function-re nil t)
(goto-char (match-end 0))
;; Move back a char, so that point is right on
;; the opening brace.
(backward-char)
(forward-list)
(point))))
(defun as-beginning-of-defun()
(interactive)
(let ((pos (as-get-beginning-of-defun)))
(if pos
(goto-char pos)
(message "Can't find any functions."))))
(defun as-inside-defun?()
(let ((cur (point))
(start (as-get-beginning-of-defun))
(end (as-get-end-of-defun)))
(and start
end
(> cur start)
(< cur end))))
(defun as-end-of-defun()
(interactive)
(if (as-inside-defun?)
(goto-char (as-get-end-of-defun))
(let ((pos (as-get-end-of-defun2)))
(if pos
(goto-char pos)
(message "Can't find any functions.")))))
(defun as-mark-defun()
(interactive)
(let ((start (as-get-beginning-of-defun))
(end (as-get-end-of-defun)))
(if (not (or start end))
(message "Can't find any functions.")
(set-mark end)
(goto-char start)
(beginning-of-line))))
;; --------------------------------------------------------------------------------
;; Indentation (by Aemon Cannon: http://github.com/aemoncannon/as3-mode/tree/master/as3-mode.el)
(defun actionscript-indent-line ()
"Indent current line of As3 code."
(interactive)
(indent-line-to (max 0 (as3-calculate-indentation))))
(defun as3-calculate-indentation ()
"Return the column to which the current line should be indented."
(save-excursion
(as3-maybe-skip-leading-close-delim)
(let ((pos (point)))
(beginning-of-line)
(if (not (search-backward-regexp "[^\n\t\r ]" 1 0))
0
(progn
(as3-maybe-skip-leading-close-delim)
(+ (current-indentation) (* standard-indent (as3-count-scope-depth (point) pos))))))))
(defun as3-maybe-skip-leading-close-delim ()
(beginning-of-line)
(forward-to-indentation 0)
(if (looking-at "\\s)")
(forward-char)
(beginning-of-line)))
(defun as3-face-at-point (pos)
"Return face descriptor for char at point."
(plist-get (text-properties-at pos) 'face))
(defun as3-count-scope-depth (rstart rend)
"Return difference between open and close scope delimeters."
;;Attempting Steve Yegge's solution..
;; (save-excursion
;; (let ((result (parse-partial-sexp rstart rend)))
;; (if (or (nth 3 result) (nth 4 result) (nth 7 result))
;; 0
;; (nth 0 result)))))
(save-excursion
(goto-char rstart)
(let ((open-count 0)
(close-count 0)
opoint)
(while (and (< (point) rend)
(progn (setq opoint (point))
(re-search-forward "\\s)\\|\\s(" rend t)))
(if (= opoint (point))
(forward-char 1)
(cond
;; Don't count if in string or comment.
((as3-face-at-point (- (point) 1)))
((looking-back "\\s)")
(incf close-count))
((looking-back "\\s(")
(incf open-count))
)))
(- open-count close-count))))
;; --------------------------------------------------------------------------------
;;;###autoload
(defun actionscript-mode ()
"Major mode for editing Actionscript files.
\\{actionscript-mode-map}"
(interactive)
(kill-all-local-variables)
(use-local-map actionscript-mode-map)
(setq major-mode 'actionscript-mode
mode-name "Actionscript")
;; local-abbrev-table actionscript-mode-abbrev-table)
(make-local-variable 'indent-line-function)
(setq indent-line-function 'actionscript-indent-line)
(set-syntax-table actionscript-mode-syntax-table)
(make-local-variable 'parse-sexp-ignore-comments)
(setq parse-sexp-ignore-comments t)
(make-local-variable 'comment-start)
(setq comment-start "//")
(make-local-variable 'comment-start-skip)
(setq comment-start-skip "\\(//+\\|/\\*+\\)\\s *")
(make-local-variable 'font-lock-defaults)
(setq font-lock-defaults
`((,(cond
((= actionscript-font-lock-level 1) 'actionscript-font-lock-keywords-1)
((= actionscript-font-lock-level 2) 'actionscript-font-lock-keywords-2)
((= actionscript-font-lock-level 3) 'actionscript-font-lock-keywords-3)))
nil nil))
(run-mode-hooks 'actionscript-mode-hook))
;; We need to make an adjustment to hideshow to work properly with AS syntax.
(add-to-list 'hs-special-modes-alist '(actionscript-mode "{" "}" "/[*/]" nil hs-c-like-adjust-block-beginning))
(provide 'actionscript-mode)
;; For testing
(defun reload-actionscript-mode ()
(interactive)
(with-current-buffer (get-buffer "actionscript-mode.el")
(eval-buffer))
(with-current-buffer (get-buffer "test.as")
(actionscript-mode)
(message "actionscript-mode reloaded.")))
(define-key global-map [f5] 'reload-actionscript-mode)

Related

Recompile components in a test-op call in ASDF

I'm trying to find a way to always recompile the components (test-1, test-2, test-3, test-4) every time I call (asdf: test-system: my-system), but I do not know how to do it yet.
(defsystem :my-system/test
:author "noloop"
:description "Test."
:depends-on (:test-lib :my-system)
:components ((:module "test"
:components
((:file "test-1")
(:file "test-2")
(:file "test-3")
(:file "test-4"))))
:perform (test-op (op system)
(symbol-call :test-lib '#:run)))
An imaginary function to show where I want to go:
:perform (test-op (op system)
(progn (recompile-components system)
(symbol-call :test-lib '#:run))))
I solved the question thus:
First I created an asdf.lisp file with a package lib-test-asdf.lisp:
(in-package #:cl-user)
(defpackage #:lib-test-asdf
(:nicknames #:lib-test-asdf)
(:use #:common-lisp
#:asdf)
(:export #:test-file
#:run-lib-test-asdf))
(in-package #:lib-test-asdf)
(defvar *system-test-files* (make-hash-table))
(defclass test-file (asdf:cl-source-file) ())
(defmethod asdf:perform ((op asdf:compile-op) (c test-file))
;; do nothing
)
(defmethod asdf:perform ((op asdf:load-op) (c test-file))
(pushnew c (gethash (asdf:component-system c) *system-test-files*)
:key #'asdf:component-pathname
:test #'equal))
(defun run-lib-test-asdf (system-designator)
"Runs a testing ASDF system."
#+quicklisp (ql:quickload (if (typep system-designator 'asdf:system)
(asdf:component-name system-designator)
system-designator))
#-quicklisp (asdf:load-system system-designator)
(restart-case
(dolist (c (reverse
(gethash (asdf:find-system system-designator) *system-test-files*)))
(restart-case
(asdf:perform 'asdf:load-source-op c)))))
(import 'test-file :asdf)
then I imported the following lib-test-asdf functions into the package.lisp file, where it is my defpackage of lib-test:
(:import-from #:lib-test-asdf
#:test-file
#:run-lib-test-asdf)
I created a new system definition for lib-test-asdf:
(defsystem :lib-test-asdf
:components ((:module "src"
:components
((:file "asdf")))))
With this I can use lib-test like this in my apps:
(defsystem :your-app
;; ...
:in-order-to ((test-op (test-op your-app/test))))
(defsystem :your-app/test
:author "your <your#youremail.com>"
:depends-on (:your-app :lib-test)
:defsystem-depends-on (:lib-test-asdf)
:components ((:module "test"
:components
((:test-file "your-app-test"))))
:perform (test-op :after (op c)
(progn (funcall (intern #.(string :run-lib-test-asdf) :lib-test) c)
(symbol-call :lib-test '#:run))))
To run tests with ASDF:
(asdf:test-system :your-app)
I based on Prove:
https://github.com/fukamachi/prove/blob/master/src/asdf.lisp

How to integrate flowtype with spacemacs

I'm spacemacs fan. I want to use Facebook Flow but I have not idea how to integrate it with spacemacs. I'm using flow with nuclide but I need to relearn everything to be productive. There is this script on flow repository to use it with emacs. I need a guide for how to use it within spacemacs.
Thanks.
I used Bodil's flow flycheck config here: https://github.com/bodil/emacs.d/blob/d28264cf072bb8a62459a48813d0cb30804b4f5b/bodil/bodil-js.el#L121-L154
I made it work with spacemacs's react-mode and default eslint flychecker by adding the following to my dotspacemacs/user-config (https://github.com/saltycrane/.spacemacs.d/blob/9d985ace9251529c2b8d7857e2ec9835b103084c/init.el#L383-L414):
;; Flow (JS) flycheck config (http://flowtype.org)
;; from https://github.com/bodil/emacs.d/blob/master/bodil/bodil-js.el
(require 'f)
(require 'json)
(require 'flycheck)
(defun flycheck-parse-flow (output checker buffer)
(let ((json-array-type 'list))
(let ((o (json-read-from-string output)))
(mapcar #'(lambda (errp)
(let ((err (cadr (assoc 'message errp))))
(flycheck-error-new
:line (cdr (assoc 'line err))
:column (cdr (assoc 'start err))
:level 'error
:message (cdr (assoc 'descr err))
:filename (f-relative
(cdr (assoc 'path err))
(f-dirname (file-truename
(buffer-file-name))))
:buffer buffer
:checker checker)))
(cdr (assoc 'errors o))))))
(flycheck-define-checker javascript-flow
"Javascript type checking using Flow."
:command ("flow" "--json" source-original)
:error-parser flycheck-parse-flow
:modes react-mode
:next-checkers ((error . javascript-eslint))
)
(add-to-list 'flycheck-checkers 'javascript-flow)
Also be sure the Flow command line tool is installed. Install it like this:
npm install -g flow-bin
I think Bodil intended to make the messages short, but I would like to have flycheck display more verbose messages. If anyone knows how to do that, I'd appreciate it.
EDIT 2016-08-12: the original version I posted gave a Symbol's function definition is void: flycheck-define-checker error on initial load. I updated the code above to add a require 'flycheck to get rid of that error.
The answer by saltycrane worked fine for me. Thanks! The solution gives a very short error messages as he points out. I have improved the error messages to be more verbose and look more like the output from flow cli output.
A note to new users who wants to use this script is to make sure you edit it to use the correct mode in flycheck-define-checker at the bottom. I use this in js2-mode, and saltycrane uses react-mode. Edit it to use whatever you are using.
(require 'f)
(require 'json)
(require 'flycheck)
(defun flycheck-parse-flow (output checker buffer)
(let ((json-array-type 'list))
(let ((o (json-read-from-string output)))
(mapcar #'(lambda (errp)
(let ((err (cadr (assoc 'message errp)))
(err2 (cadr (cdr (assoc 'message errp)))))
(flycheck-error-new
:line (cdr (assoc 'line err))
:column (cdr (assoc 'start err))
:level 'error
:message (concat (cdr (assoc 'descr err)) ". " (cdr (assoc 'descr err2)))
:filename (f-relative
(cdr (assoc 'path err))
(f-dirname (file-truename
(buffer-file-name))))
:buffer buffer
:checker checker)))
(cdr (assoc 'errors o))))))
(flycheck-define-checker javascript-flow
"Static type checking using Flow."
:command ("flow" "--json" source-original)
:error-parser flycheck-parse-flow
:modes js2-mode)
(add-to-list 'flycheck-checkers 'javascript-flow)

common lisp: how to suppress newline or "soft return"

This code
(defun arabic_to_roman (filename)
(let ((arab_roman_dp '())
(arab nil)
(roman nil))
(with-open-file (in filename
:direction :input
:if-does-not-exist nil)
(when in
(loop for line = (read-line in nil)
while line do
(setq arab (subseq line 0 (search "=" line)))
(setq roman (subseq line (1+ (search "=" line)) (length line)))
(setf arab_roman_dp (acons arab roman arab_roman_dp))
;(format t "~S ~S~%" arab roman)
)))
(with-open-file (stream #p"ar_out.txt"
:direction :output
:if-exists :overwrite
:if-does-not-exist :create )
(write arab_roman_dp :stream stream :escape nil :readably nil))
'done!))
seems to work well. It takes a file with entries like this
1=I
2=II
...
and builds one large list of dotted pairs. However when I look at the output file, it seems as though soft returns or newlines have been inserted.
((4999 . MMMMCMXCIX) (4998 . MMMMCMXCVIII) (4997 . MMMMCMXCVII)
(4996 . MMMMCMXCVI) (4995 . MMMMCMXCV) (4994 . MMMMCMXCIV)
(4993 . MMMMCMXCIII) (4992 . MMMMCMXCII) (4991 . MMMMCMXCI) (4990 . MMMMCMXC)
...
I was expecting the output to look more like just one continuous line:
((4999 . MMMMCMXCIX) (4998 . MMMMCMXCVIII) (4997 . MMMMCMXCVII) (4996 . MMMCMXCVI) (4995 . MMMMCMXCV) (4994 . MMMMCMXCIV) (4993 . MMMMCMXCIII) (4992 . MMMMCMXCII) (4991 . MMMMCMXCI) (4990 . MMMMCMXC) ...
Is my code the way it is indeed throwing in newlines somehow? I've used the write version of princ which supposedly suppresses newlines. Later I want to read this file back into the program as just one big list, so I don't want newline issues.
It looks like the pretty-printer is being invoked (the default is implementation-dependent), to print it with indentation and human-readable line lengths. Use :pretty nil to disable this.
(write arab_roman_dp :stream stream :escape nil :readably nil :pretty nil)
A better way to write it:
use functions to create blocks of code, which can be combined
less side effects and fewer variables
no need to test for in
easy to understand control flow
Example:
(defun arabic_to_roman (filename)
(flet ((read-it ()
(with-open-file (in filename
:direction :input
:if-does-not-exist nil)
(loop for line = (read-line in nil)
for pos = (position #\= line)
while line collect (cons (subseq line 0 pos)
(subseq line (1+ pos))))))
(write-it (list)
(with-open-file (stream #p"ar_out.txt"
:direction :output
:if-exists :overwrite
:if-does-not-exist :create)
(write list :stream stream :escape nil :readably nil :pretty nil))))
(write-it (read-it))
'done-read-write))

How to interact with a process input/output in SBCL/Common Lisp

I have a text file with one sentence per line. I would like to lemmatize the worlds in each line using hunspell (-s option). Since I want to have the lemmas of each line separately, it wouldn't make sense to submit the whole text file to hunspell. I do need to send one line after another and have the hunspell output for each line.
Following the answers from How to process input and output streams in Steel Bank Common Lisp?, I was able to send the whole text file for hunspell one line after another but I was not able to capture the output of hunspell for each line. How interact with the process sending the line and reading the output before send another line?
My current code to read the whole text file is
(defun parse-spell-sb (file-in)
(with-open-file (in file-in)
(let ((p (sb-ext:run-program "/opt/local/bin/hunspell" (list "-i" "UTF-8" "-s" "-d" "pt_BR")
:input in :output :stream :wait nil)))
(when p
(unwind-protect
(with-open-stream (o (process-output p))
(loop
:for line := (read-line o nil nil)
:while line
:collect line))
(process-close p))))))
Once more, this code give me the output of hunspell for the whole text file. I would like to have the output of hunspell for each input line separately.
Any idea?
I suppose you have a buffering problem with the program you want to run. For example:
(defun program-stream (program &optional args)
(let ((process (sb-ext:run-program program args
:input :stream
:output :stream
:wait nil
:search t)))
(when process
(make-two-way-stream (sb-ext:process-output process)
(sb-ext:process-input process)))))
Now, on my system, this will work with cat:
CL-USER> (defparameter *stream* (program-stream "cat"))
*STREAM*
CL-USER> (format *stream* "foo bar baz~%")
NIL
CL-USER> (finish-output *stream*) ; will hang without this
NIL
CL-USER> (read-line *stream*)
"foo bar baz"
NIL
CL-USER> (close *stream*)
T
Notice the finish-output – without this, the read will hang. (There's also force-output.)
Python in interactive mode will work, too:
CL-USER> (defparameter *stream* (program-stream "python" '("-i")))
*STREAM*
CL-USER> (loop while (read-char-no-hang *stream*)) ; skip startup message
NIL
CL-USER> (format *stream* "1+2~%")
NIL
CL-USER> (finish-output *stream*)
NIL
CL-USER> (read-line *stream*)
"3"
NIL
CL-USER> (close *stream*)
T
But if you try this without the -i option (or similar options like -u), you'll probably be out of luck, because of the buffering going on. For example, on my system, reading from tr will hang:
CL-USER> (defparameter *stream* (program-stream "tr" '("a-z" "A-Z")))
*STREAM*
CL-USER> (format *stream* "foo bar baz~%")
NIL
CL-USER> (finish-output *stream*)
NIL
CL-USER> (read-line *stream*) ; hangs
; Evaluation aborted on NIL.
CL-USER> (read-char-no-hang *stream*)
NIL
CL-USER> (close *stream*)
T
Since tr doesn't provide a switch to turn off buffering, we'll wrap the call with a pty wrapper (in this case unbuffer from expect):
CL-USER> (defparameter *stream* (program-stream "unbuffer"
'("-p" "tr" "a-z" "A-Z")))
*STREAM*
CL-USER> (format *stream* "foo bar baz~%")
NIL
CL-USER> (finish-output *stream*)
NIL
CL-USER> (read-line *stream*)
"FOO BAR BAZ
"
NIL
CL-USER> (close *stream*)
T
So, long story short: Try using finish-output on the stream before reading. If that doesn't work, check for command line options preventing buffering. If it still doesn't work, you could try wrapping the programm in some kind of pty-wrapper.

Using stdout from shell script in common lisp

I am writing a common lisp program that needs to handle the output of a command. However, when I try to use the results in another function, I only get a NIL as return-value.
Here is the function I use to run commands:
(defun run-command (command &optional arguments)
(with-open-stream (pipe
(ext:run-program command :arguments arguments
:output :stream :wait nil))
(loop
:for line = (read-line pipe nil nil)
:while line :collect line)))
Which, when run by itself gives:
CL-USER> (run-command "ls" '("-l" "/tmp/test"))
("-rw-r--r-- 1 petergil petergil 0 2011-06-23 22:02 /tmp/test")
However, when I run it through a function, only NIL is returned:
(defun sh-ls (filename)
(run-command "ls" '( "-l" filename)))
CL-USER> (sh-ls "/tmp/test")
NIL
How can I use the results in my functions?
Try this:
(defun sh-ls (filename)
(run-command "ls" (list "-l" filename)))
The '("-l" filename) is quoting the list, AND the symbol 'filename', rather than evaluating filename.
You can also use backquote ` before sexpr and , before filename to evaluate it:
(defun sh-ls (filename)
(run-command "ls" `("-l" ,filename)))

Resources