Pull to refresh

Почему бы не сделать свой Lisp для веба?

Reading time3 min
Views11K

Вот есть JavaScript — прекрасная вещь. И прекрасная она на мой взгляд по большей части потому, что дебаггер и отладочные инструменты встроены в каждый Браузер. Без дебаггера и инспектора DOM-дерева было гораздо тяжелее что-либо разрабатывать на JavaScript.


Вот я и решил запилить свой Lisp, но с хорошим дебаггером. Взял маленький, простенький littlelisp и отрефакторил его на 99%. Заточил под пошаговое выполнение и создал свою IDE, простую но работающую.


В итоге получилось следующее.


image

.
Сам язык получился смесью Lisp и JavaScript-a потому, что Lisp я знал плохо, а JavaScript хорошо:)


Далее примеры кода.


(alert document.location.href)

(setq x "ok!") 
(alert (x.substring 0 2))

;; пример с if

(setq x (prompt "Value of X:"))
(if (x == null)
    (setq x "(none)")
    (alert (+ "x = " x))
:else
    (alert (+ "x = " x))
)

;; кстати тела while/if не обязательно оборачивать в свой список

(setq i 0)
(while (i < 10)
    (console.log i)
    (++ i)
)

;; есть синтаксический сахар для арифметических и логических операторов

(x < 2)                ;; -> (< x 2)
((x < 2) && (x > -1))  ;; -> (&& (< x 2) (> x -1))
((x < 2) || (x > -1))  ;; -> (|| (< x 2) (> x -1))
(x + 1 2 3)            ;; -> (+ x 1 2 3) => x+1+2+3
(x ++)                 ;; -> (++ x) 
(++ x y z)             ;; -> увеличит на 1 переменные x, y, z последовательно

;; создание и заполнение JS-объекта

(setq obj1 (new Object))
(setq obj1.a 123 obj1.b "abc") ;; можно за один вызов

(setq name "c")
(setq obj1.@name ())           ;; @ - разыменовывает
(setq obj1.@name.0 111)
(setq obj1.@name.1 222)
(setq obj1.@name.2 333) 

(alert (obj1.c.join " "))      ;; покажет "111 222 333"

;; классы пока-что сделаны по простому

(defclass Трамвай
    :extends Object
    :instvars "из_чего номер"
    :classvars "всего"
    :constructor init
)

(setq Трамвай.всего 0)

;; тела методов и функций тоже не обязательно оборачивать в свой список
(defmeth Трамвай.init (номер основа)
    (setq this.номер номер)
    (setq this.из_чего основа)
    (Трамвай.всего ++)
)

(defmeth Трамвай.поехали ()
    (window.alert (+ " Трамвай №" this.номер " из " this.из_чего " за вами выехал!"))
)

(setq Трамвай1 (new Трамвай 9 "буханки хлеба"))
(Трамвай1.поехали)
(alert (+ "Трамваев всего: " Трамвай.всего))

;; лямбды и функции тоже есть конечно-же

((lambda (x y) (alert (x + y)) "a" "b")

(defun f1 (x) (alert (x + " - ok!")))
(f1 "f1")

;; выполнение произвольного кода пока-что через создание и вызов JS-функции

((new Function "msg" "alert(msg);") document.title)

;; эксепшены тоже можно кидать и отлавливать

(catch (a.b)) ;; вернёт Exception object (Error: "a" is undefined!)
(throw (new Error "Exception!"))

Собственно, сам проект на гитхабе здесь: https://github.com/SaemonZixel/littlelisp.js


Напоследок скажу, что код пока-что в бете (на 17 мая 2020г). Ошибок пока что много и я их потихоньку нахожу и правлю :(


Но зато дебаггер почти как в Smalltalk-средах разработки получилось сделать! :)


Можно доводить код прям в деббагере. Выделить любой фрагмент кода и просмотреть результат или запустить вложенный деббагер.


(Добавлено 30 мая 2020г)
Так же я благодарен изначальному автору Mary Rose Cook за столь замечательную реализацию Lisp-a на JavaScript. Она компактна, мала и очень проста.
Страница на гитхабе: https://github.com/maryrosecook/littlelisp


Но к сожалению от этой красоты ничего не осталось. Я переписал всё под корень :(


(Добавлено 10 июля 2020г)
В версии 2.0beta4 я добавил реализацию макросов. Из-за особенностей JavaScript она пока-что в экспериментальном состоянии.

Tags:
Hubs:
Total votes 33: ↑28 and ↓5+36
Comments71

Articles