-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathfendo.links.fs
523 lines (439 loc) · 17 KB
/
fendo.links.fs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
.( fendo.links.fs ) cr
\ This file is part of Fendo
\ (http://programandala.net/en.program.fendo.html).
\ This file provides the words needed to create links,
\ by the markup words or by the user application.
\ Last modified 202011160218.
\ See change log at the end of the file.
\ Copyright (C) 2013,2014,2017,2018,2019,2020 Marcos Cruz (programandala.net)
\ Fendo is free software; you can redistribute
\ it and/or modify it under the terms of the GNU General
\ Public License as published by the Free Software
\ Foundation; either version 2 of the License, or (at your
\ option) any later version.
\ Fendo is distributed in the hope that it will be useful,
\ but WITHOUT ANY WARRANTY; without even the implied
\ warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
\ PURPOSE. See the GNU General Public License for more
\ details.
\ You should have received a copy of the GNU General Public
\ License along with this program; if not, see
\ <http://gnu.org/licenses>.
\ ==============================================================
\ Requirements {{{1
forth_definitions
require galope/minus-prefix.fs \ `-prefix`
fendo_definitions
require fendo.markup.common.fs
require fendo.markup.html.fs
\ ==============================================================
\ Tools for links {{{1
: file://? ( ca len -- f )
s" file://" string-prefix? ;
\ Does a string start with "file://"?
' file://? alias file_href?
: http://? ( ca len -- f )
s" http://" string-prefix? ;
\ Does a string start with "http://"?
: https://? ( ca len -- f )
s" https://" string-prefix? ;
\ Does a string start with "https://"?
: ftp://? ( ca len -- f )
s" ftp://" string-prefix? ;
\ Does a string start with "ftp://"?
: external_href? ( ca len -- f )
2dup http://? >r 2dup https://? >r ftp://? r> or r> or ;
\ Is a href attribute external?
link_text_as_attribute? 0= [if] \ XXX TMP -- XXX FIXME unbalanced condition
$variable (link_text)
' (link_text) is link_text \ defered in <fendo.fs>
: link_text! ( ca len -- )
link_text $! ;
: link_text@ ( -- ca len )
link_text $@ ;
: link_text?! ( ca len -- )
link_text@ empty? if link_text! else 2drop then ;
\ If the the string variable `link_text` is empty,
\ store the given string into it.
[then] \ XXX 2014-11-07 this was missing,
\ but still not sure if this is the right position:
\ it was placed here after comparison with
\ <fendo.markup.fendo.link>.
variable link_text_already_evaluated? \ flag
link_text_already_evaluated? off
: evaluate_link_text ( -- )
\ separate? @ \ XXX TMP 2014-08-13 try to fix the bug described in the to-do
link_text@
\ cr ." In evaluate_link_text in fendo.links.fs link_text is " 2dup type \ XXX INFORMER
link_text_already_evaluated? @
\ dup cr ." ( in evaluate_link_text if fendo.links.fs link_text_already_evaluated? = ) " . key drop \ XXX INFORMER
if echo link_text_already_evaluated? off
else separate? @ >r evaluate_content r> separate? ! then
\ separate? ! \ XXX TMP 2014-08-13 try to fix the bug described in the to-do
\ XXX TMP saving and restoring `separate?` makes no difference
;
\ Note: `link_text_already_evaluated?` is turned on by
\ `(parse_link_text)` in <fendo.parser.fs>.
$variable (link_anchor)
' (link_anchor) is link_anchor \ defered in <fendo.fs>
\ variable to_local_anchor? \ XXX OLD
: /anchor ( ca1 len1 -- ca2 len2 ca3 len3 )
s" #" sides/ drop
\ XXX OLD
\ 2 pick 0= over 0<> and
\ dup if ~~ then \ XXX INFORMER
\ to_local_anchor? !
\ ." At the end of `/anchor` `to_local_anchor?` = " to_local_anchor? ? cr \ XXX INFORMER
;
\ Divide a href attribute at its anchor.
\ ca1 len1 = href attribute
\ ca2 len2 = href attribute without the anchor
\ ca3 len3 = anchor without the "#" character
:noname ( ca len -- ca len | ca' len' )
\ ." href parameter in `-anchor` = " 2dup type cr \ XXX INFORMER
/anchor 2drop
; is -anchor \ defered in <fendo.fs>
\ Remove the anchor from a href attribute.
\ ca len = href attribute
\ ca' len' = href attribute without anchor
:noname ( ca len -- ca len | ca' len' )
\ ." href parameter in `-anchor!` = " 2dup type cr \ XXX INFORMER
/anchor link_anchor $!
; is -anchor! \ defered in <fendo.fs>
\ Extract the anchor from a href attribute and store it.
\ ca len = href attribute
\ ca' len' = href attribute without anchor
:noname ( ca len -- ca len | ca' len' )
\ ." href parameter in `-anchor?!` = " 2dup type cr \ XXX INFORMER
/anchor dup if link_anchor $! else 2drop then
; is -anchor?! \ defered in <fendo.fs>
\ Extract the anchor from a href attribute and store it, if not empty.
\ ca len = href attribute
\ ca' len' = href attribute without anchor
: +anchor ( ca1 len1 ca2 len2 -- ca1 len1 | ca3 len3 )
\ ." Anchor parameter in `+anchor` = " 2dup type cr \ XXX INFORMER
dup if 2>r s" #" s+ 2r> s+ else 2drop then ;
\ Add a link anchor to a href attribute.
\ ca1 len1 = href attribute
\ ca2 len2 = anchor, without "#"
variable link_type
1 enum local_link
enum external_link
enum file_link drop
: >link_type_id ( ca len -- n )
2dup external_href? if 2drop external_link exit then
file_href? if file_link exit then
local_link ;
\ Convert an href attribute to its type ID.
: set_link_type ( ca len -- )
>link_type_id link_type ! ;
\ Get and store the type id of an href attribute.
\ XXX TODO -- no href means local, if there is/was an anchor label
: external_link? ( -- f )
link_type @ external_link = ;
: local_link? ( -- f )
link_type @ local_link = ;
: file_link? ( -- f )
link_type @ file_link = ;
: missing_local_link_text ( -- ca len )
\ ." missing_local_link_text" cr \ XXX INFORMER
href=@ -extension 2dup required_data<pid$
>stringer
evaluate title
save_echo echo>string
save_attributes -attributes
evaluate_content echoed $@
save-mem \ XXX TODO needed?
restore_attributes
restore_echo ;
: missing_external_link_text ( -- ca len )
href=@ ;
: missing_file_link_text ( -- ca len )
href=@ basename ;
: missing_link_text ( -- ca len )
local_link? if missing_local_link_text exit then
external_link? if missing_external_link_text exit then \ XXX
file_link? if missing_file_link_text exit then \ XXX
true abort" Wrong link type" \ XXX TMP
;
\ Set a proper link text if it's missing.
\ XXX TODO
: external_class ( -- )
class=@ s" " s+ s" external" s+ class=! ;
\ Add "external" to the class attribute.
:noname ( ca len -- )
\ ." Parameter in `link_anchor+` = " 2dup type cr \ XXX INFORMER
link_anchor $@
\ ." `link_anchor` in `link_anchor+` = " 2dup type cr \ XXX INFORMER
+anchor
; is link_anchor+
\ Restore the link anchor of the local href attribute, if any.
: local_anchor? ( ca len -- f )
dup 0= >r \ is it empty?
current_pid$ str= r> or \ or is it the current page?
link_anchor $@len 0<> and \ and a link anchor exists?
;
\ Is the given href an anchor to the current page?
: local_anchor_href? ( -- f )
href=@ local_anchor? ;
\ Is the `href=` attribute an anchor to the current page?
: (convert_local_link_href) ( ca1 len1 -- ca2 len2 )
dup if pid$>pid# target_file then ;
\ Convert a raw local href to a finished href, if not empty.
: convert_local_link_href ( ca1 len1 -- ca2 len2 )
\ ." Parameter in `convert_local_link_href` = " 2dup type cr \ XXX INFORMER
2dup local_anchor? 0= if (convert_local_link_href) then
\ ." Result in `convert_local_link_href` = " 2dup type cr \ XXX INFORMER
;
\ Convert a raw local href to a finished href, if not a local anchor.
: -file:// ( ca len -- ca' len' )
s" file://" -prefix ;
: convert_file_link_href ( ca len -- ca' len' )
-file:// files_subdir $@ 2swap s+ ;
: convert_link_href ( ca len -- ca' len' )
\ ." Parameter in `convert_link_href` = " 2dup type cr \ XXX INFORMER
link_type @ case
local_link of convert_local_link_href endof
file_link of convert_file_link_href endof
endcase ;
\ ca len = href attribute, without anchor
variable local_link_to_draft_page?
: (tune_local_hreflang) ( a -- )
s" pid#>lang$ 2dup current_lang$" evaluate str=
if 2drop else hreflang=?! then ;
\ Set the hreflang attribute of a local link, if needed.
\ a = page ID of the link destination
: tune_local_hreflang ( a -- )
multilingual? if (tune_local_hreflang) else drop then ;
\ Set the hreflang attribute of a local link, if needed.
\ a = page ID of the link destination
: ?href>current_pid$ ( ca len -- ca' len' )
\ ." In `?href>current_pid$` parameter = " 2dup type cr \ XXX INFORMER
dup 0= if 2drop current_pid$ then
\ ." In `?href>current_pid$` result = " 2dup type cr \ XXX INFORMER
;
: (tune_local_link) ( ca len -- )
\ ." `title=` at the start of `(tune_local_link)` = " title=@ ." «" type ." »" cr \ XXX INFORMER
?href>current_pid$ pid$>(data>)pid# >r
r@ draft? local_link_to_draft_page? !
local_anchor_href? 0= if
r@ description|title unmarkup title=?!
r@ tune_local_hreflang
r@ access_key accesskey=?!
then
r> title link_text?!
\ ." `title=` at the end of `(tune_local_link)` = " title=@ type cr \ XXX INFORMER
;
: tune_local_link ( -- )
\ XXX TODO -- fetch alternative language title and description
href=@ dup if (tune_local_link) else 2drop then ;
: tune_link ( -- ) \ XXX TODO
\ ." `href=` in `tune_link` = " href=@ type cr \ XXX INFORMER
local_link? if tune_local_link then
href=@ convert_link_href href=!
link_text@ empty? if missing_link_text link_text! then
external_link? if external_class then ;
\ Tune the attributes parsed from the link.
: echo_link_text ( -- )
\ ." At `echo_link_text`" cr \ XXX INFORMER
echo_space \ XXX FIXME not always required, but how to know?
evaluate_link_text separate? on ;
\ Echo just the link text.
\ Two hooks for the application,
\ e.g. to add the size of a linked file:
defer link_text_suffix
defer link_suffix
' noop dup is link_text_suffix is link_suffix
\ doc{
\
\ link_text_suffix ( -- )
\
\ A deferred word. A hook for the application. Its default action is
\ ``noop``.
\
\ Used by `linked_file_size`.
\
\ }doc
: anchor_only ( -- )
\ s" #" href=@ /anchor 2swap 2drop s+ href=! \ XXX OLD
s" #" link_anchor $@ s+ href=! ;
\ Remove the target file from the href attribute,
\ leaving only the anchor.
: (echo_link) ( -- )
\ to_local_anchor? @ if \ XXX OLD
\ ." `title=` in `(echo_link)` before `local_anchor_href?` = " title=@ type cr \ XXX INFORMER
local_anchor_href?
\ ." In (echo_link) `local_anchor_href?` = " dup . cr \ XXX INFORMER
\ ." `title=` in `(echo_link)` after `local_anchor_href?` = " title=@ type cr \ XXX INFORMER
if anchor_only then
[<a>] evaluate_link_text link_text_suffix [</a>] link_suffix ;
\ Echo the final link.
: echo_link? ( -- f )
href=@ nip 0<> link_anchor $@len 0<> or
local_link_to_draft_page? @ 0= and ;
\ Can the current link be echoed?
: reset_link ( -- )
0 link_anchor $!len s" " link_text!
local_link_to_draft_page? off ;
\ Reset the link attributes that are not actual HTML attributes,
\ and are not reseted by the HTML tags layer.
: echo_link ( -- )
\ ." In `echo_link`, `link_text$` = " link_text@ type cr \ XXX INFORMER
\ ." `href=` in `echo_link` = " href=@ 2dup type ." [" .s 2drop ." ]" cr \ XXX INFORMER
\ ." `href=` in `echo_link` = " href=@ type cr \ XXX INFORMER
tune_link echo_link?
if (echo_link) else -attributes echo_link_text then reset_link ;
\ Echo a link, if possible.
\ All link attributes have been set.
\ XXX FIXME link_text@ here returns a string with macros already
\ parsed! why?
\ ==============================================================
\ Links {{{1
defer (get_link_href) ( ca len -- )
\ ca len = page ID, URL or shortcut
\ Defined in <fendo.markup.fendo.link.fs>.
: (link) ( ca len -- )
\ ." `title=` in `(link)` = " title=@ type cr \ XXX INFORMER
(get_link_href) echo_link ;
\ Create a link.
\ Its attributes and link text have to be set previously.
\ ca len = page ID, URL or shortcut
: link ( ca1 len1 ca2 len2 -- )
\ ." In `link` the link text is " 2dup type cr \ XXX INFORMER
\ ." In `link` the page ID is " 2over type cr \ XXX INFORMER
\ ." `title=` in `link` = " title=@ type cr \ XXX INFORMER
link_text! (link) ;
\ doc{
\
\ link ( ca1 len1 ca2 len2 -- )
\
\ Create a link of any type out of a string _ca1 len1_, which
\ contains a page ID, an URL or a shortcut, and a second string _ca2
\ len2_, which contains the link text.
\
\ The attributes have to be set before executing ``link``, using the
\ corresponding attribute-parsing words.
\
\ Usage example:
\ ----
\ <[ s" http://programandala.net" s" My website" title=" Home
\ page of programandala.net" link ]>
\
\ \ The string notation recognized by Gforth and other Forth
\ \ systems can be used instead of ``s"``:
\
\ <[ "http://programandala.net" "My website" title=" Home
\ page of programandala.net" link ]>
\ ----
\ Note that ``title="``, ``class="`` etc. are Forth parsing words,
\ therefore, contrary to actual HTML, they must be followed by a
\ space. Leading, trailing and double spaces will be removed from
\ the parsed string before storing it in the corresponding attribute
\ variable. Therefore, the parsed string can span on multiple lines
\ and the line breaks will be ignored, replaced by single spaces.
\
\ An alternative link markup is provided by `[[`.
\
\ See also: `<[`.
\
\ }doc
: link<pid$ ( ca len -- )
2dup pid$>pid# title link_text?! (link) ;
\ Create a link to a local page.
\ Its attributes have to be set previously.
\ If `link_text` is not set, the page title will be used.
\ ca len = page ID or shortcut to it
\ XXX TODO -- make it work with anchors!?
: link<pid# ( pid -- )
dup title link_text?! pid#>pid$ (link) ;
\ Create a link to a local page.
\ Its attributes have to be set previously.
\ If `link_text` is not set, the page title will be used.
.( fendo.links.fs ) cr
\ ==============================================================
\ Change log {{{1
\ 2013-11-11: Code extracted from <fendo_markup_wiki.fs>: `link`.
\
\ 2013-11-26: Change: several words renamed, after a new uniform
\ notation: "pid$" and "pid#" for both types of page IDs.
\
\ 2014-03-03: New: `link<pid#`.
\
\ 2014-03-03: Change: `title_link` renamed to `link<pid$`.
\
\ 2014-06-15: Fix: repeated evaluation of link texts is solved with
\ the new `link_text_already_evaluated?` flag.
\
\ 2014-07-11: Change: `pid$>url` moved to <fendo.data.fs>.
\
\ 2014-08-15: Fix: comment updated.
\
\ 2014-08-15: Fix: 'link_text_already_evaluated? off' was missing in
\ `evaluate_link_text`.
\
\ 2014-10-12: Fix: `evaluate_link_text` now preserves the content of
\ `separate?` in the return stack; it ruined the stack before calling
\ `evaluate_content`.
\
\ 2014-11-07: a `[then]` was missing (more details in a comment marked
\ with this date).
\
\ 2014-11-08: Change: `unmarkup` (just implemented) is used instead of
\ hard-coded plain text versions of some data fields.
\
\ 2014-11-09: Change: all 'true [if]' that compiled code that long ago
\ was moved from <fendo.markup.fendo.link.fs> have been removed. Those
\ conditions were needed just in case strange things happened.
\
\ 2014-11-11: Change: `+anchor` rewritten without locals.
\
\ 2014-11-14: Change: `reset_link` resets also `link_anchor`.
\
\ 2014-11-16: Fix: `link_anchor+` removed from `convert_link_href`.
\ This is done in a lower level, in `target_file` (defined in
\ <fendo.data.fs>).
\
\ 2014-11-27: Fix: now `echo_link` executes `-attributes` when only
\ the link text is printed; formerly the link attributes were used by
\ the next HTML tag.
\
\ 2014-11-27: Fix: now `echo_link_text` does 'separate? on' at the
\ end.
\
\ 2014-11-27: New: `to_local_anchor?` flag, set by `/anchor`.
\
\ 2014-11-27: Fix: now `echo_link?` uses also `link_anchor`.
\
\ 2014-11-28: Fix: The `to_local_anchor?` flag is removed, because it
\ was overwritten and ruined several times during the link process.
\ The word `local_anchor_href?` and `local_anchor?` are used instead;
\ the calculation is done only when required, with the contents of
\ `href=` and `link_anchor`. `convert_local_link_href` is updated
\ accordingly.
\
\ 2014-12-06: Fix: Now `(tune_local_link)` uses the calcutated datum
\ `description|title`, just written for the fix (in <fendo.data.fs>),
\ instead of the datum `description`. This makes sure links to
\ description-less pages have a link title.
\
\ 2015-02-12: Change: `link_anchor` and `link_text` are defered in
\ <fendo.fs> and defined here. Required because of a fix.
\
\ 2017-06-22: Update source style, layout and header.
\
\ 2017-11-04: Update to Galope 0.103.0: Replace `-path` with Gforth's
\ `basename`.
\
\ 2018-12-08: Update notation of Forth words in comments and strings.
\
\ 2018-12-08: Update notation of page IDs in comments and strings.
\
\ 2018-12-17: Update: replace `pid$>data>pid#` with `pid$>pid#`.
\
\ 2019-03-21: Document `link_text_suffix`.
\
\ 2020-04-14: Rewrite `external_class` without Galope's `sb&`. Replace
\ old `>sb` with `>stringer`.
\
\ 2020-11-14: Document `link`.
\ vim: filetype=gforth