All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.android.ide.common.res2.ValueXmlHelper Maven / Gradle / Ivy

There is a newer version: 25.3.0
Show newest version
/*
 * Copyright (C) 2013 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.android.ide.common.res2;

import static com.android.SdkConstants.AMP_ENTITY;
import static com.android.SdkConstants.APOS_ENTITY;
import static com.android.SdkConstants.GT_ENTITY;
import static com.android.SdkConstants.LT_ENTITY;
import static com.android.SdkConstants.QUOT_ENTITY;

import com.android.annotations.Nullable;
import com.android.annotations.VisibleForTesting;

/**
 * Helper class to help with XML values resource file.
 */
public class ValueXmlHelper {

    /**
     * Replaces escapes in an XML resource string with the actual characters,
     * performing unicode substitutions (replacing any {@code \\uNNNN} references in the
     * given string with the corresponding unicode characters), etc.
     *
     * @param s the string to unescape
     * @param escapeEntities XML entities
     * @param trim whether surrounding space and quotes should be trimmed
     * @return the string with the escape characters removed and expanded
     */
    @SuppressWarnings("UnnecessaryContinue")
    @Nullable
    public static String unescapeResourceString(
            @Nullable String s,
            boolean escapeEntities, boolean trim) {
        if (s == null) {
            return null;
        }

        // Trim space surrounding optional quotes
        int i = 0;
        int n = s.length();
        boolean quoted = false;
        if (trim) {
            while (i < n) {
                char c = s.charAt(i);
                if (!Character.isWhitespace(c)) {
                    break;
                }
                i++;
            }
            while (n > i) {
                char c = s.charAt(n - 1);
                if (!Character.isWhitespace(c)) {
                    //See if this was a \, and if so, see whether it was escaped
                    if (n < s.length() && isEscaped(s, n)) {
                        n++;
                    }
                    break;
                }
                n--;
            }

            // Trim surrounding quotes. Note that there can be *any* number of these, and
            // the left side and right side do not have to match; e.g. you can have
            //    """"f"" => f
            int quoteEnd = n;
            while (i < n) {
                char c = s.charAt(i);
                if (c != '"') {
                    break;
                }
                quoted = true;
                i++;
            }
            // Searching backwards is slightly more complicated; make sure we don't trim
            // quotes that have been escaped.
            if (quoted) {
                while (n > i) {
                    char c = s.charAt(n - 1);
                    if (c != '"') {
                        if (n < s.length() && isEscaped(s, n)) {
                            n++;
                        }
                        break;
                    }
                    n--;
                }
            }
            if (n == i) {
                return ""; //$NON-NLS-1$
            }

            // Only trim leading spaces if we didn't already process a leading quote:
            if (!quoted) {
                while (i < n) {
                    char c = s.charAt(i);
                    if (!Character.isWhitespace(c)) {
                        break;
                    }
                    i++;
                }

                // Only trim trailing spaces if we didn't already process a trailing quote:
                if (n == quoteEnd) {
                    while (n > i) {
                        char c = s.charAt(n - 1);
                        if (!Character.isWhitespace(c)) {
                            //See if this was a \, and if so, see whether it was escaped
                            if (n < s.length() && isEscaped(s, n)) {
                                n++;
                            }
                            break;
                        }
                        n--;
                    }
                }
                if (n == i) {
                    return ""; //$NON-NLS-1$
                }
            }
        }

        // Perform a single pass over the string and see if it contains
        // (1) spaces that should be converted (e.g. repeated spaces or a newline which
        // should be converted to a space)
        // (2) escape characters (\ and &) which will require expansions
        // If we find neither of these, we can simply return the string
        boolean rewriteWhitespace = false;
        if (!quoted) {
            // See if we need to fold adjacent spaces
            boolean prevSpace = false;
            boolean hasEscape = false;
            for (int curr = i; curr < n; curr++) {
                char c = s.charAt(curr);
                if (c == '\\' || c == '&') {
                    hasEscape = true;
                }
                boolean isSpace = Character.isWhitespace(c);
                if (isSpace && prevSpace) {
                    // fold adjacent spaces
                    rewriteWhitespace = true;
                } else if (c == '\n') {
                    // rewrite newlines as spaces
                    rewriteWhitespace = true;
                }
                prevSpace = isSpace;
            }

            if (!trim) {
                rewriteWhitespace = false;
            }

            // If no surrounding whitespace and no escape characters, no need to do any
            // more work
            if (!rewriteWhitespace && !hasEscape && i == 0 && n == s.length()) {
                return s;
            }
        }

        StringBuilder sb = new StringBuilder(n - i);
        boolean prevSpace = false;
        for (; i < n; i++) {
            char c = s.charAt(i);
            if (c == '\\' && i < n - 1) {
                prevSpace = false;
                char next = s.charAt(i + 1);
                // Unicode escapes
                if (next == 'u' && i < n - 5) { // case sensitive
                    String hex = s.substring(i + 2, i + 6);
                    try {
                        int unicodeValue = Integer.parseInt(hex, 16);
                        sb.append((char) unicodeValue);
                        i += 5;
                        continue;
                    } catch (NumberFormatException e) {
                        // Invalid escape: Just proceed to literally transcribe it
                        sb.append(c);
                    }
                } else if (next == 'n') {
                    sb.append('\n');
                    i++;
                    continue;
                } else if (next == 't') {
                    sb.append('\t');
                    i++;
                    continue;
                } else {
                    sb.append(next);
                    i++;
                    continue;
                }
            } else {
                if (c == '&' && escapeEntities) {
                    prevSpace = false;
                    if (s.regionMatches(true, i, LT_ENTITY, 0, LT_ENTITY.length())) {
                        sb.append('<');
                        i += LT_ENTITY.length() - 1;
                        continue;
                    } else if (s.regionMatches(true, i, AMP_ENTITY, 0, AMP_ENTITY.length())) {
                        sb.append('&');
                        i += AMP_ENTITY.length() - 1;
                        continue;
                    } else if (s.regionMatches(true, i, QUOT_ENTITY, 0, QUOT_ENTITY.length())) {
                      sb.append('"');
                      i += QUOT_ENTITY.length() - 1;
                      continue;
                    } else if (s.regionMatches(true, i, APOS_ENTITY, 0, APOS_ENTITY.length())) {
                      sb.append('\'');
                      i += APOS_ENTITY.length() - 1;
                      continue;
                    } else if (s.regionMatches(true, i, GT_ENTITY, 0, GT_ENTITY.length())) {
                      sb.append('>');
                      i += GT_ENTITY.length() - 1;
                      continue;
                    } else if (i < n - 2 && s.charAt(i + 1) == '#') {
                        int end = s.indexOf(';', i + 1);
                        if (end != -1) {
                            char first = s.charAt(i + 2);
                            boolean hex = first == 'x' || first == 'X';
                            String number = s.substring(i + (hex ? 3 : 2), end);
                            try {
                                int unicodeValue = Integer.parseInt(number, hex ? 16 : 10);
                                sb.append((char) unicodeValue);
                                i = end;
                                continue;
                            } catch (NumberFormatException e) {
                                // Invalid escape: Just proceed to literally transcribe it
                                sb.append(c);
                            }
                        } else {
                            // Invalid escape: Just proceed to literally transcribe it
                            sb.append(c);
                        }
                    }
                }

                if (rewriteWhitespace) {
                    boolean isSpace = Character.isWhitespace(c);
                    if (isSpace) {
                        if (!prevSpace) {
                            sb.append(' '); // replace newlines etc with a plain space
                        }
                    } else {
                        sb.append(c);
                    }
                    prevSpace = isSpace;
                } else {
                    sb.append(c);
                }
            }
        }
        s = sb.toString();

        return s;
    }

