package net.knarcraft.ffmpegconverter.utility;

import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;

/**
 * A class which helps with file handling
 */
public final class FileUtil {

    private FileUtil() {

    }

    /**
     * Gets the path described by the input, but changed to account for collisions
     *
     * @param folder       <p>The folder containing the output file.</p>
     * @param file         <p>The input file.</p>
     * @param outExtension <p>The extension of the output file.</p>
     * @return <p>A file name with the new extension and without any collisions.</p>
     */
    public static String getNonCollidingPath(File folder, File file, String outExtension) {
        return FileUtil.getNonCollidingFilename(folder.getAbsolutePath() + File.separator +
                FileUtil.stripExtension(file.getName()) + "." + outExtension, outExtension);
    }

    /**
     * Recursively lists all files in a folder
     *
     * @param folder        <p>The folder to start from.</p>
     * @param maxRecursions <p>Maximum number of recursions</p>
     * @return A list of files
     */
    public static File[] listFilesRecursive(File folder, String[] extensions, int maxRecursions) {
        //Return if the target depth has been reached
        if (maxRecursions == 0) {
            return null;
        }
        //Get a list of all files which are folders and has one of the extensions specified
        File[] foundFiles = folder.listFiles((file) -> file.isFile() &&
                ListUtil.listContains(extensions, (item) -> file.getName().toLowerCase().endsWith(item)));
        //Return if recursion is finished
        if (maxRecursions == 1) {
            return foundFiles;
        }
        //Get all folders in the directory
        File[] subFolders = folder.listFiles((dir, name) -> new File(dir, name).isDirectory());
        //Return if the folder has no sub folders
        if (subFolders == null) {
            return foundFiles;
        }
        for (File subFolder : subFolders) {
            //Get all relevant files contained within the sub folder
            File[] nextLevel = listFilesRecursive(subFolder, extensions, maxRecursions - 1);
            //Add found files to the output
            if (nextLevel != null) {
                if (foundFiles == null) {
                    foundFiles = nextLevel;
                } else {
                    foundFiles = ListUtil.concatenate(foundFiles, nextLevel);
                }
            }
        }
        return foundFiles;
    }

    /**
     * Reads a file's contents to a string list
     *
     * <p>The file must contain the number of lines to read in the first line.</p>
     *
     * @param fileName <p>The file to read.</p>
     * @return <p>A string list where each element is one line of the file.</p>
     * @throws IOException <p>If the file cannot be read.</p>
     */
    public static String[] readFileLines(String fileName) throws IOException {
        BufferedReader reader = new BufferedReader(new InputStreamReader(getResourceAsStream(fileName)));
        int numberOfLines = Integer.parseInt(reader.readLine());
        String[] lines = new String[numberOfLines];
        for (int i = 0; i < lines.length; i++) {
            lines[i] = reader.readLine();
        }
        return lines;
    }

    /**
     * Gets a resource as an InputStream
     *
     * @param resourceName <p>The name of the resource you want to read.</p>
     * @return <p>An input stream which can be used to access the resource.</p>
     */
    private static InputStream getResourceAsStream(String resourceName) {
        ClassLoader classloader = Thread.currentThread().getContextClassLoader();
        return classloader.getResourceAsStream(resourceName);
    }

    /**
     * Adds parentheses with an integer if the output file already exists
     *
     * @param targetPath <p>The path the file should ideally be saved at.</p>
     * @param extension  <p>The extension of the target file.</p>
     * @return <p>A filename guaranteed not to collide with other files.</p>
     */
    private static String getNonCollidingFilename(String targetPath, String extension) {
        File newFile = new File(targetPath);
        String fileName = stripExtension(targetPath).replaceAll("\\([0-9]+\\)$", "");
        int i = 1;
        while (newFile.exists()) {
            newFile = new File(fileName + "(" + i++ + ")" + "." + extension);
        }
        return newFile.toString();
    }

    /**
     * Gets the extension of the given filename
     *
     * @param file <p>The filename to check</p>
     * @return <p>The file's extension</p>
     */
    public static String getExtension(String file) {
        if (file.contains(".")) {
            return file.substring(file.lastIndexOf('.') + 1);
        } else {
            return "";
        }
    }

    /**
     * Removes the extension from a file name
     *
     * @param file <p>A filename.</p>
     * @return <p>A filename without its extension.</p>
     */
    public static String stripExtension(String file) {
        return file.substring(0, file.lastIndexOf('.'));
    }

    /**
     * Gets the locale specifying the language of the given file name
     *
     * @param fileName <p>The file name to check</p>
     * @return <p>The locale, or null if no locale could be parsed</p>
     */
    public static @Nullable String getLanguage(@NotNull String fileName) {
        fileName = stripExtension(fileName);

        String possibleLanguage = getExtension(fileName);
        // NRK Nett-TV has a tendency to use nb-ttv for Norwegian for some reason 
        possibleLanguage = possibleLanguage.replace("nb-ttv", "nb-nor");

        // TODO: Some languages are specified by using "-en" or "-English" or ".en" or ".English" at the end of file names
        if (possibleLanguage.length() <= 1 || (possibleLanguage.length() >= 4 &&
                (!possibleLanguage.contains("-") || possibleLanguage.length() >= 8))) {
            return null;
        }

        // Hope the text is an actual valid language
        if (!possibleLanguage.contains("-")) {
            return possibleLanguage;
        }

        // Make sure the "-" has at least two characters on each side
        String[] parts = possibleLanguage.split("-");
        if (parts[0].length() < 2 || parts[1].length() < 2) {
            return null;
        }

        if (parts[1].length() == 3) {
            // Return three-letter country code
            return parts[1].toLowerCase();
        } else {
            // Return en-US country code
            return parts[0].substring(0, 2).toLowerCase() + "-" + parts[1].substring(0, 2).toUpperCase();
        }
    }

}