#include "inc/mc.h"
#include "inc/zslib.h"
#include "inc/mui.h"
#include "inc/encode.h"
#include "inc/popup.h"

const char zip_ext[] = "zip";
const char jar_ext[] = "jar";
const char sdt_ext[] = "sdt";

int PassResult;
char pass[32];
int pass_gui;
void _WaitPass(WSHDR *wsname)
{
    ws2char8(wsname, pass, 32, 1);
    PassResult = 1;
}

int ShowWaitPass(ARCHINFO* pzi)
{
    PassResult = -1;
    pass_gui = TextInput(muitxt(ind_pass), muitxt(ind_pass), 0, NULL, _WaitPass);
    CSM_RAM *csm;

    do
    {
        FindGUIbyId(pass_gui, &csm);
    }
    while(csm!=NULL);

    pzi->password = malloc(32);
    strcpy(pzi->password, pass);

    if(strlen(pzi->password)== 0 || PassResult!=1)
        PassResult = 2;
    return PassResult;
}

void ZipError(int zipErrNum, char* procName)
{
    sprintf(msgbuf, muitxt(ind_err_ziperr), zipErrNum, procName);
    DoShowMsg(msgbuf);
}

void CloseZip(ARCHINFO* pzi)
{
    if (pzi->info.zip.uf)
    {
        unzClose(pzi->info.zip.uf);
        FreeZipInfo(pzi);
    }
}

void CloseTabZip(int tab)
{
    CloseZip(tabs[tab]->zipInfo);
}

int IsZipOpened(int tab)
{
    return (tabs[tab]->zipInfo->info.zip.uf != NULL);
}

int IsInZip()
{
    return IsZipOpened(curtab);
}

void FreeZipInfo(ARCHINFO* pzi)
{
    if (pzi)
    {
        if (pzi->info.zip.pfi)
        {
            for (uint32_t i=0; i < pzi->info.zip.gi.number_entry; ++i)
                if (pzi->info.zip.pfi[i]) mfree(pzi->info.zip.pfi[i]);
            mfree(pzi->info.zip.pfi);
        }

        if (pzi->pszNames)
        {
            for (uint32_t i=0; i < pzi->info.zip.gi.number_entry; ++i)
                if (pzi->pszNames[i]) mfree(pzi->pszNames[i]);
            mfree(pzi->pszNames);
        }

        if (pzi->password) mfree(pzi->password);

        zeromem(pzi, sizeof(ARCHINFO));
    }
}

// Считываем и сохраняем информацию о содержимом архива в память
int ReadZipInfo(ARCHINFO* pzi)
{
    int zerr = UNZ_OK;

    if (pzi->info.zip.uf)
    {
        zerr = unzGetGlobalInfo(pzi->info.zip.uf, &pzi->info.zip.gi);
        if (zerr == UNZ_OK)
        {
            if (pzi->info.zip.gi.number_entry > 0)
            {
                int useProgressBar = !progr_start;
                if (useProgressBar)
                {
                    // запускаем прогресс бар, если уже не запущен другой
                    Busy = 1;
                    initprogr(ind_msg_zreading);
                    progr_max = pzi->info.zip.gi.number_entry;
                    incprogr(0);
                    incfname(strrchr(pzi->szZipPath,'\\')+1);
                }

                int allocsize = pzi->info.zip.gi.number_entry * sizeof(char*);
                pzi->pszNames = malloc(allocsize);
                if (pzi->pszNames)
                {
                    zeromem(pzi->pszNames, allocsize);
                    pzi->info.zip.pfi = malloc(allocsize);
                    if (pzi->info.zip.pfi) zeromem(pzi->info.zip.pfi, allocsize);
                }

                if (pzi->pszNames && pzi->info.zip.pfi)
                {
                    zerr = unzGoToFirstFile(pzi->info.zip.uf);

                    if (zerr == UNZ_OK)
                    {
                        uint32_t i=0;
                        while(i < pzi->info.zip.gi.number_entry && zerr == UNZ_OK && !progr_stop)
                        {
                            if (!msgopen)
                            {
                                pzi->pszNames[i] = malloc(MAX_PATH_INZIP);
                                pzi->info.zip.pfi[i] = malloc(sizeof(unz_file_info));
                                if (pzi->pszNames[i] != NULL && pzi->info.zip.pfi[i] != NULL)
                                {
                                    zerr = unzGetCurrentFileInfo(pzi->info.zip.uf, pzi->info.zip.pfi[i], pzi->pszNames[i], MAX_PATH_INZIP, NULL, 0, NULL, 0);
                                    if (zerr == UNZ_OK)
                                    {
                                        // Фиксим слеши
                                        strreplace(pzi->pszNames[i], '/', '\\');

                                        // Переводим
                                        dos2utf8(pzi->pszNames[i]);

                                        // Переходим к следующему файлу
                                        if ((i+1) < pzi->info.zip.gi.number_entry)
                                        {
                                            zerr = unzGoToNextFile(pzi->info.zip.uf);
                                            if (zerr != UNZ_OK)
                                                ZipError(zerr, "unzGoToNextFile");
                                        }
                                    }
                                    else ZipError(zerr, "unzGetCurrentFileInfo");
                                }
                                else ZipError(zerr = UNZ_ERRNO, "no memory for fileinfo");
                                if (useProgressBar && (i%100)==0)
                                    incprogrset(i);
                                i++;
                            }
                        }
                    }
                    else ZipError(zerr, "unzGoToFirstFile");
                }
                else ZipError(zerr = UNZ_ERRNO, "no memory");

                if (useProgressBar)
                {
                    endprogr(); // останавливаем прогресс бар
                    Busy = 0;
                }
            }
        }
        else ZipError(zerr, "unzGetGlobalInfo");
    }

    if (progr_stop) zerr = -11111;

    return zerr;
}

int OpenZip(ARCHINFO* pzi, char* zipFileName)
{
    int zerr = UNZ_OK;

    unzFile uf = unzOpen(zipFileName);
    if (uf != NULL)
    {
        pzi->info.zip.uf = uf;
        pzi->szCurDir[0] = '\0';
        pzi->type = ZIP_ARCH;
        strcpy(pzi->szZipPath, zipFileName);
    }
    else zerr = UNZ_ERRNO;

    if (zerr == UNZ_OK)
        zerr = ReadZipInfo(pzi);

    if (zerr != UNZ_OK)
        CloseZip(pzi);

    return zerr;
}

int OpenTabZip(int tab, char* zipFileName)
{
    int zerr = UNZ_ERRNO;

    if (!IsArchiveOpened(tab)) // не обрабатываем вложенные архивы
        zerr = OpenZip(tabs[tab]->zipInfo, zipFileName);

    return zerr;
}

void strcharplase(char* s, char cprev)
{
    for(int i = 0; s[i]; s++)
    {
        if(s[i]==cprev)
        {
            s[i]=0;
            break;
        }
    }
}

int FILEINFO_inlist(int tab, char *name, int num)
{
    FILEINF* file = FileListBase[tab];

    if(file)
    {
        for(int i=0; i<=num; i++)
        {
            if (!strcmp(file->sname, name))
                return 1;
            file = file->next;
        }
    }

    return 0;
}

