1
1
/*
2
- * Copyright (c) 1998, 2019 , Oracle and/or its affiliates. All rights reserved.
2
+ * Copyright (c) 1998, 2023 , Oracle and/or its affiliates. All rights reserved.
3
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
4
*
5
5
* This code is free software; you can redistribute it and/or modify it
@@ -98,6 +98,11 @@ private static class Item {
98
98
*/
99
99
final boolean relative ;
100
100
101
+ /**
102
+ * Indicates that docs use old-form of anchors.
103
+ */
104
+ final boolean useOldFormId ;
105
+
101
106
/**
102
107
* Constructor to build a Extern Item object and map it with the element name.
103
108
* If the same element name is found in the map, then the first mapped
@@ -108,10 +113,11 @@ private static class Item {
108
113
* file is picked.
109
114
* @param relative True if path is URL, false if directory path.
110
115
*/
111
- Item (String elementName , DocPath path , boolean relative ) {
116
+ Item (String elementName , DocPath path , boolean relative , boolean useOldFormId ) {
112
117
this .elementName = elementName ;
113
118
this .path = path ;
114
119
this .relative = relative ;
120
+ this .useOldFormId = useOldFormId ;
115
121
}
116
122
117
123
/**
@@ -179,7 +185,7 @@ public DocLink getExternalLink(Element element, DocPath relativepath, String fil
179
185
DocPath p = fnd .relative ?
180
186
relativepath .resolve (fnd .path ).resolve (filename ) :
181
187
fnd .path .resolve (filename );
182
- return new DocLink (p , "is-external=true" , memberName );
188
+ return new DocLink (p , "is-external=true" , fnd . useOldFormId ? getOldFormHtmlName ( memberName ) : memberName );
183
189
}
184
190
185
191
/**
@@ -212,6 +218,18 @@ public boolean link(String url, String elemlisturl, Reporter reporter) throws Do
212
218
return link (url , elemlisturl , reporter , true );
213
219
}
214
220
221
+ /**
222
+ * Checks if platform docs for the specified version use old-form anchors.
223
+ * Old-form anchors are used by Oracle docs for JDKs 8 and 9.
224
+ * It can be checked on https://docs.oracle.com/javase/<version>/docs/api
225
+ *
226
+ * @param version
227
+ * @return True if docs use old-form anchors
228
+ */
229
+ private boolean isOldFormPlatformDocs (int version ) {
230
+ return 8 == version || 9 == version ;
231
+ }
232
+
215
233
/*
216
234
* Build the extern element list from given URL or the directory path.
217
235
* Flag error if the "-link" or "-linkoffline" option is already used.
@@ -292,7 +310,7 @@ private String adjustEndFileSeparator(String url) {
292
310
private void readElementListFromURL (String urlpath , URL elemlisturlpath ) throws Fault {
293
311
try {
294
312
URL link = elemlisturlpath .toURI ().resolve (DocPaths .ELEMENT_LIST .getPath ()).toURL ();
295
- readElementList (link .openStream (), urlpath , false );
313
+ readElementList (link .openStream (), urlpath , false , false );
296
314
} catch (URISyntaxException | MalformedURLException exc ) {
297
315
throw new Fault (configuration .getText ("doclet.MalformedURL" , elemlisturlpath .toString ()), exc );
298
316
} catch (IOException exc ) {
@@ -309,7 +327,7 @@ private void readElementListFromURL(String urlpath, URL elemlisturlpath) throws
309
327
private void readAlternateURL (String urlpath , URL elemlisturlpath ) throws Fault {
310
328
try {
311
329
URL link = elemlisturlpath .toURI ().resolve (DocPaths .PACKAGE_LIST .getPath ()).toURL ();
312
- readElementList (link .openStream (), urlpath , false );
330
+ readElementList (link .openStream (), urlpath , false , true );
313
331
} catch (URISyntaxException | MalformedURLException exc ) {
314
332
throw new Fault (configuration .getText ("doclet.MalformedURL" , elemlisturlpath .toString ()), exc );
315
333
} catch (IOException exc ) {
@@ -332,27 +350,27 @@ private void readElementListFromFile(String path, DocFile elemListPath)
332
350
file = file .resolveAgainst (DocumentationTool .Location .DOCUMENTATION_OUTPUT );
333
351
}
334
352
if (file .exists ()) {
335
- readElementList (file , path );
353
+ readElementList (file , path , false );
336
354
} else {
337
355
DocFile file1 = elemListPath .resolve (DocPaths .PACKAGE_LIST );
338
356
if (!(file1 .isAbsolute () || linkoffline )) {
339
357
file1 = file1 .resolveAgainst (DocumentationTool .Location .DOCUMENTATION_OUTPUT );
340
358
}
341
359
if (file1 .exists ()) {
342
- readElementList (file1 , path );
360
+ readElementList (file1 , path , true );
343
361
} else {
344
362
throw new Fault (configuration .getText ("doclet.File_error" , file .getPath ()), null );
345
363
}
346
364
}
347
365
}
348
366
349
- private void readElementList (DocFile file , String path ) throws Fault , DocFileIOException {
367
+ private void readElementList (DocFile file , String path , boolean isOldFormDoc ) throws Fault , DocFileIOException {
350
368
try {
351
369
if (file .canRead ()) {
352
370
boolean pathIsRelative
353
371
= !isUrl (path )
354
372
&& !DocFile .createFileForInput (configuration , path ).isAbsolute ();
355
- readElementList (file .openInputStream (), path , pathIsRelative );
373
+ readElementList (file .openInputStream (), path , pathIsRelative , isOldFormDoc );
356
374
} else {
357
375
throw new Fault (configuration .getText ("doclet.File_error" , file .getPath ()), null );
358
376
}
@@ -370,7 +388,7 @@ private void readElementList(DocFile file, String path) throws Fault, DocFileIOE
370
388
* @param relative Is path relative?
371
389
* @throws IOException if there is a problem reading or closing the stream
372
390
*/
373
- private void readElementList (InputStream input , String path , boolean relative )
391
+ private void readElementList (InputStream input , String path , boolean relative , boolean isOldFormDoc )
374
392
throws IOException {
375
393
try (BufferedReader in = new BufferedReader (new InputStreamReader (input ))) {
376
394
String elemname = null ;
@@ -382,7 +400,7 @@ private void readElementList(InputStream input, String path, boolean relative)
382
400
elempath = basePath ;
383
401
if (elemname .startsWith (DocletConstants .MODULE_PREFIX )) {
384
402
moduleName = elemname .replace (DocletConstants .MODULE_PREFIX , "" );
385
- Item item = new Item (moduleName , elempath , relative );
403
+ Item item = new Item (moduleName , elempath , relative , isOldFormDoc );
386
404
moduleItems .put (moduleName , item );
387
405
} else {
388
406
DocPath pkgPath = DocPath .create (elemname .replace ('.' , '/' ));
@@ -392,7 +410,7 @@ private void readElementList(InputStream input, String path, boolean relative)
392
410
elempath = elempath .resolve (pkgPath );
393
411
}
394
412
String actualModuleName = checkLinkCompatibility (elemname , moduleName , path );
395
- Item item = new Item (elemname , elempath , relative );
413
+ Item item = new Item (elemname , elempath , relative , isOldFormDoc );
396
414
packageItems .computeIfAbsent (actualModuleName , k -> new TreeMap <>())
397
415
.put (elemname , item );
398
416
}
@@ -455,4 +473,65 @@ private String checkLinkCompatibility(String packageName, String moduleName, Str
455
473
}
456
474
return moduleName == null ? DocletConstants .DEFAULT_ELEMENT_NAME : moduleName ;
457
475
}
476
+
477
+ /**
478
+ * Converts a name to an old-form HTML name (old-form id).
479
+ *
480
+ * @param name the string that needs to be converted to a valid HTML name
481
+ * @return old-form HTML name
482
+ */
483
+ public static String getOldFormHtmlName (String name ) {
484
+ /* The HTML 4 spec at http://www.w3.org/TR/html4/types.html#h-6.2 mentions
485
+ * that the name/id should begin with a letter followed by other valid characters.
486
+ * The HTML 5 spec (draft) is more permissive on names/ids where the only restriction
487
+ * is that it should be at least one character long and should not contain spaces.
488
+ * The spec draft is @ http://www.w3.org/html/wg/drafts/html/master/dom.html#the-id-attribute.
489
+ *
490
+ * For HTML 4, we need to check for non-characters at the beginning of the name and
491
+ * substitute it accordingly, "_" and "$" can appear at the beginning of a member name.
492
+ * The method substitutes "$" with "Z:Z:D" and will prefix "_" with "Z:Z".
493
+ */
494
+
495
+ if (null == name )
496
+ return name ;
497
+
498
+ StringBuilder sb = new StringBuilder ();
499
+ for (int i = 0 ; i < name .length (); i ++) {
500
+ char ch = name .charAt (i );
501
+ switch (ch ) {
502
+ case '(' :
503
+ case ')' :
504
+ case '<' :
505
+ case '>' :
506
+ case ',' :
507
+ sb .append ('-' );
508
+ break ;
509
+ case ' ' :
510
+ case '[' :
511
+ break ;
512
+ case ']' :
513
+ sb .append (":A" );
514
+ break ;
515
+ // Any appearance of $ needs to be substituted with ":D" and not with hyphen
516
+ // since a field name "P$$ and a method P(), both valid member names, can end
517
+ // up as "P--". A member name beginning with $ needs to be substituted with
518
+ // "Z:Z:D".
519
+ case '$' :
520
+ if (i == 0 )
521
+ sb .append ("Z:Z" );
522
+ sb .append (":D" );
523
+ break ;
524
+ // A member name beginning with _ needs to be prefixed with "Z:Z" since valid anchor
525
+ // names can only begin with a letter.
526
+ case '_' :
527
+ if (i == 0 )
528
+ sb .append ("Z:Z" );
529
+ sb .append (ch );
530
+ break ;
531
+ default :
532
+ sb .append (ch );
533
+ }
534
+ }
535
+ return sb .toString ();
536
+ }
458
537
}
1 commit comments
openjdk-notifier[bot] commentedon Jul 5, 2023
Review
Issues