mirror of
				https://github.com/cookiengineer/audacity
				synced 2025-10-26 15:23:48 +01:00 
			
		
		
		
	Locate and position the current Audacity source code, and clear a variety of old junk out of the way into junk-branches
This commit is contained in:
		
							
								
								
									
										352
									
								
								src/AutoRecovery.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										352
									
								
								src/AutoRecovery.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,352 @@ | ||||
| /********************************************************************** | ||||
|  | ||||
|   Audacity: A Digital Audio Editor | ||||
|  | ||||
|   AutoRecovery.cpp | ||||
|  | ||||
| *******************************************************************//** | ||||
|  | ||||
| \class AutoRecoveryDialog | ||||
| \brief The AutoRecoveryDialog prompts the user whether to  | ||||
| recover previous Audacity projects that were closed incorrectly. | ||||
|  | ||||
| *//********************************************************************/ | ||||
|  | ||||
| #include "AutoRecovery.h" | ||||
| #include "Audacity.h" | ||||
| #include "AudacityApp.h" | ||||
| #include "FileNames.h" | ||||
| #include "blockfile/SimpleBlockFile.h" | ||||
|  | ||||
| #include <wx/wxprec.h> | ||||
| #include <wx/filefn.h> | ||||
| #include <wx/dir.h> | ||||
| #include <wx/dialog.h> | ||||
| #include <wx/app.h> | ||||
|  | ||||
| enum { | ||||
|    ID_RECOVER_ALL = 10000, | ||||
|    ID_RECOVER_NONE, | ||||
|    ID_QUIT_AUDACITY, | ||||
|    ID_FILE_LIST | ||||
| }; | ||||
|  | ||||
| class AutoRecoveryDialog : public wxDialog | ||||
| { | ||||
| public: | ||||
|    AutoRecoveryDialog(wxWindow *parent); | ||||
|  | ||||
| private: | ||||
|    void PopulateList(); | ||||
|    void PopulateOrExchange(ShuttleGui & S); | ||||
|     | ||||
|    void OnQuitAudacity(wxCommandEvent &evt); | ||||
|    void OnRecoverNone(wxCommandEvent &evt); | ||||
|    void OnRecoverAll(wxCommandEvent &evt); | ||||
|  | ||||
|    wxListCtrl *mFileList; | ||||
|  | ||||
| public: | ||||
|    DECLARE_EVENT_TABLE() | ||||
| }; | ||||
|  | ||||
| AutoRecoveryDialog::AutoRecoveryDialog(wxWindow *parent) : | ||||
|    wxDialog(parent, -1, _("Automatic Crash Recovery"), | ||||
|             wxDefaultPosition, wxDefaultSize, | ||||
|             wxDEFAULT_DIALOG_STYLE & (~wxCLOSE_BOX)) // no close box | ||||
| { | ||||
|    ShuttleGui S(this, eIsCreating); | ||||
|    PopulateOrExchange(S); | ||||
| } | ||||
|  | ||||
| BEGIN_EVENT_TABLE(AutoRecoveryDialog, wxDialog) | ||||
|    EVT_BUTTON(ID_RECOVER_ALL, AutoRecoveryDialog::OnRecoverAll) | ||||
|    EVT_BUTTON(ID_RECOVER_NONE, AutoRecoveryDialog::OnRecoverNone) | ||||
|    EVT_BUTTON(ID_QUIT_AUDACITY, AutoRecoveryDialog::OnQuitAudacity) | ||||
| END_EVENT_TABLE() | ||||
|  | ||||
| void AutoRecoveryDialog::PopulateOrExchange(ShuttleGui& S) | ||||
| { | ||||
|    S.SetBorder(5); | ||||
|    S.StartVerticalLay(); | ||||
|    { | ||||
|       S.AddVariableText(_("Some projects were not saved properly the last time Audacity was run.\nFortunately, the following projects can automatically be recovered:"), false); | ||||
|  | ||||
|       S.StartStatic(_("Recoverable projects")); | ||||
|       {   | ||||
|          mFileList = S.Id(ID_FILE_LIST).AddListControlReportMode(); | ||||
|          mFileList->InsertColumn(0, _("Name")); | ||||
|          mFileList->SetColumnWidth(0, 220); | ||||
|          PopulateList(); | ||||
|       } | ||||
|       S.EndStatic(); | ||||
|  | ||||
|       S.AddVariableText(_("Recovering a project will not change any files on disk before you save it."), false); | ||||
|  | ||||
|       S.StartHorizontalLay(true); | ||||
|       { | ||||
|          S.Id(ID_QUIT_AUDACITY).AddButton(_("Quit Audacity")); | ||||
|          S.Id(ID_RECOVER_NONE).AddButton(_("Do Not Recover")); | ||||
|          S.Id(ID_RECOVER_ALL).AddButton(_("Recover Projects")); | ||||
|       } | ||||
|       S.EndHorizontalLay(); | ||||
|    } | ||||
|    S.EndVerticalLay(); | ||||
|  | ||||
|    Layout(); | ||||
|    Fit(); | ||||
|    SetMinSize(GetSize()); | ||||
|    Center(); | ||||
| } | ||||
|  | ||||
| void AutoRecoveryDialog::PopulateList() | ||||
| { | ||||
|    mFileList->DeleteAllItems(); | ||||
|     | ||||
|    wxDir dir(FileNames::AutoSaveDir()); | ||||
|    if (!dir.IsOpened()) | ||||
|       return; | ||||
|        | ||||
|    wxString filename; | ||||
|    int i = 0; | ||||
|    for (bool c = dir.GetFirst(&filename, wxT("*.autosave"), wxDIR_FILES); | ||||
|         c; c = dir.GetNext(&filename)) | ||||
|       mFileList->InsertItem(i++, wxFileName(filename).GetName()); | ||||
|        | ||||
|    mFileList->SetColumnWidth(0, wxLIST_AUTOSIZE); | ||||
| } | ||||
|  | ||||
| void AutoRecoveryDialog::OnQuitAudacity(wxCommandEvent &evt) | ||||
| { | ||||
|    EndModal(ID_QUIT_AUDACITY); | ||||
| } | ||||
|  | ||||
| void AutoRecoveryDialog::OnRecoverNone(wxCommandEvent &evt) | ||||
| { | ||||
|    int ret = wxMessageBox( | ||||
|       _("Are you sure you don't want to recover any projects?\nThey can't be recovered later."), | ||||
|       _("Confirm?"), wxICON_QUESTION | wxYES_NO | wxNO_DEFAULT, this); | ||||
|  | ||||
|    if (ret == wxYES) | ||||
|       EndModal(ID_RECOVER_NONE); | ||||
| } | ||||
|  | ||||
| void AutoRecoveryDialog::OnRecoverAll(wxCommandEvent &evt) | ||||
| { | ||||
|    EndModal(ID_RECOVER_ALL); | ||||
| } | ||||
|  | ||||
| //////////////////////////////////////////////////////////////////////////// | ||||
|  | ||||
| static bool HaveFilesToRecover() | ||||
| { | ||||
|    wxDir dir(FileNames::AutoSaveDir()); | ||||
|    if (!dir.IsOpened()) | ||||
|    { | ||||
|       wxMessageBox(_("Could not enumerate files in auto save directory"), | ||||
|                    _("Error"), wxICON_STOP); | ||||
|       return false; | ||||
|    } | ||||
|     | ||||
|    wxString filename; | ||||
|    bool c = dir.GetFirst(&filename, wxT("*.autosave"), wxDIR_FILES); | ||||
|     | ||||
|    return c; | ||||
| } | ||||
|  | ||||
| static bool RemoveAllAutoSaveFiles() | ||||
| { | ||||
|    wxArrayString files; | ||||
|    wxDir::GetAllFiles(FileNames::AutoSaveDir(), &files, | ||||
|                       wxT("*.autosave"), wxDIR_FILES); | ||||
|  | ||||
|    for (unsigned int i = 0; i < files.GetCount(); i++) | ||||
|    { | ||||
|       if (!wxRemoveFile(files[i])) | ||||
|       { | ||||
|          // I don't think this error message is actually useful. | ||||
|          // -dmazzoni | ||||
|          //wxMessageBox(wxT("Could not remove auto save file: " + files[i]), | ||||
|          //             _("Error"), wxICON_STOP); | ||||
|          return false; | ||||
|       } | ||||
|    } | ||||
|     | ||||
|    return true; | ||||
| } | ||||
|  | ||||
| static bool RecoverAllProjects(AudacityProject** pproj) | ||||
| { | ||||
|    wxDir dir(FileNames::AutoSaveDir()); | ||||
|    if (!dir.IsOpened()) | ||||
|    { | ||||
|       wxMessageBox(_("Could not enumerate files in auto save directory"), | ||||
|                    _("Error"), wxICON_STOP); | ||||
|       return false; | ||||
|    } | ||||
|     | ||||
|    // Open a project window for each auto save file | ||||
|    wxString filename; | ||||
|    AudacityProject* proj = NULL; | ||||
|     | ||||
|    wxArrayString files; | ||||
|    wxDir::GetAllFiles(FileNames::AutoSaveDir(), &files, | ||||
|                       wxT("*.autosave"), wxDIR_FILES); | ||||
|  | ||||
|    for (unsigned int i = 0; i < files.GetCount(); i++) | ||||
|    { | ||||
|       if (*pproj) | ||||
|       { | ||||
|          // Reuse existing project window | ||||
|          proj = *pproj; | ||||
|          *pproj = NULL; | ||||
|       } else | ||||
|       { | ||||
|          // Create new project window | ||||
|          proj = CreateNewAudacityProject(); | ||||
|       } | ||||
|  | ||||
|       // Open project. When an auto-save file has been opened successfully, | ||||
|       // the opened auto-save file is automatically deleted and a new one | ||||
|       // is created. | ||||
|       proj->OpenFile(files[i], false); | ||||
|  | ||||
|       //fit project after recovery | ||||
|       proj->OnZoomFit(); | ||||
|    } | ||||
|     | ||||
|    return true; | ||||
| } | ||||
|  | ||||
| bool ShowAutoRecoveryDialogIfNeeded(AudacityProject** pproj, | ||||
|                                     bool *didRecoverAnything) | ||||
| { | ||||
|    if (didRecoverAnything) | ||||
|       *didRecoverAnything = false; | ||||
|  | ||||
|    if (HaveFilesToRecover()) | ||||
|    { | ||||
|       AutoRecoveryDialog dlg(*pproj); | ||||
|       int ret = dlg.ShowModal(); | ||||
|     | ||||
|       switch (ret) | ||||
|       { | ||||
|       case ID_RECOVER_NONE: | ||||
|          return RemoveAllAutoSaveFiles(); | ||||
|     | ||||
|       case ID_RECOVER_ALL: | ||||
|          if (didRecoverAnything) | ||||
|             *didRecoverAnything = true; | ||||
|          return RecoverAllProjects(pproj); | ||||
|  | ||||
|       default: | ||||
|          // This includes ID_QUIT_AUDACITY | ||||
|          return false; | ||||
|       } | ||||
|    } else | ||||
|    { | ||||
|       // Nothing to recover, move along | ||||
|       return true; | ||||
|    } | ||||
| } | ||||
|  | ||||
| //////////////////////////////////////////////////////////////////////////// | ||||
| /// Recording recovery handler | ||||
|  | ||||
| RecordingRecoveryHandler::RecordingRecoveryHandler(AudacityProject* proj) | ||||
| { | ||||
|    mProject = proj; | ||||
|    mChannel = -1; | ||||
|    mNumChannels = -1; | ||||
| } | ||||
|  | ||||
| bool RecordingRecoveryHandler::HandleXMLTag(const wxChar *tag, | ||||
|                                             const wxChar **attrs) | ||||
| { | ||||
|    if (wxStrcmp(tag, wxT("simpleblockfile")) == 0) | ||||
|    { | ||||
|       // Check if we have a valid channel and numchannels | ||||
|       if (mChannel < 0 || mNumChannels < 0 || mChannel >= mNumChannels) | ||||
|       { | ||||
|          // This should only happen if there is a bug | ||||
|          wxASSERT(false); | ||||
|          return false; | ||||
|       } | ||||
|  | ||||
|       // We need to find the track and sequence where the blockfile belongs | ||||
|       WaveTrackArray tracks = mProject->GetTracks()->GetWaveTrackArray(false); | ||||
|       int index = tracks.GetCount() - mNumChannels + mChannel; | ||||
|       if (index < 0 || index >= (int)tracks.GetCount()) | ||||
|       { | ||||
|          // This should only happen if there is a bug | ||||
|          wxASSERT(false); | ||||
|          return false; | ||||
|       } | ||||
|       WaveTrack* track = tracks.Item(index); | ||||
|       Sequence* seq = track->GetLastOrCreateClip()->GetSequence(); | ||||
|        | ||||
|       // Load the blockfile from the XML | ||||
|       BlockFile* blockFile = NULL; | ||||
|       DirManager* dirManager = mProject->GetDirManager(); | ||||
|       dirManager->SetLoadingFormat(seq->GetSampleFormat()); | ||||
|       dirManager->SetLoadingTarget(&blockFile); | ||||
|       if (!dirManager->HandleXMLTag(tag, attrs) || !blockFile) | ||||
|       { | ||||
|          // This should only happen if there is a bug | ||||
|          wxASSERT(false); | ||||
|          return false; | ||||
|       } | ||||
|  | ||||
|       seq->AppendBlockFile(blockFile); | ||||
|  | ||||
|    } else if (wxStrcmp(tag, wxT("recordingrecovery")) == 0) | ||||
|    { | ||||
|       // loop through attrs, which is a null-terminated list of | ||||
|       // attribute-value pairs | ||||
|       long nValue; | ||||
|       while(*attrs) | ||||
|       { | ||||
|          const wxChar *attr = *attrs++; | ||||
|          const wxChar *value = *attrs++; | ||||
|  | ||||
|          if (!value) | ||||
|             break; | ||||
|           | ||||
|          const wxString strValue = value; | ||||
|          if (wxStrcmp(attr, wxT("channel")) == 0) | ||||
|          { | ||||
|             if (!XMLValueChecker::IsGoodInt(strValue) || !strValue.ToLong(&nValue) ||  | ||||
|                   !XMLValueChecker::IsValidChannel(nValue)) | ||||
|                return false; | ||||
|             mChannel = nValue; | ||||
|          } | ||||
|          else if (wxStrcmp(attr, wxT("numchannels")) == 0) | ||||
|          { | ||||
|             if (!XMLValueChecker::IsGoodInt(strValue) || !strValue.ToLong(&nValue) ||  | ||||
|                   (nValue < 1)) | ||||
|                return false; | ||||
|             mNumChannels = nValue; | ||||
|          } | ||||
|       } | ||||
|    } | ||||
|     | ||||
|    return true; | ||||
| } | ||||
|  | ||||
| XMLTagHandler* RecordingRecoveryHandler::HandleXMLChild(const wxChar *tag) | ||||
| { | ||||
|    if (wxStrcmp(tag, wxT("simpleblockfile")) == 0) | ||||
|       return this; // HandleXMLTag also handles <simpleblockfile> | ||||
|     | ||||
|    return NULL; | ||||
| } | ||||
|  | ||||
| // Indentation settings for Vim and Emacs. | ||||
| // Please do not modify past this point. | ||||
| // | ||||
| // Local Variables: | ||||
| // c-basic-offset: 3 | ||||
| // indent-tabs-mode: nil | ||||
| // End: | ||||
| // | ||||
| // vim: et sts=3 sw=3 | ||||
		Reference in New Issue
	
	Block a user