int FillZipFiles(int tab, char* subdname)
{
    int num = 0;
    char buf[MAX_PATH_INZIP];
    char* pszDirName = subdname[0] == '\\' ? subdname + 1 : subdname; // Пропустим в начале обратный слеш
    int dirLen = strlen(pszDirName);
    ARCHINFO* pzi = tabs[tab]->zipInfo;

    if (pzi->info.zip.uf)
    {
        for (uint32_t i=0; i < pzi->info.zip.gi.number_entry; ++i)
        {
            int ignore = 1;
            strcpy(buf, pzi->pszNames[i]);
            char* pFileNameStart = &buf[0];
            unz_file_info* pfi = pzi->info.zip.pfi[i];

            if (dirLen == 0)
            {
                // Если ищем корневые элементы
                // То добавляем все файлы у которых нет слешей '\'
                char* slashPos = strstr(buf, "\\");
                ignore = (slashPos != NULL);
                if (ignore)
                {
                    // И добавляем все директории - у них один слеш, и тот последний
                    ignore = (*(slashPos + 1) != 0);
                    if (!ignore)
                        // Затрем этот слеш, чтобы не выводился
                        *slashPos = 0;
                }
            }
            else
            {
                // Сравниваем начало, если равно и потом слеш - то это элементы нашей директории
                pFileNameStart = pFileNameStart + dirLen;
                ignore = (strnicmp(buf, pszDirName, dirLen) != 0 || *pFileNameStart != '\\');
                if (!ignore)
                {
                    pFileNameStart = pFileNameStart + 1; // пропустим слеш после имени директории

                    // игнорим саму текущую, директорию
                    ignore = (*pFileNameStart == 0);
                    if (!ignore)
                    {
                        // проверка на поддиректорию
                        char* pFileNameEnd = strstr(pFileNameStart, "\\");
                        if (pFileNameEnd != NULL)
                        {
                            // проверка на то что это не сама поддиректория а ее элементы, оные проигнорим
                            ignore = (*(pFileNameEnd+1) != 0);
                            if (!ignore)
                                *pFileNameEnd = 0; // грохнем слеш в конце
                        }
                    }
                }
            }

            if (!ignore)
            {
                if(!FILEINFO_inlist(tab, pFileNameStart, num))
                {
                    AddFile(tab, i, pFileNameStart, pfi->uncompressed_size, pfi->external_fa, pfi->dosDate,
                            pfi->compressed_size, (pfi->external_fa & FA_DIRECTORY) ? TYPE_ZIP_DIR : TYPE_ZIP_FILE, subdname);
                    num++;
                }
            }

            if(ignore)
            {
                if (dirLen == 0)
                {
                    ignore = 0;
                    strcpy(pFileNameStart, buf);
                    strcharplase(pFileNameStart, '\\');
                }
                else
                {
                    strcharplase(pFileNameStart, '\\');
                    ignore = (strnicmp(buf, pszDirName, dirLen) != 0);
                    if (!ignore)
                    {
                        // проверка на поддиректорию
                        char* pFileNameEnd = pFileNameStart+dirLen;
                        if (pFileNameEnd != 0)
                            // проверка на то что это не сама поддиректория а ее элементы, оные проигнорим
                            ignore = (*(pFileNameEnd+1) != '\\');
                    }
                }

                if(!ignore &&  strlen(pFileNameStart))
                {
                    if(!FILEINFO_inlist(tab, pFileNameStart, num))
                    {
                        AddFile(tab, num, pFileNameStart, 0, FA_DIRECTORY, 0, 0, TYPE_ZIP_DIR, subdname);
                        num++;
                    }
                }
            }

        }
    }

    return num;
}

int SetCurrentFileInZip(ARCHINFO* pzi, int id)
{
    int zerr = unzGoToFirstFile(pzi->info.zip.uf);
    if (zerr != UNZ_OK)
        return 1;

    for (int i = 1; i <= id; i++)
    {
        zerr = unzGoToNextFile(pzi->info.zip.uf);
        if (zerr != UNZ_OK)
            return i + 1;
    }

    return 0;
}

