49

What special characters are allowed for file names on Android?

~!@#$%^&*()_+/\.,

Also, can I save file with Unicode name?

naXa
  • 26,677
  • 15
  • 154
  • 213
alex2k8
  • 39,732
  • 56
  • 158
  • 214
  • 2
    Do you mean on the SD card (or equivalent) ? – Nicolas Raoul Apr 21 '10 at 02:21
  • 1
    https://superuser.com/a/693819/687273 –  Jul 23 '17 at 11:26
  • 1
    On Unix (and thus Android), it’s simply `NULL` (`0x00`) and `/` which are invalid. For interoperability, though, you may be interested in the Windows list from the answers below. – caw Aug 04 '18 at 19:10

8 Answers8

38
  1. On Android (at least by default) the file names encoded as UTF-8.

  2. Looks like reserved file name characters depend on filesystem mounted (http://en.wikipedia.org/wiki/Filename).

I considered as reserved:

private static final String ReservedChars = "|\\?*<\":>+[]/'";
alex2k8
  • 39,732
  • 56
  • 158
  • 214
  • 10
    '+[] are not reserved – xmen Jan 04 '15 at 09:01
  • 3
    Without `'+[]` (as @xmen noted), this is actually the *Windows* set. But it describes the invalid *printable* characters only. The control characters `0x00-0x1f` and `0x7f` are also invalid on Windows. For interoperability, all this may be useful. But on Unix (and thus Android) alone, the only invalid characters are `NULL` (`0x00`) and `/`. – caw Aug 04 '18 at 19:08
  • 1
    Came here researching firefox downloads on android. For some reason firefox thinks the plus sign makes for an invalid filename. – GGets Oct 09 '18 at 05:15
10

According to wiki and assuming that you are using external data storage which has FAT32.

Allowable characters in directory entries

are

Any byte except for values 0-31, 127 (DEL) and: " * / : < > ? \ | + , . ; = [] (lowcase a-z are stored as A-Z). With VFAT LFN any Unicode except NUL

kreker
  • 5,453
  • 4
  • 36
  • 32
  • 1
    "; , . = " are allowed for file names in android – fire in the hole Aug 09 '14 at 13:02
  • Is it true on both external and internal storage? Are all of the rest of the characters allowed? Do lowercard=uppercase as the text is showing here (meaning I can't have "Hello.txt" and "hello.txt" on the same folder, for example) ? – android developer Apr 20 '15 at 23:33
7
final String[] ReservedChars = {"|", "\\", "?", "*", "<", "\"", ":", ">"};

for(String c :ReservedChars){
    System.out.println(dd.indexOf(c));
    dd.indexOf(c);
}
Mat
  • 188,820
  • 38
  • 367
  • 383
Blackbern
  • 257
  • 3
  • 4
5

This is correct InputFilter for File Names in Android:

    InputFilter filter = new InputFilter()
    {
        public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend) 
        { 
            if (source.length() < 1) return null;
            char last = source.charAt(source.length() - 1);
            String reservedChars = "?:\"*|/\\<>";
            if(reservedChars.indexOf(last) > -1) return source.subSequence(0, source.length() - 1);
            return null;
        }  
    };
Quark
  • 1,349
  • 2
  • 18
  • 30
  • Good idea, poor implementation. As it is it will only filter properly when entering characters one by one. If you paste something in chances are it won't be filtered properly. – Slion Jan 10 '21 at 17:25
5

From android.os.FileUtils

    private static boolean isValidFatFilenameChar(char c) {
        if ((0x00 <= c && c <= 0x1f)) {
            return false;
        }
        switch (c) {
            case '"':
            case '*':
            case '/':
            case ':':
            case '<':
            case '>':
            case '?':
            case '\\':
            case '|':
            case 0x7F:
                return false;
            default:
                return true;
        }
    }
    private static boolean isValidExtFilenameChar(char c) {
        switch (c) {
            case '\0':
            case '/':
                return false;
            default:
                return true;
        }
    }

Note: FileUtils are hidden APIs (not for apps; for AOSP usage). Use as a reference (or by reflection at one's own risk)

minsk
  • 515
  • 1
  • 6
  • 15
2

I tested this quickly on my Galaxy Note 8 on Android 4.4.2. The default My Files app helpfully greys out invalid characters which are as follows:

? : " * | / \ < >

I put all the other special chars available into a filename and it saved. This may not be consistent across all Android versions so maybe it's best to be conservative and replace them with similarly meaningful characters.

TopherC
  • 69
  • 3
1

This is clearly filesystem and Android operating system dependent. On my oneplus/oxygenOS, the only characters in the accepted answer

private static final String ReservedChars = "|\\?*<\":>+[]/'";

that I could not use to rename a file were / and *

However, Android wide, the list above would seem to be sensible.

marzetti
  • 101
  • 1
  • 6
1

On Android as suggested there you can use an input filter to prevent user entering invalid characters, here is a better implementation of it:

/**
 * An input filter which can be attached to an EditText widget to filter out invalid filename characters
 */
class FileNameInputFilter: InputFilter
{
override fun filter(source: CharSequence?, start: Int, end: Int, dest: Spanned?, dstart: Int, dend: Int): CharSequence? {
    if (source.isNullOrBlank()) {
        return null
    }

    val reservedChars = "?:\"*|/\\<>\u0000"
    // Extract actual source
    val actualSource = source.subSequence(start, end)
    // Filter out unsupported characters
    val filtered = actualSource.filter { c -> reservedChars.indexOf(c) == -1 }
    // Check if something was filtered out
    return if (actualSource.length != filtered.length) {
        // Something was caught by our filter, provide visual feedback
            if (actualSource.length - filtered.length == 1) {
                // A single character was removed
                BrowserApp.instance.applicationContext.toast(R.string.invalid_character_removed)
            } else {
                // Multiple characters were removed                    
     BrowserApp.instance.applicationContext.toast(R.string.invalid_characters_removed)
                }
            // Provide filtered results then
            filtered
        } else {
            // Nothing was caught in our filter
            null
        }
    }
}
Slion
  • 1,407
  • 16
  • 19