#lang racket (provide ~sort ~sort-seq ~key) ;; Note: when using nested ~sort, the inner sort is not performed during the ;; first pass for the outer ~sort. Once the values for the outer ~sort have been ;; gathered and sorted, then the innder ~sort is applied. This means that the ;; comparison operator for the outer ~sort should work with unsorted ;; sub-sequences.s (require syntax/parse "aliases.rkt" syntax/stx racket/stxparam (for-syntax racket/syntax)) (define-for-syntax sort-scope (make-syntax-introducer)) (define-syntax-parameter current-key-id #f) (define-for-syntax (~sort-ish op*) (pattern-expander (λ (stx) (syntax-case stx (…) [(self pat …) (if (syntax-parameter-value #'current-key-id) #`(#,@op* _ …) #`(~and (#,@op* tmp …) (~parse (pat …) (sort/stx self #'(tmp …) pat))))])))) (define-syntax ~sort (~sort-ish #'{})) (define-syntax ~sort-seq (~sort-ish #'{~seq})) (define-syntax (sort/stx stx) (syntax-case stx () [(_ ctx stxlist pat) #'(syntax-parameterize ([current-key-id (generate-temporary #'key)]) (def-cls tmpcls pat) (and (syntax-parse stxlist [({~var || tmpcls} …) #t] [_ (displayln (format "Failed to parse ~a as ~a." stxlist 'pat)) #f]) (sort (syntax->list stxlist) (λ (a b) (cond [(and (symbol? a) (symbol? b)) (symbol