int ExtractCurrentFile(ARCHINFO* pzi, int ind, char* extractDir, int usePaths, int ip)
{
    PassResult = -1;
    unzFile uf = pzi->info.zip.uf;
    char* filePathInZip = pzi->pszNames[ind];
    int res = UNZ_OK;
    int fout = 0;
    char extractFilePath[MAX_PATH];
    char fnbuf[MAX_PATH];

    if (filePathInZip == NULL)
    {
        // Считываем имя сами если нужно
        unz_file_info fi;
        filePathInZip = fnbuf;
        unzGetCurrentFileInfo(uf, &fi, filePathInZip, MAX_PATH, NULL, 0, NULL, 0);
    }

    uInt size_buf = WRITEBUFFERSIZE;
    void* buf = (void*)malloc(size_buf);
    if (buf == NULL)
    {
        ZipError(1, "ExtractCurrentFile.malloc");
        return 1;
    }

    char* fileNameInZip = GetFileName(filePathInZip);
    if ((*fileNameInZip) == '\0')
    {
        if (usePaths)
        {
            // Директория... создаем
            sprintf(extractFilePath, _s_s, extractDir, filePathInZip);
            mktree(extractFilePath);
            if (ip && progrsp_start)
                incprogr(1);
        }
    }
    else if ((*fileNameInZip) != '\0')
    {
        // File
        if (usePaths)
            sprintf(extractFilePath, _s_s, extractDir, filePathInZip);
        else
            sprintf(extractFilePath, _s_s, extractDir, fileNameInZip);

        //////////////////////////////////////////////////////////////
        if (fexist(extractFilePath))
        {
            if(CONFIG_TEMP_PATH!=extractDir)
            {
                if(CONFIG_CONFIRM_REPLACE)
                {
                    int ret=MsgBoxYesNoWithParam(ind_pmt_exists, fileNameInZip);
                    if (ret==IDYES && ret==IDYES2ALL)
                    {
                        if (!_unlink(extractFilePath, &err))
                            ZipError(res = 2, "ExtractCurrentFile.fsrm");
                    }
                    else
                        return 0;
                }
                else
                {
                    if (!_unlink(extractFilePath, &err))
                        ZipError(res = 2, "ExtractCurrentFile.fsrm");
                }
            }
            else
            {
                if(!_unlink(extractFilePath, &err))
                    ZipError(res = 2, "ExtractCurrentFile.fsrm");
            }
        }
        /////////////////////////////////////////////////////////////
        if (res == UNZ_OK)
        {
            fout = _open(extractFilePath,
                         A_ReadWrite+A_Create+A_Truncate+A_BIN, P_READ+P_WRITE, &err);

            // some zipfile don't contain directory alone before file
            if (fout == -1 && usePaths && fileNameInZip != filePathInZip)
            {
                char c = *(fileNameInZip - 1);
                *(fileNameInZip - 1) = '\0';
                mktree(GetFileDir(extractFilePath, pathbuf));
                *(fileNameInZip - 1) = c;
                fout = _open(extractFilePath, A_ReadWrite+A_Create+A_Truncate+A_BIN, P_READ+P_WRITE, &err);
            }

            if (fout == -1)
                ZipError(res = 3, "ExtractCurrentFile._open");
        }

        if (res == UNZ_OK)
        {
            // Собственно извлекаем
            res = unzOpenCurrentFilePassword(uf, pzi->password); //== UNZ_OK

            if(res!=UNZ_OK)
            {
PASS_ENT:
                ShowWaitPass(pzi);
                if(PassResult != 1)
                    res = -1;
                else
                    res = unzOpenCurrentFilePassword(uf, pzi->password);
            }
            // Откроем файл
            if (res==UNZ_OK)
            {
                int isnewprogr = 0;

                if (ip || !progrsp_start)
                {
                    if (!progr_start)
                    {
                        initprogr(ind_msg_zreading);
                        progr_max = 1;
                        incprogr(0);
                        isnewprogr = 1;
                    }
                    ip = 1;
                    progrsp_start = 1;
                    progrsp_max = pzi->info.zip.pfi[ind]->uncompressed_size;
                    incfname(fileNameInZip);
                    incprogrsp(0);
                }


                int cb = 0;
                do
                {
                    if (ip && progr_stop)
                    {
                        res = UNZ_ERRNO;
                        break;
                    }
                    cb = (int)unzReadCurrentFile(uf, buf, size_buf);
                    if (cb < 0)
                    {
                        if(PassResult != 2)
                        {
                            if (ip && progrsp_start)
                            {
                                endprogrsp();
                                progrsp_start = 0;

                                if (isnewprogr)
                                {
                                    endprogr();
                                    progr_start = 0;
                                }
                            }
                            goto PASS_ENT;
                        }
                        else
                        {
                            ZipError(cb, "ExtractCurrentFile.unzReadCurrentFile");
                            res = 5;
                            break;
                        }
                    }
                    else if (cb > 0)
                    {
                        if (_write(fout, buf, cb, &err) != cb)
                        {
                            ZipError(res = 6, "ExtractCurrentFile._write");
                            break;
                        }
                        if (ip && progrsp_start) incprogrsp(cb);
                    }
                }
                while (cb > 0);

                _close(fout, &err);

                if (ip && progrsp_start)
                {
                    endprogrsp();
                    progrsp_start = 0;
                    incprogr(1);

                    if (isnewprogr)
                    {
                        endprogr();
                        progr_start = 0;
                    }
                }
            }
            else ZipError(res = 7, "ExtractCurrentFile.unzOpenCurrentFile");

            if (fout) _close(fout, &err);
        }

        if (res == UNZ_OK)
        {
            if (unzCloseCurrentFile(uf) != UNZ_OK)
                ZipError(res = 8, "ExtractCurrentFile.unzCloseCurrentFile");
        }
        else unzCloseCurrentFile(uf); // don't lose the error
    }

    if(fout) _close(fout, &err);

    mfree(buf);
    return res;
}

int ExtractFile(ARCHINFO* pzi, char* fname, char* extractDir, int usePaths)
{
// Позиционируемся на нужном файле
    int found = 0;
    char buf[MAX_PATH + 1];
    strcpy(buf + 1, fname);
    buf[0] = '\\'; // В начало запишем слеш на тот случай если fname без слеша
    char* nameWithoutSlash = ( (buf[1] == '\\') ? (buf + 2) : (buf + 1) );
    char* nameWithSlash = ( (buf[1] == '\\') ? (buf + 1) : (buf) );

    for (uint32_t i = 0; i < pzi->info.zip.gi.number_entry; ++i)
    {
        found = (stricmp(nameWithoutSlash, pzi->pszNames[i]) == 0);
        if (!found)
            // Пробуем поискать со слешем
            found = (stricmp(nameWithSlash, pzi->pszNames[i]) == 0);

        if (found)
            // Извлекаем его, если нашли
            return ExtractFileByID(pzi, i, extractDir, usePaths, 0);
    }

    return 1;
}

int ExtractFileByID(ARCHINFO* pzi, int id, char* extractDir, int usePaths, int ip)
{
// Позиционируемся на нужном файле
    if (SetCurrentFileInZip(pzi, id) == UNZ_OK)
    {
        // И извлекаем его
        return ExtractCurrentFile(pzi, id, extractDir, usePaths, ip);
    }
    else return 1;
}

int ExtractDir(ARCHINFO* pzi, char* dname, char* extractDir, int usePaths)
{
    int found = 0;
    char buf[MAX_PATH + 1];
    strcpy(buf + 1, dname);
    buf[0] = '\\'; // В начало запишем слеш на тот случай если fname без слеша
    char* nameWithoutSlash = ( (buf[1] == '\\') ? (buf + 2) : (buf + 1) );
    char* nameWithSlash = ( (buf[1] == '\\') ? (buf + 1) : (buf) );

    int zerr = UNZ_OK;
    int dlen = strlen(nameWithoutSlash);

// Перебираем все файлы
    for (uint32_t i = 0; i < pzi->info.zip.gi.number_entry; ++i)
    {
        if (i == 0) zerr = unzGoToFirstFile(pzi->info.zip.uf);
        else zerr = unzGoToNextFile(pzi->info.zip.uf);

        if (zerr) return zerr;

        // Проверяем начинается ли имя файла в зипе с этой директории, если да - извлекаем
        found = (strnicmp(nameWithoutSlash, pzi->pszNames[i], dlen) == 0);
        if (!found)
        {
            // Пробуем поискать со слешем
            found = (strnicmp(nameWithSlash, pzi->pszNames[i], dlen + 1) == 0);
        }

        if (found)
            ExtractCurrentFile(pzi, i, extractDir, 1, 0);
    }

    return zerr;
}

void ZipBufferExtractBegin()
{
}

int ZipBufferExtract(FN_ITM* pi, char* extractDir)
{
    if (pi && pi->zipPath && pi->full)
    {
        ARCHINFO* pzi = &zi;
        int usingZipFromTab = 0;

        for (int i = 0; i < MAX_TABS; i++)
        {
            if (IsArchiveOpened(i) && stricmp(tabs[i]->zipInfo->szZipPath, pi->zipPath) == 0)
            {
                // Если уже открыт в табе, то используем его
                pzi = tabs[i]->zipInfo;
                usingZipFromTab = 1;
                break;
            }
        }

        if (!usingZipFromTab)
        {
            // Если файл уже открыт, но имена не совпадают, то закрываем старый
            if (pzi->info.zip.uf != 0 && stricmp(pzi->szZipPath, pi->zipPath) != 0)
                CloseZip(pzi);

            // Если файл еще не открыт - то открываем
            if (pzi->info.zip.uf == 0)
                if (OpenZip(pzi, pi->zipPath) != UNZ_OK)
                    return UNZ_ERRNO;
        }

        if (pi->ftype == TYPE_ZIP_FILE)
            return ExtractFile(pzi, pi->full, extractDir, 0);
        else if (pi->ftype == TYPE_ZIP_DIR)
            return ExtractDir(pzi, pi->full, extractDir, 1);
    }

    return 0;
}

void ZipBufferExtractEnd()
{
    if (zi.info.zip.uf != 0)
        CloseZip(&zi);
}

int EnumZipFiles(ARCHINFO* pzi, char* subdname, ENUM_SEL_PROC enumproc, unsigned int param)
{
    unsigned int ccFiles   = 0;
    unsigned int ccSubDirs = 0;

    char buf[MAX_PATH_INZIP];
    char pszDirName[256]; // Пропустим в начале обратный слеш
    strcpy(pszDirName, subdname[0] == '\\' ? subdname + 1 : subdname);
    int dirLen = strlen(pszDirName);

    if (pzi->info.zip.uf)
    {
        for (uint32_t i=0; i < pzi->info.zip.gi.number_entry; ++i)
        {
            int ignore = 1;
            strcpy(buf, pzi->pszNames[i]);
            char* pFileNameStart = &buf[0];
            unz_file_info* pfi = pzi->info.zip.pfi[i];

            if (dirLen == 0)
            {
                ignore = 0;
            }
            else
            {
                // Сравниваем начало, если равно и потом слеш - то это элементы нашей директории
                pFileNameStart = pFileNameStart + dirLen;
                ignore = (strnicmp(buf, pszDirName, dirLen) != 0 || *pFileNameStart != '\\');
                if (!ignore)
                {
                    pFileNameStart = pFileNameStart + 1; // пропустим слеш после имени директории

                    // игнорим саму текущую, директорию
                    ignore = (*pFileNameStart == 0);
                }
            }
            if (!ignore)
            {
                int stop = 0;

                if (pfi->external_fa & FA_DIRECTORY)
                    ccSubDirs++;
                else
                    ccFiles++;

                if (enumproc)
                {
                    FILEINF* file = CreateFileInfo(i, pFileNameStart, pfi->uncompressed_size,
                                                   pfi->external_fa, pfi->dosDate, pfi->compressed_size,
                                                   (pfi->external_fa & FA_DIRECTORY) ? TYPE_ZIP_DIR : TYPE_ZIP_FILE, pszDirName);

                    if (enumproc(file, param) == 0)
                        stop = 1;

                    FreeFileInfo(file);
                }

                if (stop) break;
            }
        }
    }

    return (ccSubDirs << 16 | ccFiles);
}

int IsItZipFile(char* fname)
{
    if(IsInArchive())
        return (stricmp(GetFileExt(fname), (char*)zip_ext) == 0);
    else
    {
        if ((!CONFIG_JAR_ENABLE)&&(stricmp(GetFileExt(fname), (char*)jar_ext) == 0))
            return -1;
        if ((!CONFIG_SDT_ENABLE)&&(stricmp(GetFileExt(fname), (char*)sdt_ext) == 0))
            return -1;

        if (CONFIG_ZIP_DETECT_BY == 0) // by content
        {
            unsigned char buf[4];
            uLong header = 0;
            int f = _open(fname, A_ReadOnly + A_BIN, P_READ, &err);
            if (f != -1)
            {
                if (_read(f, &buf, 4, &err) == 4)
                    header = (uLong)buf[0] + (((uLong)buf[1])<<8) + (((uLong)buf[2])<<16) + (((uLong)buf[3])<<24);
                _close(f, &err);
            }
            return (header == LOCALHEADERMAGIC);
        }
        else // by extension
            return (stricmp(GetFileExt(fname), (char*)zip_ext) == 0);
    }
}

int _UnZip(void)
{
    int ret = 0;
    ARCHINFO *pzi=&zi;

    char *extractdir = malloc(256);
    strcpy(extractdir, zippathbuf);
    strreplace(extractdir, '.', '_');
    mktree(extractdir);

    initprogr(ind_copying);
    incprogr(0);
    ret = OpenZip(pzi, zippathbuf);
    if (ret == UNZ_OK)
    {
        progr_max = pzi->info.zip.gi.number_entry;

        uint32_t i = 0;
        while( i < pzi->info.zip.gi.number_entry && !progr_stop)
        {
            if (!msgopen)
            {
                if (i == 0)
                    ret = unzGoToFirstFile(pzi->info.zip.uf);
                else
                    ret = unzGoToNextFile(pzi->info.zip.uf);

                if(ret) return ret;

                ExtractCurrentFile(pzi, i, extractdir, 1, 0);
                i++;
            }
        }
    }
    CloseZip(pzi);
    DoRefresh();

    int ind = GetCurTabFileIndex(GetFileName(extractdir));
    SetCurTabIndex(ind, 0);

    endprogr();

    return ret;
}

