Есть ли функция Qt для перемещения файлов в Корзину вместо действительно их удаления, для поддерживающих ее ОС или мне нужно использовать код, специфичный для ОС?
Перемещение файлов в корзину/корзину в Qt
Ответ 1
Пока нет API.
https://bugreports.qt.io/browse/QTBUG-181
Проблема закрыта, и версия исправления: Некоторая версия в будущем
Изменить: Новая проблема была открыта в https://bugreports.qt.io/browse/QTBUG-47703.
Ответ 2
Qt не предоставляет MoveToTrash. Здесь часть моего кода
для Windows
#ifdef Q_OS_WIN32
#include "windows.h"
void MoveToTrashImpl( QString file ){
QFileInfo fileinfo( file );
if( !fileinfo.exists() )
throw OdtCore::Exception( "File doesnt exists, cant move to trash" );
WCHAR from[ MAX_PATH ];
memset( from, 0, sizeof( from ));
int l = fileinfo.absoluteFilePath().toWCharArray( from );
Q_ASSERT( 0 <= l && l < MAX_PATH );
from[ l ] = '\0';
SHFILEOPSTRUCT fileop;
memset( &fileop, 0, sizeof( fileop ) );
fileop.wFunc = FO_DELETE;
fileop.pFrom = from;
fileop.fFlags = FOF_ALLOWUNDO | FOF_NOCONFIRMATION | FOF_NOERRORUI | FOF_SILENT;
int rv = SHFileOperation( &fileop );
if( 0 != rv ){
qDebug() << rv << QString::number( rv ).toInt( 0, 8 );
throw OdtCore::Exception( "move to trash failed" );
}
}
#endif
и для Linux
#ifdef Q_OS_LINUX
bool TrashInitialized = false;
QString TrashPath;
QString TrashPathInfo;
QString TrashPathFiles;
void MoveToTrashImpl( QString file ){
#ifdef QT_GUI_LIB
if( !TrashInitialized ){
QStringList paths;
const char* xdg_data_home = getenv( "XDG_DATA_HOME" );
if( xdg_data_home ){
qDebug() << "XDG_DATA_HOME not yet tested";
QString xdgTrash( xdg_data_home );
paths.append( xdgTrash + "/Trash" );
}
QString home = QStandardPaths::writableLocation( QStandardPaths::HomeLocation );
paths.append( home + "/.local/share/Trash" );
paths.append( home + "/.trash" );
foreach( QString path, paths ){
if( TrashPath.isEmpty() ){
QDir dir( path );
if( dir.exists() ){
TrashPath = path;
}
}
}
if( TrashPath.isEmpty() )
throw Exception( "Cant detect trash folder" );
TrashPathInfo = TrashPath + "/info";
TrashPathFiles = TrashPath + "/files";
if( !QDir( TrashPathInfo ).exists() || !QDir( TrashPathFiles ).exists() )
throw Exception( "Trash doesnt looks like FreeDesktop.org Trash specification" );
TrashInitialized = true;
}
QFileInfo original( file );
if( !original.exists() )
throw Exception( "File doesnt exists, cant move to trash" );
QString info;
info += "[Trash Info]\nPath=";
info += original.absoluteFilePath();
info += "\nDeletionDate=";
info += QDateTime::currentDateTime().toString("yyyy-MM-ddThh:mm:ss.zzzZ");
info += "\n";
QString trashname = original.fileName();
QString infopath = TrashPathInfo + "/" + trashname + ".trashinfo";
QString filepath = TrashPathFiles + "/" + trashname;
int nr = 1;
while( QFileInfo( infopath ).exists() || QFileInfo( filepath ).exists() ){
nr++;
trashname = original.baseName() + "." + QString::number( nr );
if( !original.completeSuffix().isEmpty() ){
trashname += QString( "." ) + original.completeSuffix();
}
infopath = TrashPathInfo + "/" + trashname + ".trashinfo";
filepath = TrashPathFiles + "/" + trashname;
}
QDir dir;
if( !dir.rename( original.absoluteFilePath(), filepath ) ){
throw Exception( "move to trash failed" );
}
File infofile;
infofile.createUtf8( infopath, info );
#else
Q_UNUSED( file );
throw Exception( "Trash in server-mode not supported" );
#endif
}
#endif
Ответ 3
Я думаю, что нет кросс-платформенного способа. Простое перемещение файлов в "мусорное" место не даст эффекта, так как пользователь может отключить эту возможность.
Возможно, этот url поможет: http://www.hardcoded.net/articles/send-files-to-trash-on-all-platforms.htm
Ответ 4
Я уверен, что нет Qt API, который обертывает это для всех поддерживаемых платформ. Это означает, к сожалению, что вам придется писать код, специфичный для платформы.
Я ничего не знаю о том, где/как дистрибутивы Linux хранят удаленные файлы, и я полагаю, что это, вероятно, меняется в зависимости от того, какой файловый менеджер вы используете. Я считаю, что перемещение файлов в папку ~/.Trash
- это стандартный способ сделать это, но я не уверен, что это надежный. Например, в случае файлов, хранящихся на внешних томах.
В Mac OS X ситуация немного проще, где для этого есть поддерживаемый API: FSMoveObjectToTrashSync
, предоставляемый Core Services. По крайней мере, как я помню, вы должны это делать. Документация утверждает, что этот метод теперь устарел в OS X 10.8. Я понятия не имею, что такое рекомендуемая альтернатива.
Как программист Windows, я думаю, что платформа намного проще.:-) Основным решением является вызов функции SHFileOperation
:
#include <Windows.h> // general Windows header file
#include <ShellAPI.h> // for shell functions, like SHFileOperation
#include <string> // (or use QString)
void RecycleFileOnWindows()
{
std::wstring path = L"C:\\Users\\Administrator\\Documents\\deleteme.txt";
path.append(1, L'\0'); // path string must be double nul-terminated
SHFILEOPSTRUCT shfos = {};
shfos.hwnd = nullptr; // handle to window that will own generated windows, if applicable
shfos.wFunc = FO_DELETE;
shfos.pFrom = path.c_str();
shfos.pTo = nullptr; // not used for deletion operations
shfos.fFlags = FOF_ALLOWUNDO; // use the recycle bin
const int retVal = SHFileOperation(&shfos);
if (retVal != 0)
{
// The operation failed...
if (shfos.fAnyOperationsAborted)
{
// ...but that because the user canceled.
MessageBox(nullptr, L"Operation was canceled", nullptr, MB_OK | MB_ICONINFORMATION);
}
else
{
// ...for one of the other reasons given in the documentation.
MessageBox(nullptr, L"Operation failed", nullptr, MB_OK | MB_ICONERROR);
}
}
}
Существуют также флаги, которые вы можете настроить для настройки подтверждения, отчетов об ошибках и другого поведения. Связанная документация содержит все сведения, необходимые для построения этого базового примера.
В Windows Vista и более поздних версиях функция SHFileOperation
была заменена методами, предоставляемыми интерфейсом IFileOperation
. Если вы ориентируетесь только на эти более поздние версии Windows, вы должны использовать этот интерфейс. В противном случае SHFileOperation
будет продолжать работать нормально.
Ответ 5
if(QSysInfo::kernelType()=="linux")
{
QDateTime currentTime(QDateTime::currentDateTime()); // save System time
QString trashFilePath=QDir::homePath()+"/.local/share/Trash/files/"; // trash file path contain delete files
QString trashInfoPath=QDir::homePath()+"/.local/share/Trash/info/"; // trash info path contain delete files information
// create file format for trash info file----- START
QFile infoFile(trashInfoPath+FileName.completeBaseName()+"."+FileName.completeSuffix()+".trashinfo"); //filename+extension+.trashinfo // create file information file in /.local/share/Trash/info/ folder
infoFile.open(QIODevice::ReadWrite);
QTextStream stream(&infoFile); // for write data on open file
stream<<"[Trash Info]"<<endl;
stream<<"Path="+QString(QUrl::toPercentEncoding(FileName.absoluteFilePath(),"~_-./"))<<endl; // convert path string in percentage decoding scheme string
stream<<"DeletionDate="+currentTime.toString("yyyy-MM-dd")+"T"+currentTime.toString("hh:mm:ss")<<endl; // get date and time format YYYY-MM-DDThh:mm:ss
infoFile.close();
// create info file format of trash file----- END
QDir file;
file.rename(FileName.absoluteFilePath(),trashFilePath+FileName.completeBaseName()+"."+FileName.completeSuffix()); // rename(file old path, file trash path)
}
Ответ 6
Файлы корзины в каталоге linux существуют /home/user_name/.local/share/Trash/files/
, но для этого также требуется информационный файл для каждого файла мусора, который существует в каталоге /home/user_name/.local/share/Trash/info/
. когда мы хотим переместить файл в корзину, на самом деле переместите файл в каталог /home/user_name/.local/share/Trash/files/
и создайте информационный файл в каталоге /home/user_name/.local/share/Trash/info/
. Внутри .trashinfo формат использует процентную схему декодирования для заданного пути к файлу, где существует файл, в файле info также содержится время и дата удаления.