    /**
     * Returns true if the character at the given offset in the string is escaped
     * (the previous character is a \, and that character isn't itself an escaped \)
     *
     * @param s the string
     * @param index the index of the character in the string to check
     * @return true if the character is escaped
     */
    @VisibleForTesting
    static boolean isEscaped(String s, int index) {
        if (index == 0 || index == s.length()) {
            return false;
        }
        int prevPos = index - 1;
        char prev = s.charAt(prevPos);
        if (prev != '\\') {
            return false;
        }
        // The character *may* be escaped; not sure if the \ we ran into is
        // an escape character, or an escaped backslash; we have to search backwards
        // to be certain.
        int j = prevPos - 1;
        while (j >= 0) {
            if (s.charAt(j) != '\\') {
                break;
            }
            j--;
        }
        // If we passed an odd number of \'s, the space is escaped
        return (prevPos - j) % 2 == 1;
    }

    /**
     * Escape a string value to be placed in a string resource file such that it complies with
     * the escaping rules described here:
     *   http://developer.android.com/guide/topics/resources/string-resource.html
     * More examples of the escaping rules can be found here:
     *   http://androidcookbook.com/Recipe.seam?recipeId=2219&recipeFrom=ViewTOC
     * This method assumes that the String is not escaped already.
     *
     * Rules:
     * 
    *
  • Double quotes are needed if string starts or ends with at least one space. *
  • {@code @, ?} at beginning of string have to be escaped with a backslash. *
  • {@code ', ", \} have to be escaped with a backslash. *
  • {@code <, >, &} have to be replaced by their predefined xml entity. *
  • {@code \n, \t} have to be replaced by a backslash and the appropriate character. *
* @param s the string to be escaped * @return the escaped string as it would appear in the XML text in a values file */ public static String escapeResourceString(String s) { return escapeResourceString(s, true); } /** * Escape a string value to be placed in a string resource file such that it complies with * the escaping rules described here: * http://developer.android.com/guide/topics/resources/string-resource.html * More examples of the escaping rules can be found here: * http://androidcookbook.com/Recipe.seam?recipeId=2219&recipeFrom=ViewTOC * This method assumes that the String is not escaped already. * * Rules: *
    *
  • Double quotes are needed if string starts or ends with at least one space. *
  • {@code @, ?} at beginning of string have to be escaped with a backslash. *
  • {@code ', ", \} have to be escaped with a backslash. *
  • {@code <, >, &} have to be replaced by their predefined xml entity. *
  • {@code \n, \t} have to be replaced by a backslash and the appropriate character. *
* @param s the string to be escaped * @param escapeXml whether XML characters like {@code <} and {@code &} should be * escaped; this should normally be true, but is optional since * some callers need to pass the string over to code which already escapes * XML content, and you wouldn't want the ampersand in the escaped character * entity to itself be escaped. * @return the escaped string as it would appear in the XML text in a values file */ public static String escapeResourceString(String s, boolean escapeXml) { int n = s.length(); if (n == 0) { return ""; } StringBuilder sb = new StringBuilder(s.length() * 2); boolean hasSpace = s.charAt(0) == ' ' || s.charAt(n - 1) == ' '; if (hasSpace) { sb.append('"'); } else if (s.charAt(0) == '@' || s.charAt(0) == '?') { sb.append('\\'); } for (int i = 0; i < n; ++i) { char c = s.charAt(i); switch (c) { case '\'': if (!hasSpace) { sb.append('\\'); } sb.append(c); break; case '"': case '\\': sb.append('\\'); sb.append(c); break; case '<': if (escapeXml) { sb.append(LT_ENTITY); } else { sb.append(c); } break; case '&': if (escapeXml) { sb.append(AMP_ENTITY); } else { sb.append(c); } break; case '\n': sb.append("\\n"); //$NON-NLS-1$ break; case '\t': sb.append("\\t"); //$NON-NLS-1$ break; default: sb.append(c); break; } } if (hasSpace) { sb.append('"'); } return sb.toString(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy