- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
Hunchentoot/cl-who 页面组合
我正在尝试将 hunchentoot 中的几页放在一起作为实验,但我遇到了意想不到的墙。例如,我有以下模板宏。
(defmacro page-template ((&key title) &body body)
`(with-html-output-to-string
(*标准输出* nil :prologue t :indent t)
(:html :xmlns "http://www.w3.org/1999/xhtml":xml\:lang "en":lang "en"
(:head (:meta :http-equiv "Content-Type":content "text/html;charset=utf-8")
(:title ,(format nil "~@[~A - ~]Test Site"title)))
(:body ,@body)))))
现在当我有一个纯文本页面,或者一个充满 html 文字的页面时
(define-easy-handler (test-page :uri "/") ()
(页面模板 (:title "Splash Page") (:p "Testing testing")))
一切都很好。页面输出正确,我可以立即看到我的代码所做的努力。但是,当我有一个由冗余元素组成的页面时,就没有那么简单了。例如,假设我有一个页面,无论出于何种原因,我想在该页面上显示三个 RSS 新闻提要。这是一个足够复杂的组件,我想将其抽象出来,所以在我看来,我应该能够做类似的事情
(define-easy-handler (test-feed :uri "/feeds") ()
(页面模板(:标题“启动页面”)
(发布新闻源“http://nf-one.html”)
(发布新闻源“http://nf-two.html”)
(发布新闻源“http://nf-three.html”)))
(defmacro publish-newsfeed (url &optional (item-limit 5))
(flet ((get-text (s-tree node-path) (car (last (xmls-tools:find-subtree s-tree node-path)))))
(让 ((rss-feed (xmls:parse (drakma:http-request url))))
`(:div:class "rss-feed"
(:a :href ,(get-text rss-feed '("channel""link")) :target "_top"(:h1 ,(get-text rss-feed '("channel""title"))) )
(:ul ,@(mapcar #'(lambda (item)
`(:li (:a :href ,(get-text item '("link")) :target "_top"(:h2 ,(get-text item '("title"))))
(:p :class "date",(get-text item '("pubDate")))
(:p ,(get-text item '("description")))))
(让((项目(xmls-tools:find-all-children(xmls-tools:find-subtree rss-feed '(“ channel ”))“项目”)))
(if (> (length items) item-limit) (subseq items 0 item-limit) items))))))))
但是上面的结果是一个“服务器错误”页面。我不知道为什么; page-template
是一个宏,所以调用 publish-newsfeed
不应该扩展,直到它们处于 with-html-output-to-string
的上下文中.谁能告诉我我做错了什么?
此外,仔细检查各种 Hunchentoot/cl-who 教程,它们似乎都没有进行这种页面组合。任何有一些 Hunchentoot 经验的人都可以告诉我将页面分解为组件的正确/规范方法是什么?
编辑:
下面是Ramarren的正确回答; with-html-output
宏在不同的评估规则下工作。在这种情况下实际工作的发布新闻源版本实际上是
(defun publish-newsfeed (url &optional (item-limit 5))
(flet ((get-text (s-tree node-path) (car (last (xmls-tools:find-subtree s-tree node-path)))))
(让* ((rss-feed (xmls:parse (drakma:http-request url))))
(项目 (xmls-tools:find-all-children (xmls-tools:find-subtree rss-feed '("channel")) "item"))
(ltd-items (if (> (length items) item-limit) (subseq items 0 item-limit) items)))
(带-html-输出
(*标准输出* nil :indent t)
(:div:class "rss-feed"
(:a :href (get-text rss-feed '("channel""link")) :target "_top"(:h1 (str (get-text rss-feed '("channel""title"))) ))
(:ul (dolist (item ltd-items)
(htm (:li (:h2 (:a :href (get-text item '("link")) :target "_top"(str (get-text item '("title")))))
(:p :class "date"(str (get-text item '("pubDate"))))
(:p (str (get-text item '("description")))))))))))))
注意删除 mapcar
为 dolist
(我是一名 Schemer,不要因为喜欢 lambdas 而给我太多困难,但它们在这里不是正确的选择),以及 htm
的使用转义 html s-exps (h-exps?) 块,否则这些块不会出现在 with-html-output
的上下文中.最后,我不得不换行但不是 :href
在 (str )
中的属性让它们动态扩展。
最佳答案
宏with-html-output-to-string
使用 special evaluation rules 扩展它的 body .特别是,任何未识别的表单都保持原样,这意味着在生成 html 生成代码之前不会扩展宏,这意味着在您的 publish-newsfeed
之前。宏被标准编译器扩展,它不再在 with-html-output-to-string
的上下文中.这在扩展宏 manually 时很明显,尤其是使用 Slime 宏扩展功能。
为了让它工作,你应该做 publish-newsfeed
一个函数和使用 with-html-output
在它内部使用相同的流(要么假设到处都是标准输出,要么显式地传递流)。
关于common-lisp - Hunchentoot/cl-who 页面组成,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3794223/
最近,我开始学习 cuis-smalltalk,我没有意识到与 CLOS 相比,Smalltalk 的 OOP 有多么深刻和深入(我使用的是 Ruby)。我了解到 Smalltalk 是一个自己实现的
Maven存储库包含以下两个依赖项:org.apache.commons:commons-io:1.3.2和commons-io:commons-io:1.3.2。有什么区别,我应该在pom.xml中
我刚刚在我的 pom 文件中看到 Apache commons-collections 有两个不同的组 ID: commons-collections commons-collect
Windows 上的 Common Lisp 中是否有用于串行端口通信的库? 最佳答案 下面是一些使用 SBCL 外部函数 POSIX 调用实现串行通信的函数。它不如完整的库好,但我解决了根据此协议(
SBCL 64位,1.1.7 如果我想创建一个包并使用package:CL中的一些符号,我将创建一个像这样的包: (defpackage :foo (:import-from :cl
我正在忙着学习Common Lisp,并且正在寻找一种静态代码分析工具,该工具将帮助我开发更好的样式并避免陷入常见的陷阱。 我找到了Lisp Critic,看起来不错,但我希望有人可以推荐其他一些工具
我正在阅读《Practical Common Lisp》一书,在第 22 章第 284 页的脚注 5 中,我看到一段让我感到困惑的代码片段。 我知道变量list和tail有一个共同的列表结构,但我很困
我正在阅读 Practical Common Lisp ,并且对 Lisp 的 COPY-TREE 函数有疑问。 书中给出了调用的例子 (copy-tree '( '(1 2) '(3 4) '(5
我正在尝试使用 user guide 中的抓取示例运行 geb用于引入依赖项: $ cat my.groovy @Grapes([ @Grab("org.gebish:geb-core:0.9
这里一定有更好的方法,对吧? (format t "Enter your age: ~%") (defun age-case (age) (case age (1 (format t "Y
如何在 do 循环中绑定(bind)从函数返回的多个值? 以下显然是非常错误的,但是这样的事情可能吗? (do (((x y z) (3-val-fn) (3-val-fn))) ((equa
所以我正在学习 Lisp 做分数,这很棒。但是为什么这个相等性检查返回 NIL: * (= 0.2 1/5) NIL ...如果转换为 float 则返回 True第一的: * (=
是否可以“统计”一个文件并找到它的文件类型 - 常规或目录? 最佳答案 阅读关于 portable pathname library 的章节来自 Peter Seibel 的 Practical Co
我是 CL 的新手,正在使用 AllegroCL。我试图弄清楚如何组织我的源代码以满足以下要求: 我想阻止 src 代码包含我的测试套件。 我想以可移植的方式声明项目依赖项(src 和 test de
谁能告诉我最新的标准化 Common Lisp 的文档是什么(应该遵循各种实现的文档)?我问是因为我可以在网上找到很多关于 CL 的书都来自 90 年代,所以我想知道它们是否是最新的。我也来自于在 R
假设我必须定义一个名为foo 的函数。假设,为了定义它,我使用了一些辅助函数 foo1, foo2, foo3, ... 当我加载包含这些函数的文件时,我可以从顶层使用所有这些函数。相反,我只想从顶层
这拒绝编译。注释掉 (setf roll行让它编译。然而,(setf roll...本身在 REPL 中正确评估。 程序: ;; loop n times ; sum up number of hit
我目前正在学习 Common Lisp,并尝试将一些 JSON 发送到网络服务。我要发送的 JSON 以类似于以下的结构表示: ((:ITEMS ((:KEY . "value1") (:IGNO
我有一个带波浪号的目录名(作为字符串):~/projects . 我想得到它的完整路径:/home/user/projects .我怎么做 ? 目标是将它传递给 uiop:run-program ,这
我想从输入文件中读取一个字符串(用户可能修改也可能没有修改)。我想将此字符串视为使用固定数量的参数调用的格式指令。但是,我知道某些格式指令(特别是我想到的 ~/)可能会用于注入(inject)函数调用
我是一名优秀的程序员,十分优秀!