|
1 | 1 | /*
|
2 |
| - * Copyright (c) 2007, 2021, Oracle and/or its affiliates. All rights reserved. |
| 2 | + * Copyright (c) 2007, 2022, 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
|
|
31 | 31 | import java.nio.file.spi.FileSystemProvider;
|
32 | 32 | import java.util.Iterator;
|
33 | 33 | import java.util.NoSuchElementException;
|
| 34 | +import java.util.Objects; |
34 | 35 |
|
35 | 36 | /**
|
36 | 37 | * An object that may be used to locate a file in a file system. It will
|
|
49 | 50 | * file system. {@code Path} defines the {@link #getFileName() getFileName},
|
50 | 51 | * {@link #getParent getParent}, {@link #getRoot getRoot}, and {@link #subpath
|
51 | 52 | * subpath} methods to access the path components or a subsequence of its name
|
52 |
| - * elements. |
| 53 | + * elements, and {@link #getExtension() getExtension} to obtain its extension. |
53 | 54 | *
|
54 | 55 | * <p> In addition to accessing the components of a path, a {@code Path} also
|
55 | 56 | * defines the {@link #resolve(Path) resolve} and {@link #resolveSibling(Path)
|
@@ -248,6 +249,63 @@ public static Path of(URI uri) {
|
248 | 249 | */
|
249 | 250 | Path getFileName();
|
250 | 251 |
|
| 252 | + /** |
| 253 | + * Returns the file extension of this path's file name as a {@code String}. |
| 254 | + * The extension is derived from this {@code Path} by obtaining the |
| 255 | + * {@linkplain #getFileName file name element}, deriving its {@linkplain |
| 256 | + * #toString string representation}, and then extracting a substring |
| 257 | + * determined by the position of a period character ('.', U+002E FULL STOP) |
| 258 | + * within the file name string. If the file name element is {@code null}, |
| 259 | + * or if the file name string does not contain a period character, or if |
| 260 | + * the only period in the file name string is its first character, then |
| 261 | + * the extension is {@code null}. Otherwise, the extension is the substring |
| 262 | + * after the last period in the file name string. If this last period is |
| 263 | + * also the last character in the file name string, then the extension is |
| 264 | + * {@linkplain String#isEmpty empty}. |
| 265 | + * |
| 266 | + * @implSpec |
| 267 | + * The default implementation is equivalent for this path to: |
| 268 | + * <pre>{@code |
| 269 | + * int lastPeriod = fileName.lastIndexOf('.'); |
| 270 | + * if (lastPeriod <= 0) |
| 271 | + * return null; |
| 272 | + * return (lastPeriod == fileName.length() - 1) |
| 273 | + * ? "" |
| 274 | + * : fileName.substring(lastPeriod + 1); |
| 275 | + * }</pre> |
| 276 | + * |
| 277 | + * @return the file name extension of this path, which might be the |
| 278 | + * empty string, or {@code null} if no extension is found |
| 279 | + * |
| 280 | + * @since 20 |
| 281 | + */ |
| 282 | + default String getExtension() { |
| 283 | + Path fileName = getFileName(); |
| 284 | + if (fileName == null) |
| 285 | + return null; |
| 286 | + |
| 287 | + String fileNameString = fileName.toString(); |
| 288 | + int length = fileNameString.length(); |
| 289 | + |
| 290 | + // An empty or unity length file name string has a null extension |
| 291 | + if (length > 1) { |
| 292 | + int lastPeriodIndex = fileNameString.lastIndexOf('.'); |
| 293 | + |
| 294 | + // Indeterminate if there is no period character or |
| 295 | + // only the first character is a period character |
| 296 | + if (lastPeriodIndex > 0) { |
| 297 | + if (lastPeriodIndex == length - 1) { |
| 298 | + // empty string |
| 299 | + return ""; |
| 300 | + } else { |
| 301 | + return fileNameString.substring(lastPeriodIndex + 1); |
| 302 | + } |
| 303 | + } |
| 304 | + } |
| 305 | + |
| 306 | + return null; |
| 307 | + } |
| 308 | + |
251 | 309 | /**
|
252 | 310 | * Returns the <em>parent path</em>, or {@code null} if this path does not
|
253 | 311 | * have a parent.
|
|
0 commit comments