(defmethod carpet-append ((this carpet) (rect image-rectangle))
(destructuring-bind (rect-width . rect-height)
(rectangle-size rect)
(destructuring-bind (bitmap-width . bitmap-height)
(carpet-size this)
(if this
(iter:iter
(iter:with min-area = (* (+ bitmap-width rect-width)
(+ bitmap-height rect-height)))
(iter:with min-pos = nil)
(iter:for pos in (awailable-positions this))
(iter:for test-area = (try-fit rect pos (carpet-bitmap this)))
(when (and test-area (< test-area min-area))
(setf min-pos pos))
(iter:finally
(let ((new-carpet
(make-carpet
:bitmap (make-array
(list (+ (car min-pos) rect-width)
(+ (cdr min-pos) rect-height))
:element-type 'bit)
:rectangles (cons rect (carpet-rectangles this)))))
(copy-bitmap-state this new-carpet)
(setf (rectangle-position rect) min-pos)
(place-image new-carpet rect)
(return new-carpet))))
(make-carpet
:bitmap (make-array
(list rect-width rect-height)
:element-type 'bit)
:rectangles (list rect))))))
image-rectangle
和carpet
是结构。
这样调用此方法时:
(carpet-append
nil
#s(image-rectangle
:position (0 . 0)
:size (48 . 76)
:file "/home/wvxvw/projects/spritesheet/test-images/test-0.png"))
我得到了:
#<SIMPLE-ERROR "~@<There is no applicable method for the generic function ~2I~_~S~
~I~_when called with arguments ~2I~_~S.~:>"
这是不是意味着这样?也许有一些方法可以让它接受nil
作为论据?我如何指定只有nil
和carpet
类型的参数适用?
答案 0 :(得分:8)
如果你有一个包含类carpet
和image-rectangle
的arglist,那么参数最好是这些类或它们的子类。当您的参数声明为类NIL
时,您无法传递carpet
。
因此(if this
没有意义。如果您传递地毯对象,并且无法传递任何其他内容,则测试this
将始终为真。
如果要为NIL
对象和矩形编写方法,则可以使用类NULL。
(defmethod carpet-append ((this null) (rect image-rectangle))
...)
由于CLOS没有像OR或AND这样的类组合器,你必须为每种情况编写一个方法。