View Javadoc
1   /*
2    * Copyright (c) 2002-2025 Gargoyle Software Inc.
3    *
4    * Licensed under the Apache License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    * https://www.apache.org/licenses/LICENSE-2.0
8    *
9    * Unless required by applicable law or agreed to in writing, software
10   * distributed under the License is distributed on an "AS IS" BASIS,
11   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12   * See the License for the specific language governing permissions and
13   * limitations under the License.
14   */
15  package org.htmlunit;
16  
17  import static org.htmlunit.javascript.configuration.SupportedBrowser.CHROME;
18  import static org.htmlunit.javascript.configuration.SupportedBrowser.EDGE;
19  import static org.htmlunit.javascript.configuration.SupportedBrowser.FF;
20  import static org.htmlunit.javascript.configuration.SupportedBrowser.FF_ESR;
21  
22  import org.htmlunit.javascript.configuration.BrowserFeature;
23  
24  /**
25   * Constants of various features of each {@link BrowserVersion}.
26   *
27   * @author Ahmed Ashour
28   * @author Marc Guillemot
29   * @author Sudhan Moghe
30   * @author Ronald Brill
31   * @author Frank Danek
32   * @author Carsten Steul
33   * @author Anton Demydenko
34   */
35  public enum BrowserVersionFeatures {
36  
37      /** Ignore target when {@code href} is a javascript snippet. */
38      @BrowserFeature({CHROME, EDGE})
39      ANCHOR_SEND_PING_REQUEST,
40  
41      /** Background image is 'initial'. */
42      @BrowserFeature({CHROME, EDGE})
43      CSS_BACKGROUND_INITIAL,
44  
45      /** Background image is 'rgba(0, 0, 0, 0)'. */
46      @BrowserFeature({FF, FF_ESR})
47      CSS_BACKGROUND_RGBA,
48  
49      /** {@code CSSFontFaceRule.cssText} uses one more blank. */
50      @BrowserFeature(FF_ESR)
51      CSS_CSSTEXT_FF_STYLE,
52  
53      /** Is display style 'block'. */
54      @BrowserFeature({FF, FF_ESR})
55      CSS_DISPLAY_BLOCK,
56  
57      /** The default value of the display property for the 'noscript' tag is 'inline' instead of the default one. */
58      @BrowserFeature({CHROME, EDGE})
59      CSS_NOSCRIPT_DISPLAY_INLINE,
60  
61      /** The default value of the display property for the 'rp' tag is 'none'. */
62      @BrowserFeature({FF, FF_ESR})
63      CSS_RP_DISPLAY_NONE,
64  
65      /** The default value of the display property for the 'rt' tag is always 'ruby-text'. */
66      @BrowserFeature({FF, FF_ESR})
67      CSS_RT_DISPLAY_RUBY_TEXT_ALWAYS,
68  
69      /** The context menu MouseEvent has a detail of 1. */
70      @BrowserFeature(FF)
71      EVENT_CONTEXT_MENU_HAS_DETAIL_1,
72  
73      /** Triggers the onfocus event when focusing the body on load. */
74      @BrowserFeature({FF, FF_ESR})
75      EVENT_FOCUS_ON_LOAD,
76  
77      /** <code>AnimationEvent</code> can not be created by calling document.createEvent('AnimationEvent'). */
78      @BrowserFeature({FF, FF_ESR})
79      EVENT_ONANIMATION_DOCUMENT_CREATE_NOT_SUPPORTED,
80  
81      /** Triggers 'onclick' event handler using <code>PointerEvent</code>. */
82      @BrowserFeature({CHROME, EDGE, FF})
83      EVENT_ONCLICK_USES_POINTEREVENT,
84  
85      /** <code>CloseEvent</code> can not be created by calling document.createEvent('CloseEvent'). */
86      @BrowserFeature({FF, FF_ESR})
87      EVENT_ONCLOSE_DOCUMENT_CREATE_NOT_SUPPORTED,
88  
89      /** <code>PopStateEvent</code> can not be created by calling document.createEvent('PopStateEvent'). */
90      @BrowserFeature({FF, FF_ESR})
91      EVENT_ONPOPSTATE_DOCUMENT_CREATE_NOT_SUPPORTED,
92  
93      /** Supports event type 'TextEvent'. */
94      @BrowserFeature({FF, FF_ESR})
95      EVENT_TYPE_MUTATIONEVENT,
96  
97      /** Supports event type 'TextEvent'. */
98      @BrowserFeature({CHROME, EDGE, FF})
99      EVENT_TYPE_TEXTEVENT,
100 
101     /** Supports event type 'WheelEvent'. */
102     @BrowserFeature({CHROME, EDGE})
103     EVENT_TYPE_WHEELEVENT,
104 
105     /** Form elements are able to refer to the for by using the from attribute. */
106     @BrowserFeature({CHROME, EDGE})
107     FORM_IGNORE_REL_NOREFERRER,
108 
109     /** Form submit includes the Cache-Control: max-age=0 header. */
110     @BrowserFeature({CHROME, EDGE})
111     FORM_SUBMISSION_HEADER_CACHE_CONTROL_MAX_AGE,
112 
113     /** Should org.htmlunit.javascript.host.html.HTMLBaseFontElement#isEndTagForbidden(). */
114     @BrowserFeature({FF, FF_ESR})
115     HTMLBASEFONT_END_TAG_FORBIDDEN,
116 
117     /** willValidate does not check the readonly property. */
118     @BrowserFeature({FF, FF_ESR})
119     HTMLBUTTON_WILL_VALIDATE_IGNORES_READONLY,
120 
121     /** HtmlCollection.namedItem searches by id first. */
122     @BrowserFeature({CHROME, EDGE})
123     HTMLCOLLECTION_NAMED_ITEM_ID_FIRST,
124 
125     /** Calling cookies setter with blank string does not reset the cookies. */
126     @BrowserFeature({CHROME, EDGE, FF})
127     HTMLDOCUMENT_COOKIES_IGNORE_BLANK,
128 
129     /**
130     /** {@code document.getElementsByName} returns an empty list if called with the empty string.
131      */
132     @BrowserFeature({FF, FF_ESR})
133     HTMLDOCUMENT_ELEMENTS_BY_NAME_EMPTY,
134 
135     /** Calls to <code>document.XYZ</code> also looks at frames. */
136     @BrowserFeature({CHROME, EDGE})
137     HTMLDOCUMENT_GET_ALSO_FRAMES,
138 
139     /** Removing the active element from the dom tree triggers the onblur event. */
140     @BrowserFeature({CHROME, EDGE})
141     HTMLELEMENT_REMOVE_ACTIVE_TRIGGERS_BLUR_EVENT,
142 
143     /** Handle blank source like empty. */
144     @BrowserFeature({CHROME, EDGE})
145     HTMLIMAGE_BLANK_SRC_AS_EMPTY,
146 
147     /** Empty src attribute sets display to false. */
148     @BrowserFeature({FF, FF_ESR})
149     HTMLIMAGE_EMPTY_SRC_DISPLAY_FALSE,
150 
151     /** Is document.cretaeElement('image') an HTMLElement. */
152     @BrowserFeature({FF, FF_ESR})
153     HTMLIMAGE_HTMLELEMENT,
154 
155     /** Is document.cretaeElement('image') an HTMLUnknownElement. */
156     @BrowserFeature({CHROME, EDGE})
157     HTMLIMAGE_HTMLUNKNOWNELEMENT,
158 
159     /** Clicking an image input submits the value as param if defined. */
160     @BrowserFeature({CHROME, EDGE})
161     HTMLIMAGE_NAME_VALUE_PARAMS,
162 
163     /** HTMLInputElement image type is not supported. */
164     @BrowserFeature({CHROME, EDGE})
165     HTMLINPUT_TYPE_IMAGE_IGNORES_CUSTOM_VALIDITY,
166 
167     /** HTMLInputElement month type is supported. */
168     @BrowserFeature({CHROME, EDGE})
169     HTMLINPUT_TYPE_MONTH_SUPPORTED,
170 
171     /** HTMLInputElement week type is supported. */
172     @BrowserFeature({CHROME, EDGE})
173     HTMLINPUT_TYPE_WEEK_SUPPORTED,
174 
175     /** If the type is present for a link only use if type is text/css. */
176     @BrowserFeature({CHROME, EDGE})
177     HTMLLINK_CHECK_TYPE_FOR_STYLESHEET,
178 
179     /** willValidate does not check the readonly property. */
180     @BrowserFeature({FF, FF_ESR})
181     HTMLSELECT_WILL_VALIDATE_IGNORES_READONLY,
182 
183     /** Should org.htmlunit.javascript.host.html.HTMLTrackElement#isEndTagForbidden(). */
184     @BrowserFeature({FF, FF_ESR})
185     HTMLTRACK_END_TAG_FORBIDDEN,
186 
187     /** HTML parser supports the 'command' tag. */
188     @BrowserFeature({CHROME, EDGE})
189     HTML_COMMAND_TAG,
190 
191     /** HTML parser supports the 'layer' tag. */
192     @BrowserFeature({CHROME, EDGE})
193     HTML_LAYER_TAG,
194 
195     /** Additionally support dates in format "d/M/yyyy". */
196     @BrowserFeature({FF, FF_ESR})
197     HTTP_COOKIE_EXTENDED_DATE_PATTERNS_1,
198 
199     /** Dates format pattern 2. */
200     @BrowserFeature({CHROME, EDGE})
201     HTTP_COOKIE_EXTENDED_DATE_PATTERNS_2,
202 
203     /** domain '.org' is handled as 'org'. */
204     @BrowserFeature({FF, FF_ESR})
205     HTTP_COOKIE_REMOVE_DOT_FROM_ROOT_DOMAINS,
206 
207     /** Browser sends Sec-ch headers. */
208     @BrowserFeature({CHROME, EDGE})
209     HTTP_HEADER_CH_UA,
210 
211     /** Browser sends Priority headers. */
212     @BrowserFeature({FF, FF_ESR})
213     HTTP_HEADER_PRIORITY,
214 
215     /** The anchor hostname setter ignores blank url's. */
216     @BrowserFeature({FF, FF_ESR})
217     JS_ANCHOR_HOSTNAME_IGNORE_BLANK,
218 
219     /** The anchor pathname detects url's starting with one letter as file url's. */
220     @BrowserFeature(FF_ESR)
221     JS_ANCHOR_PATHNAME_DETECT_WIN_DRIVES_URL,
222 
223     /**
224      * The anchor pathname detects url's starting with one letter as file url's
225      * and replaces them with the file protocol. */
226     @BrowserFeature({CHROME, EDGE})
227     JS_ANCHOR_PATHNAME_DETECT_WIN_DRIVES_URL_REPLACE,
228 
229     /** The anchor pathname prefixes file url's with '/'. */
230     @BrowserFeature({CHROME, EDGE})
231     JS_ANCHOR_PATHNAME_PREFIX_WIN_DRIVES_URL,
232 
233     /** The anchor protocol property converts drive letters to uppercase. */
234     @BrowserFeature({CHROME, EDGE})
235     JS_ANCHOR_PROTOCOL_COLON_UPPER_CASE_DRIVE_LETTERS,
236 
237     /** An area element without a href attribute is focusable. */
238     @BrowserFeature({FF, FF_ESR})
239     JS_AREA_WITHOUT_HREF_FOCUSABLE,
240 
241     /** Sorting an array using a user defined comperator accepts inconsistent iterators. */
242     @BrowserFeature({FF, FF_ESR})
243     JS_ARRAY_SORT_ACCEPTS_INCONSISTENT_COMPERATOR,
244 
245     /** AudioProcessingEvent ctor is callable. */
246     @BrowserFeature({CHROME, EDGE})
247     JS_AUDIO_PROCESSING_EVENT_CTOR,
248 
249     /** toDataURL for canvas returns the CHROME version of the PNG. */
250     @BrowserFeature({CHROME, EDGE})
251     JS_CANVAS_DATA_URL_CHROME_PNG,
252 
253     /** ClientHeight for input is 17. */
254     @BrowserFeature({CHROME, EDGE})
255     JS_CLIENTHEIGHT_INPUT_17,
256 
257     /** ClientHeight for input is 18. */
258     @BrowserFeature({FF, FF_ESR})
259     JS_CLIENTHEIGHT_INPUT_18,
260 
261     /** ClientHeight for radio button and checkbox is 10. */
262     @BrowserFeature(FF_ESR)
263     JS_CLIENTHEIGHT_RADIO_CHECKBOX_10,
264 
265     /** ClientHeight for radio button and checkbox is 14. */
266     @BrowserFeature(FF)
267     JS_CLIENTHEIGHT_RADIO_CHECKBOX_14,
268 
269     /** ClientWidth for text/password input is 154. */
270     @BrowserFeature(FF_ESR)
271     JS_CLIENTWIDTH_INPUT_TEXT_154,
272 
273     /** ClientWidth for text/password input is 173. */
274     @BrowserFeature({CHROME, EDGE})
275     JS_CLIENTWIDTH_INPUT_TEXT_173,
276 
277     /** ClientWidth for radio button and checkbox is 10. */
278     @BrowserFeature(FF_ESR)
279     JS_CLIENTWIDTH_RADIO_CHECKBOX_10,
280 
281     /** ClientWidth for radio button and checkbox is 14. */
282     @BrowserFeature(FF)
283     JS_CLIENTWIDTH_RADIO_CHECKBOX_14,
284 
285     /** item is enumerated before length property of CSSRuleList. */
286     @BrowserFeature({FF, FF_ESR})
287     JS_CSSRULELIST_ENUM_ITEM_LENGTH,
288 
289     /** Javascript document.evaluate creates a new result object even if provided as param. */
290     @BrowserFeature({CHROME, EDGE})
291     JS_DOCUMENT_EVALUATE_RECREATES_RESULT,
292 
293     /** The browser has selection {@code rangeCount}. */
294     @BrowserFeature({FF, FF_ESR})
295     JS_DOCUMENT_SELECTION_RANGE_COUNT,
296 
297     /** Javascript {@code Error.captureStackTrace}. */
298     @BrowserFeature({CHROME, EDGE})
299     JS_ERROR_CAPTURE_STACK_TRACE,
300 
301     /** Javascript {@code Error.stackTraceLimit}. */
302     @BrowserFeature({CHROME, EDGE})
303     JS_ERROR_STACK_TRACE_LIMIT,
304 
305     /** Javascript InputEvent reads the inputType property from data. */
306     @BrowserFeature({FF, FF_ESR})
307     JS_EVENT_INPUT_CTOR_INPUTTYPE,
308 
309     /** Javascript KeyboardEvent reads the which property from data. */
310     @BrowserFeature({FF, FF_ESR})
311     JS_EVENT_KEYBOARD_CTOR_WHICH,
312 
313     /** form.dispatchEvent(e) submits the form if the event is of type 'submit'. */
314     @BrowserFeature({FF, FF_ESR})
315     JS_FORM_DISPATCHEVENT_SUBMITS,
316 
317     /** Executes the {@code onload} handler, regardless of the whether the element was already attached to the page. */
318     @BrowserFeature({FF, FF_ESR})
319     JS_IFRAME_ALWAYS_EXECUTE_ONLOAD,
320 
321     /**
322      * Getting the width and height of an image tag without a source returns 16x16;
323      * for invalid values returns 0.
324      */
325     @BrowserFeature({CHROME, EDGE})
326     JS_IMAGE_WIDTH_HEIGHT_RETURNS_16x16_0x0,
327 
328     /**
329      * Getting the width and height of an image tag without a source returns 24x24;
330      * for invalid values returns 0x0.
331      */
332     @BrowserFeature({FF, FF_ESR})
333     JS_IMAGE_WIDTH_HEIGHT_RETURNS_24x24_0x0,
334 
335     /** Indicates that innerText add a nl when reaching svg element. */
336     @BrowserFeature({CHROME, EDGE})
337     JS_INNER_TEXT_SVG_NL,
338 
339     /** The value is ignored when the type of an week/month input is changed. */
340     @BrowserFeature({CHROME, EDGE})
341     JS_INPUT_CHANGE_TYPE_DROPS_VALUE_WEEK_MONTH,
342 
343     /** FF accepts all chars. */
344     @BrowserFeature({FF, FF_ESR})
345     JS_INPUT_NUMBER_ACCEPT_ALL,
346 
347     /** FF comma at end is not an integer. */
348     @BrowserFeature({FF, FF_ESR})
349     JS_INPUT_NUMBER_DOT_AT_END_IS_DOUBLE,
350 
351     /** Indicates that Intl.v8BreakIterator is supported. */
352     @BrowserFeature({CHROME, EDGE})
353     JS_INTL_V8_BREAK_ITERATOR,
354 
355     /** Indicates that window.Iterator is supported. */
356     @BrowserFeature({CHROME, EDGE, FF})
357     JS_ITERATOR_VISIBLE_IN_WINDOW,
358 
359     /** For the 'about' protocol the location always returns an empty query. */
360     @BrowserFeature({FF, FF_ESR})
361     JS_LOCATION_IGNORE_QUERY_FOR_ABOUT_PROTOCOL,
362 
363     /** Reload sends a referrer header. */
364     @BrowserFeature({CHROME, EDGE})
365     JS_LOCATION_RELOAD_REFERRER,
366 
367     /** Type property of menu returns the current (maybe invalid) value. */
368     @BrowserFeature({FF, FF_ESR})
369     JS_MENU_TYPE_PASS,
370 
371     /** Indicates if the String representation of a native function is without newline. */
372     @BrowserFeature({CHROME, EDGE})
373     JS_NATIVE_FUNCTION_TOSTRING_COMPACT,
374 
375     /** Indicates if the String representation of a native function has a newline for empty parameter list. */
376     @BrowserFeature({FF, FF_ESR})
377     JS_NATIVE_FUNCTION_TOSTRING_NL,
378 
379     /** Navigator.doNotTrack returns unspecified if not set. */
380     @BrowserFeature({FF, FF_ESR})
381     JS_NAVIGATOR_DO_NOT_TRACK_UNSPECIFIED,
382 
383     /** Indicates that someObj.offsetParent returns null, it someObj has fixed style. */
384     @BrowserFeature({CHROME, EDGE})
385     JS_OFFSET_PARENT_NULL_IF_FIXED,
386 
387     /** element.outerHTML removes all children from detached node. */
388     @BrowserFeature({CHROME, EDGE})
389     JS_OUTER_HTML_THROWS_FOR_DETACHED,
390 
391     /** Indicates that the {@code Object.getOwnPropertyDescriptor.get} contains name. */
392     @BrowserFeature({FF, FF_ESR})
393     JS_PROPERTY_DESCRIPTOR_NAME,
394 
395     /** script tags created from js as child of templates are processed if added to the dom. */
396     @BrowserFeature({CHROME, EDGE})
397     JS_SCRIPT_IN_TEMPLATE_EXECUTED_ON_ATTACH,
398 
399     /** Javascript selectorText property returns selectors in lower case. */
400     @BrowserFeature({CHROME, EDGE})
401     JS_SELECTOR_TEXT_LOWERCASE,
402 
403     /** Indicates that select.options.remove ignores the call if index is too large. */
404     @BrowserFeature({FF, FF_ESR})
405     JS_SELECT_REMOVE_IGNORE_IF_INDEX_OUTSIDE,
406 
407     /** Whether to add to the storage even preserved words. */
408     @BrowserFeature({FF, FF_ESR})
409     JS_STORAGE_PRESERVED_INCLUDED,
410 
411     /** Indicates wordSpacing support percent values. */
412     @BrowserFeature({FF, FF_ESR})
413     JS_STYLE_WORD_SPACING_ACCEPTS_PERCENT,
414 
415     /** Set span zo zero if provided value is invalid. */
416     @BrowserFeature({CHROME, EDGE})
417     JS_TABLE_SPAN_SET_ZERO_IF_INVALID,
418 
419     /** window.getComputedStyle works with pseudo selectors without colon in front. */
420     @BrowserFeature({CHROME, EDGE})
421     JS_WINDOW_COMPUTED_STYLE_PSEUDO_ACCEPT_WITHOUT_COLON,
422 
423     /** Javascript InstallTrigger property set to null. */
424     @BrowserFeature({FF, FF_ESR})
425     JS_WINDOW_INSTALL_TRIGGER_NULL,
426 
427     /**
428      * Difference of window.outer/inner height is 138.
429      */
430     @BrowserFeature(EDGE)
431     JS_WINDOW_OUTER_INNER_HEIGHT_DIFF_138,
432 
433     /**
434      * Difference of window.outer/inner height is 147.
435      */
436     @BrowserFeature(CHROME)
437     JS_WINDOW_OUTER_INNER_HEIGHT_DIFF_147,
438 
439     /**
440      * Difference of window.outer/inner height is 91.
441      */
442     @BrowserFeature(FF_ESR)
443     JS_WINDOW_OUTER_INNER_HEIGHT_DIFF_91,
444 
445     /**
446      * Difference of window.outer/inner height is 93.
447      */
448     @BrowserFeature(FF)
449     JS_WINDOW_OUTER_INNER_HEIGHT_DIFF_93,
450 
451     /** Window.getSelection returns null, if the window is not visible. */
452     @BrowserFeature({FF, FF_ESR})
453     JS_WINDOW_SELECTION_NULL_IF_INVISIBLE,
454 
455     /** {@code XSLTProcessor.transformToDocument} supports output indent attribute. */
456     @BrowserFeature({CHROME, EDGE})
457     JS_XSLT_TRANSFORM_INDENT,
458 
459     /** With special keys [in .type(int)], should we trigger onkeypress event or not. */
460     @BrowserFeature({FF, FF_ESR})
461     KEYBOARD_EVENT_SPECIAL_KEYPRESS,
462 
463     /** XMLHttpRequest.getAllResponseHeaders() uses only Lf as separator. */
464     @BrowserFeature({FF, FF_ESR})
465     XHR_ALL_RESPONSE_HEADERS_SEPARATE_BY_LF,
466 
467     /**
468      * Indicates that the Browser handles async and sync network errors the same way.
469      */
470     @BrowserFeature({FF, FF_ESR})
471     XHR_HANDLE_SYNC_NETWORK_ERRORS,
472 
473     /** XMLHttpRequest triggers the load events also if the abort was signaled. */
474     @BrowserFeature({FF, FF_ESR})
475     XHR_LOAD_ALWAYS_AFTER_DONE,
476 
477     /** If state unsent the response text is empty even if the response type is wrong. */
478     @BrowserFeature({FF, FF_ESR})
479     XHR_RESPONSE_TEXT_EMPTY_UNSENT,
480 
481     /** Indicates if the XMLHttpRequest.send() method will throw if aborted. */
482     @BrowserFeature({CHROME, EDGE})
483     XHR_SEND_NETWORK_ERROR_IF_ABORTED,
484 }