@rip-lang/print 0.1.1 → 0.1.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (3) hide show
  1. package/hljs-rip.js +159 -0
  2. package/package.json +2 -1
  3. package/print.rip +13 -7
package/hljs-rip.js ADDED
@@ -0,0 +1,159 @@
1
+ // highlight.js language definition for Rip
2
+ // Derived from the Monarch grammar in docs/index.html
3
+
4
+ export default function(hljs) {
5
+ const KEYWORDS = [
6
+ 'if', 'else', 'unless', 'then', 'switch', 'when', 'for', 'while', 'until',
7
+ 'loop', 'do', 'return', 'break', 'continue', 'throw', 'try', 'catch', 'finally',
8
+ 'yield', 'await', 'import', 'export', 'from', 'default', 'delete', 'typeof',
9
+ 'instanceof', 'new', 'super', 'debugger', 'use', 'own', 'extends', 'in', 'of',
10
+ 'by', 'as', 'class', 'def', 'enum', 'interface', 'component', 'render',
11
+ 'and', 'or', 'not', 'is', 'isnt',
12
+ ];
13
+
14
+ const LITERALS = [
15
+ 'true', 'false', 'yes', 'no', 'on', 'off',
16
+ 'null', 'undefined', 'NaN', 'Infinity',
17
+ ];
18
+
19
+ const BUILT_INS = [
20
+ 'console', 'process', 'require', 'module', 'exports',
21
+ 'setTimeout', 'setInterval', 'clearTimeout', 'clearInterval',
22
+ 'Promise', 'Array', 'Object', 'String', 'Number', 'Boolean',
23
+ 'Math', 'Date', 'RegExp', 'Error', 'JSON', 'Map', 'Set',
24
+ 'Symbol', 'Buffer', 'Bun',
25
+ ];
26
+
27
+ const INTERPOLATION = {
28
+ className: 'subst',
29
+ begin: /#\{/, end: /\}/,
30
+ keywords: { keyword: KEYWORDS, literal: LITERALS },
31
+ };
32
+
33
+ const STRING_DOUBLE = {
34
+ className: 'string',
35
+ begin: '"', end: '"',
36
+ contains: [hljs.BACKSLASH_ESCAPE, INTERPOLATION],
37
+ };
38
+
39
+ const STRING_SINGLE = {
40
+ className: 'string',
41
+ begin: "'", end: "'",
42
+ contains: [hljs.BACKSLASH_ESCAPE],
43
+ };
44
+
45
+ const HEREDOC_DOUBLE = {
46
+ className: 'string',
47
+ begin: '"""', end: '"""',
48
+ contains: [hljs.BACKSLASH_ESCAPE, INTERPOLATION],
49
+ };
50
+
51
+ const HEREDOC_SINGLE = {
52
+ className: 'string',
53
+ begin: "'''", end: "'''",
54
+ contains: [hljs.BACKSLASH_ESCAPE],
55
+ };
56
+
57
+ const HEREGEX = {
58
+ className: 'regexp',
59
+ begin: '///', end: '///[gimsuy]*',
60
+ contains: [INTERPOLATION, hljs.HASH_COMMENT_MODE],
61
+ };
62
+
63
+ const REGEX = {
64
+ className: 'regexp',
65
+ begin: /\/(?![/*])(?:[^\/\\]|\\.)*\/[gimsuy]*/,
66
+ relevance: 0,
67
+ };
68
+
69
+ const BLOCK_COMMENT = {
70
+ className: 'comment',
71
+ begin: '###', end: '###',
72
+ contains: [hljs.PHRASAL_WORDS_MODE],
73
+ };
74
+
75
+ const LINE_COMMENT = hljs.COMMENT('#', '$');
76
+
77
+ const NUMBER = {
78
+ className: 'number',
79
+ variants: [
80
+ { begin: /0x[0-9a-fA-F](?:_?[0-9a-fA-F])*n?/ },
81
+ { begin: /0o[0-7](?:_?[0-7])*n?/ },
82
+ { begin: /0b[01](?:_?[01])*n?/ },
83
+ { begin: /\d[\d_]*(?:\.[\d][\d_]*)?(?:[eE][+-]?\d+)?n?/ },
84
+ ],
85
+ relevance: 0,
86
+ };
87
+
88
+ const INSTANCE_VAR = {
89
+ className: 'variable',
90
+ begin: /@[a-zA-Z_$][\w$]*/,
91
+ };
92
+
93
+ const CLASS_NAME = {
94
+ className: 'title.class',
95
+ begin: /[A-Z][\w]*/,
96
+ };
97
+
98
+ const FUNCTION_DEF = {
99
+ className: 'function',
100
+ begin: /\bdef\s+/,
101
+ end: /[(\s]/,
102
+ excludeEnd: true,
103
+ keywords: { keyword: 'def' },
104
+ contains: [
105
+ { className: 'title.function', begin: /[a-zA-Z_$][\w$]*[!?]?/ },
106
+ ],
107
+ };
108
+
109
+ const CLASS_DEF = {
110
+ className: 'class',
111
+ beginKeywords: 'class',
112
+ end: /$/,
113
+ contains: [
114
+ { className: 'title.class', begin: /[A-Z][\w]*/ },
115
+ { beginKeywords: 'extends', contains: [CLASS_NAME] },
116
+ ],
117
+ };
118
+
119
+ const OPERATORS = {
120
+ className: 'operator',
121
+ begin: /::=|::|:=|~=|~>|<=>|=!|!\?|=~|\?\?|\?\.|\.\.\.|\.\.|=>|->|\*\*|\/\/|%%|===|!==|==|!=|<=|>=|&&|\|\||[+\-*\/%&|^~<>=!?]/,
122
+ relevance: 0,
123
+ };
124
+
125
+ const TYPE_KEYWORDS = {
126
+ className: 'type',
127
+ begin: /\b(?:number|string|boolean|void|any|never|unknown|object|symbol|bigint)\b/,
128
+ };
129
+
130
+ return {
131
+ name: 'Rip',
132
+ aliases: ['rip'],
133
+ keywords: {
134
+ keyword: KEYWORDS,
135
+ literal: LITERALS,
136
+ built_in: BUILT_INS,
137
+ },
138
+ contains: [
139
+ BLOCK_COMMENT,
140
+ LINE_COMMENT,
141
+ HEREDOC_DOUBLE,
142
+ HEREDOC_SINGLE,
143
+ STRING_DOUBLE,
144
+ STRING_SINGLE,
145
+ HEREGEX,
146
+ REGEX,
147
+ FUNCTION_DEF,
148
+ CLASS_DEF,
149
+ NUMBER,
150
+ INSTANCE_VAR,
151
+ TYPE_KEYWORDS,
152
+ OPERATORS,
153
+ { // inline JS
154
+ className: 'string',
155
+ begin: /`[^`]*`/,
156
+ },
157
+ ],
158
+ };
159
+ }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@rip-lang/print",
3
- "version": "0.1.1",
3
+ "version": "0.1.3",
4
4
  "description": "Syntax-highlighted source code printer — Shiki-powered, serves once, auto-opens browser",
5
5
  "type": "module",
6
6
  "main": "print.rip",
@@ -34,6 +34,7 @@
34
34
  },
35
35
  "files": [
36
36
  "print.rip",
37
+ "hljs-rip.js",
37
38
  "bin/rip-print",
38
39
  "README.md"
39
40
  ]
package/print.rip CHANGED
@@ -9,10 +9,14 @@
9
9
  # ============================================================================
10
10
 
11
11
  import hljs from 'highlight.js'
12
+ import ripLanguage from './hljs-rip.js'
12
13
  import { existsSync, readdirSync, readFileSync, statSync } from 'fs'
13
14
  import { basename, extname, join, relative } from 'path'
14
15
  import { execSync } from 'child_process'
15
16
 
17
+ # Register Rip language with highlight.js
18
+ hljs.registerLanguage 'rip', ripLanguage
19
+
16
20
  # ============================================================================
17
21
  # CLI argument parsing
18
22
  # ============================================================================
@@ -117,7 +121,7 @@ console.log "#{allFiles.length} file#{if allFiles.length > 1 then 's' else ''}"
117
121
  # ============================================================================
118
122
 
119
123
  extToLang =
120
- rip: 'coffeescript'
124
+ rip: 'rip'
121
125
  coffee: 'coffeescript'
122
126
  js: 'javascript'
123
127
  mjs: 'javascript'
@@ -216,10 +220,12 @@ for file in allFiles
216
220
  code = stripTopComments(code) if bypass
217
221
  lang = getLang file
218
222
  lineCount = code.split('\n').length
223
+ mtime = statSync(file).mtime
224
+ timestamp = mtime.toLocaleDateString('en-US', { year: 'numeric', month: 'short', day: 'numeric' }) + ' at ' + mtime.toLocaleTimeString('en-US', { hour: 'numeric', minute: '2-digit' }).toLowerCase()
219
225
  console.log " #{file} (#{lineCount} lines) [#{lang}]"
220
226
  highlighted = highlightCode code, lang
221
227
  html = "<pre><code class=\"hljs\">#{highlighted}</code></pre>"
222
- sections.push { file, lineCount, lang, html }
228
+ sections.push { file, lineCount, lang, timestamp, html }
223
229
 
224
230
  # ============================================================================
225
231
  # HTML template
@@ -258,7 +264,7 @@ for section, i in sections
258
264
  fileSections += """
259
265
  <div class="file-section">
260
266
  <div class="file-header" id="#{section.file}">
261
- <span>#{section.file} <span class="meta">(#{section.lineCount} lines) [#{section.lang}]</span></span>
267
+ <span>#{section.file} <span class="meta">(#{section.lineCount} lines) [#{section.lang}] on #{section.timestamp}</span></span>
262
268
  <span class="nav">#{nav}</span>
263
269
  </div>
264
270
  <div class="code-container">
@@ -296,8 +302,8 @@ html = """
296
302
 
297
303
  .file-section { margin-bottom: 0; }
298
304
  .file-header {
299
- background: #{headerBg}; padding: 8px 16px; font-size: 13px; font-weight: 600;
300
- border-top: 1px solid #{borderColor}; border-bottom: 1px solid #{borderColor};
305
+ background: #{headerBg}; padding: 14px 16px 14px 5.2em; font-size: 13px; font-weight: 600;
306
+ border-top: 1px solid #{borderColor}; border-bottom: 1px solid #{borderColor}; margin-top: -1px;
301
307
  display: flex; justify-content: space-between; align-items: center;
302
308
  }
303
309
  .nav { font-weight: normal; font-size: 12px; }
@@ -306,8 +312,8 @@ html = """
306
312
 
307
313
  .code-container { overflow-x: auto; }
308
314
  .code-container pre { margin: 0; border-radius: 0; }
309
- .code-container code { font-size: 13px; line-height: 1.5; padding: 16px !important; display: block; }
310
- .line-num { color: #aaa; user-select: none; display: inline-block; min-width: 2em; text-align: right; }
315
+ .code-container code { font-size: 13px; line-height: 1.5; padding: 0 !important; display: block; }
316
+ .line-num { color: #aaa; background: #{if dark then '#161b22' else '#f4f4f4'}; user-select: none; display: inline-block; min-width: 2em; text-align: right; padding: 0 0.7em; margin-right: 0.7em; border-right: 1px solid #{borderColor}; }
311
317
 
312
318
  @media print {
313
319
  .toc { page-break-after: always; }