rework undo compression, add shift viewer overwr/copy/clip/splice, fix paste edl...
authorGood Guy <good1.2guy@gmail.com>
Thu, 8 Mar 2018 23:35:03 +0000 (16:35 -0700)
committerGood Guy <good1.2guy@gmail.com>
Thu, 8 Mar 2018 23:35:03 +0000 (16:35 -0700)
14 files changed:
cinelerra-5.1/cinelerra/awindowgui.C
cinelerra-5.1/cinelerra/clippopup.C
cinelerra-5.1/cinelerra/cwindowgui.C
cinelerra-5.1/cinelerra/filexml.C
cinelerra-5.1/cinelerra/filexml.h
cinelerra-5.1/cinelerra/mbuttons.C
cinelerra-5.1/cinelerra/mwindow.h
cinelerra-5.1/cinelerra/mwindowedit.C
cinelerra-5.1/cinelerra/undostack.C
cinelerra-5.1/cinelerra/undostack.h
cinelerra-5.1/cinelerra/vwindow.C
cinelerra-5.1/cinelerra/vwindow.h
cinelerra-5.1/cinelerra/vwindowgui.C
cinelerra-5.1/cinelerra/zwindowgui.C

index 796a739e6315b99e940bdfc968ffbf08045bfb5d..ea21a02cd70256b67222eee0d9c128eb679c3a99 100644 (file)
@@ -622,7 +622,7 @@ void AssetPicon::create_objects()
                                        icon_vframe = VFramePng::vframe_png(clip_icon_path);
                                }
                                if( !icon_vframe ) {
-printf("render clip: %s\n", name);
+//printf("render clip: %s\n", name);
                                        int edl_h = edl->get_h(), edl_w = edl->get_w();
                                        int height = edl_h > 0 ? edl_h : 1;
                                        int width = edl_w > 0 ? edl_w : 1;
index a364947a94269793fc6db9c92221f263e97cc167..dd73896330963ebdd431aaf67b58155b3a704fd0 100644 (file)
@@ -253,10 +253,14 @@ int ClipPopupCopy::handle_event()
        MWindowGUI *gui = mwindow->gui;
        gui->lock_window("ClipPopupCopy::handle_event");
        if( mwindow->session->drag_clips->total > 0 ) {
-               FileXML file;
                EDL *edl = mwindow->session->drag_clips->values[0];
+               EDL *copy_edl = new EDL; // no parent or assets wont be copied
+               copy_edl->create_objects();
+               copy_edl->copy_all(edl);
+               FileXML file;
                double start = 0, end = edl->tracks->total_length();
-               edl->copy(start, end, 1, &file, "", 1);
+               copy_edl->copy(start, end, 1, &file, "", 1);
+               copy_edl->remove_user();
                const char *file_string = file.string();
                long file_length = strlen(file_string);
                gui->to_clipboard(file_string, file_length, SECONDARY_SELECTION);
index 7b6d1de2388b7e18cce7fd6867f4189f9da14456..37dd35bf20c3c244aead401cf5705b10e811dac6 100644 (file)
@@ -765,7 +765,7 @@ CWrapper_cut(next_edit)
 
 void CWindowEditing::to_clip()
 {
-        mwindow->to_clip(mwindow->edl, _("composer window: "));
+        mwindow->to_clip(mwindow->edl, _("composer window: "), 0);
 }
 
 
index f707ad60b9b8d5abb326fa915874dfb812f654b5..065dce7db07758c48a49b9a6366a3dac0e0180e1 100644 (file)
@@ -58,7 +58,7 @@ XMLBuffer::XMLBuffer(const char *buf, long buf_size, int del)
        destroy = del;
 }
 
-XMLBuffer::XMLBuffer(long buf_size, const char *buf, int del)
+XMLBuffer::XMLBuffer(long buf_size, char *buf, int del)
 {      // writing
        bfr = (unsigned char *)buf;
        bsz = buf_size;
index c6506f02d8e108ed030afbde35d33959a1e3ca2c..a701b649e0cc13352f0137f661131409ba8fc5af 100644 (file)
@@ -41,7 +41,7 @@ class XMLBuffer
        unsigned char *&demand(long len);
 public:
        XMLBuffer(long buf_size=0x1000, long max_size=LONG_MAX, int del=1);
-       XMLBuffer(long buf_size, const char *buf, int del=0); // writing
+       XMLBuffer(long buf_size, char *buf, int del=0); // writing
        XMLBuffer(const char *buf, long buf_size, int del=0); // reading
        ~XMLBuffer();
 
index ea1abf2f474a2f22c5b090947361cf3491a494dd..3e650cb6125e6cbe35374bfa28a3db3197c8052a 100644 (file)
@@ -154,7 +154,7 @@ void MainEditing::to_clip()
 {
        MWindowGUI *gui = mwindow->gui;
        gui->unlock_window();
-       mwindow->to_clip(mwindow->edl, _("main window: "));
+       mwindow->to_clip(mwindow->edl, _("main window: "), 0);
        gui->lock_window("MainEditing::to_clip");
 }
 
index dfce9b11abc171c760a87fad8d63a44e1f762db6..8482dc06f16606211503255bed9cf785580975c7 100644 (file)
@@ -414,24 +414,17 @@ public:
        void move_tracks_up();
        void mute_selection();
        void new_folder(const char *new_folder);
-       void overwrite(EDL *source);
 // For clipboard commands
        void paste();
 // For splice and overwrite
-       int paste(double start,
-               double end,
-               FileXML *file,
-               int edit_labels,
-               int edit_plugins,
-               int edit_autos);
-       int paste_output(int64_t startproject,
-                               int64_t endproject,
-                               int64_t startsource_sample,
-                               int64_t endsource_sample,
-                               int64_t startsource_frame,
-                               int64_t endsource_frame,
-                               Asset *asset,
-                               RecordLabels *new_labels);
+       void overwrite(EDL *source, int all);
+       void splice(EDL *source, int all);
+       int paste(double start, double end, FileXML *file,
+               int edit_labels, int edit_plugins, int edit_autos);
+       int paste_output(int64_t startproject, int64_t endproject,
+               int64_t startsource_sample, int64_t endsource_sample,
+               int64_t startsource_frame, int64_t endsource_frame,
+               Asset *asset, RecordLabels *new_labels);
        void paste_silence();
 
 // Detach single transition
@@ -480,14 +473,13 @@ public:
        void set_inpoint(int is_mwindow);
        void set_outpoint(int is_mwindow);
        void unset_inoutpoint(int is_mwindow);
-       void splice(EDL *source);
        void toggle_loop_playback();
        void trim_selection();
 // Synchronize EDL settings with all playback engines depending on current
 // operation.  Doesn't redraw anything.
        void sync_parameters(int change_type = CHANGE_PARAMS);
        void save_clip(EDL *new_edl, const char *txt);
-       void to_clip(EDL *edl, const char *txt);
+       void to_clip(EDL *edl, const char *txt, int all);
        int toggle_label(int is_mwindow);
        void undo_entry(BC_WindowBase *calling_window_gui);
        void redo_entry(BC_WindowBase *calling_window_gui);
index 2d417dbd79ed939d9202b5c32adaefc17250f4d6..b406c7b207f12c4d421eedb58e25b045e3292519 100644 (file)
@@ -1058,12 +1058,14 @@ void MWindow::mute_selection()
 }
 
 
-void MWindow::overwrite(EDL *source)
+void MWindow::overwrite(EDL *source, int all)
 {
        FileXML file;
 
-       double src_start = source->local_session->get_selectionstart();
-       double overwrite_len = source->local_session->get_selectionend() - src_start;
+       double src_start = all ? 0 :
+               source->local_session->get_selectionstart();
+       double overwrite_len = all ? source->tracks->total_length() :
+               source->local_session->get_selectionend() - src_start;
        double dst_start = edl->local_session->get_selectionstart();
        double dst_len = edl->local_session->get_selectionend() - dst_start;
 
@@ -1989,13 +1991,15 @@ void MWindow::unset_inoutpoint(int is_mwindow)
        }
 }
 
-void MWindow::splice(EDL *source)
+void MWindow::splice(EDL *source, int all)
 {
        FileXML file;
 
        undo->update_undo_before();
-       double source_start = source->local_session->get_selectionstart();
-       double source_end = source->local_session->get_selectionend();
+       double source_start = all ? 0 :
+               source->local_session->get_selectionstart();
+       double source_end = all ? source->tracks->total_length() :
+               source->local_session->get_selectionend();
        source->copy(source_start, source_end, 1, &file, "", 1);
 //file.dump();
        double start = edl->local_session->get_selectionstart();
@@ -2079,14 +2083,16 @@ void MWindow::save_clip(EDL *new_edl, const char *txt)
        save_backup();
 }
 
-void MWindow::to_clip(EDL *edl, const char *txt)
+void MWindow::to_clip(EDL *edl, const char *txt, int all)
 {
        FileXML file;
        double start, end;
 
        gui->lock_window("MWindow::to_clip 1");
-       start = edl->local_session->get_selectionstart();
-       end = edl->local_session->get_selectionend();
+       start = all ? 0 :
+               edl->local_session->get_selectionstart();
+       end = all ? edl->tracks->total_length() :
+               edl->local_session->get_selectionend();
 
        if( EQUIV(end, start) ) {
                start = 0;
index fdf2c5c7a6e2ae78b3070e8c090b0bf1b66e665f..0c49981b22ff63f025deab29a4f2561de1633bad 100644 (file)
@@ -23,6 +23,7 @@
 #include "bctimer.h"
 #include "clip.h"
 #include "cstrdup.h"
+#include "filexml.h"
 #include "undostack.h"
 #include <string.h>
 
@@ -110,232 +111,59 @@ void UndoStack::dump(FILE *fp)
        fprintf(fp,"UndoStack::dump\n");
        UndoStackItem *current = last;
 // Dump most recent
-       for( int i=0; current && i<16; ++i,current=PREVIOUS ) {
-               fprintf(fp,"  %d %p %s %04jx %c\n", i, current, current->get_description(),
+       for( int i=0; current && i<32; ++i,current=PREVIOUS ) {
+               fprintf(fp,"%c %2d %14p %-24s %8d %04jx %c\n", current->is_key() ? 'k' : ' ',
+                       i, current, current->get_description(), current->get_size(),
                        current->get_flags(), current == this->current ? '*' : ' ');
 //char fn[BCSTRLEN]; sprintf(fn,"/tmp/undo%d", i); FILE *fp = fopen(fn,"w");
 //if( fp ) { char *cp=current->get_data(); fwrite(cp,strlen(cp),1,fp); fclose(fp); delete [] cp; }
        }
 }
 
+// undo can be incremental or direct, from key to current
+//#define UNDO_INCREMENTAL
 
-
-// These difference routines are straight out of the Heroinediff/Heroinepatch
-// utilities.
-
-
-// We didn't want to use the diff program because that isn't efficient enough
-// to compress EDL's.  This program also handles binary data.
-
-// Algorithm 1:
-// Search from beginning for first difference.
-// Count everything up to first difference as equal.
-// Determine length of different block by searching for transposition with most
-// similar characters.
-// For every byte in different block, search for occurrence of
-// transposed copy longer than a record header in remaining old buffer.
-// If no match is found, add 1 to the different block length and try the next
-// byte.  Once the most similar match is found, store the changed data leading up to it in the
-// difference record.
-// This can be real slow if there is a large transposition early in the file.
-
-// Algorithm 2:
-// Search backwards from end of both files for last difference.
-// Count all data from last same bytes as a transposition.
-// Search forwards from start of both files until last same bytes
-// for differences.
-
-
-// Minimum size of transposed block.
-// Smaller transpositions should be included in the same difference record.
-#define MIN_TRANS 16
-
-// Format for difference records.
-// The first 4 bytes are the offset of the change.
-// The next 4 bytes are the number of bytes of new data.
-// The next 4 bytes are the number of bytes of old data replaced.
-// The new data follows.
-static void append_record(unsigned char **result,
-       int *result_size,
-       unsigned char *new_data,
-       int new_offset,
-       int new_size,
-       int old_size)
-{
-       if( new_size || old_size ) {
-               int record_size = new_size + 12;
-               unsigned char *new_result = new unsigned char[(*result_size) + record_size];
-               memcpy(new_result, (*result), (*result_size));
-               delete [] (*result);
-
-               unsigned char *new_result_ptr = new_result + (*result_size);
-               *(int32_t*)new_result_ptr = new_offset;
-               new_result_ptr += 4;
-               *(int32_t*)new_result_ptr = new_size;
-               new_result_ptr += 4;
-               *(int32_t*)new_result_ptr = old_size;
-               new_result_ptr += 4;
-
-               memcpy(new_result_ptr, new_data, new_size);
-               (*result_size) += record_size;
-               (*result) = new_result;
-       }
-}
-
-
-
-static unsigned char* get_difference_fast(unsigned char *before,
-               int before_len,
-               unsigned char *after,
-               int after_len,
-               int *result_len,
-               int verbose)
-{
-       unsigned char *result = new unsigned char[4];
-       *result_len = 4;
-
-// Store size of new buffer
-       *(int32_t*)result = after_len;
-
-
-
-// Get last different bytes
-       unsigned char *last_difference_after = after + after_len - 1;
-       unsigned char *last_difference_before = before + before_len - 1;
-       for(;;) {
-               if( last_difference_after < after ||
-                       last_difference_before < before ) break;
-               if( *last_difference_after != *last_difference_before ) {
-                       break;
-               }
-               last_difference_after--;
-               last_difference_before--;
-       }
-       last_difference_after++;
-       last_difference_before++;
-
-       int done = 0;
-       unsigned char *before_ptr = before;
-       unsigned char *after_ptr = after;
-       unsigned char *before_end = before + before_len;
-       unsigned char *after_end = after + after_len;
-
-// Scan forward for first difference
-       while( !done ) {
-               if( before_ptr < before_end &&
-                       after_ptr < after_end ) {
-// Both characters equal
-                       if( *before_ptr == *after_ptr &&
-                               before_ptr < last_difference_before &&
-                               after_ptr < last_difference_after ) {
-                               before_ptr++;
-                               after_ptr++;
-                       }
-                       else {
-// Characters differ
-// Get length of difference.
-                               unsigned char *before_difference_start = before_ptr;
-                               unsigned char *after_difference_start = after_ptr;
-
-
-                               while( *before_ptr != *after_ptr &&
-                                   before_ptr < last_difference_before &&
-                                   after_ptr < last_difference_after ) {
-                                       before_ptr++;
-                                       after_ptr++;
-                               }
-
-// Finished comparing if either pointer hits its last difference
-                               if( before_ptr >= last_difference_before ||
-                                       after_ptr >= last_difference_after ) {
-                                       done = 1;
-                                       before_ptr = last_difference_before;
-                                       after_ptr = last_difference_after;
-                               }
-
-                               int after_start = after_difference_start - after;
-                               int before_start = before_difference_start - before;
-                               int after_len = after_ptr - after_difference_start;
-                               int before_len = before_ptr - before_difference_start;
-
-                               if( verbose ) {
-                                       char string[1024];
-                                       memcpy(string, after_difference_start, MIN(after_len, 40));
-                                       string[MIN(after_len, 40)] = 0;
-                                       printf("after_offset=0x%x before_offset=0x%x after_size=%d before_size=%d \"%s\"\n",
-                                               after_start, before_start, after_len, before_len, string);
-                               }
-
-// Create difference record
-                               append_record(&result, result_len, after_difference_start,
-                                       after_start, after_len, before_len);
-                       }
-               }
-               else
-                       done = 1;
-       }
-       return result;
-}
-
-static void get_record(unsigned char **patch_ptr, unsigned char *patch_end,
-       unsigned char **after_data, int *after_offset, int *after_size, int *before_size)
-{
-       (*after_data) = 0;
-       if( (*patch_ptr) < patch_end ) {
-               (*after_offset) = *(int32_t*)(*patch_ptr);  (*patch_ptr) += 4;
-               (*after_size) = *(int32_t*)(*patch_ptr);    (*patch_ptr) += 4;
-               (*before_size) = *(int32_t*)(*patch_ptr);   (*patch_ptr) += 4;
-               (*after_data) = (*patch_ptr);               (*patch_ptr) += (*after_size);
-       }
-}
-
+class undo_record {
+public:
+       int32_t new_ofs;
+       int32_t new_len;
+       int32_t old_len;
+};
 
 static unsigned char* apply_difference(unsigned char *before, int before_len,
                unsigned char *patch, int patch_len, int *result_len)
 {
-       unsigned char *patch_ptr = patch;
-       *result_len = *(int32_t*)patch_ptr;
-       patch_ptr += 4;
-       unsigned char *patch_end = patch + patch_len;
-       //unsigned char *before_end = before + before_len;
-
-       unsigned char *result = new unsigned char[*result_len];
-       unsigned char *result_ptr = result;
-       unsigned char *result_end = result + *result_len;
-       unsigned char *before_ptr = before;
-
-       int done = 0;
-       while( !done ) {
-               unsigned char *after_data;
-               int after_offset, after_size, before_size;
-
-               get_record(&patch_ptr, patch_end,
-                       &after_data, &after_offset, &after_size, &before_size);
-
-               if( after_data ) {
-                       int result_offset = result_ptr - result;
-                       if( after_offset > result_offset ) {
-                               int skip_size = after_offset - result_offset;
-                               memcpy(result_ptr, before_ptr, skip_size);
-                               result_ptr += skip_size;
-                               before_ptr += skip_size;
-                       }
-                       memcpy(result_ptr, after_data, after_size);
-                       result_ptr += after_size;
-                       before_ptr += before_size;
+       XMLBuffer xbfr((const char *)patch, patch_len, 0);
+       int32_t len = 0;
+       xbfr.read((char *)&len, sizeof(len));
+       char *result = new char[len];
+       char *rp = result, *bp = (char*)before;
+
+       while( xbfr.itell() < patch_len ) {
+               undo_record urecd;
+               xbfr.read((char*)&urecd, sizeof(urecd));
+               int ofs = rp - result;
+               if( urecd.new_ofs > ofs ) {
+                       int sz = urecd.new_ofs - ofs;
+                       memcpy(rp, bp, sz);
+                       bp += sz;  rp += sz;
                }
-               else {
+               if( urecd.new_len > 0 )
+                       xbfr.read(rp, urecd.new_len);
+               bp += urecd.old_len;  rp += urecd.new_len;
+       }
+
 // All data from before_ptr to end of result buffer is identical
-                       if( result_end - result_ptr > 0 )
-                               memcpy(result_ptr, before_ptr, result_end - result_ptr);
-                       done = 1;
-               }
+       int sz = bp - (char *)before;
+       if( (before_len-=sz) > 0 ) {
+               memcpy(rp, bp, before_len);
+               rp += before_len;
        }
 
-       return result;
+       *result_len = rp - result;
+       return (unsigned char *)result;
 }
 
-
 UndoStackItem::UndoStackItem()
  : ListItem<UndoStackItem>()
 {
@@ -381,127 +209,197 @@ const char* UndoStackItem::get_description()
                return "";
 }
 
-int UndoStackItem::has_data()
+// undo diff
+
+UndoHash::UndoHash(char *txt, int len, UndoHash *nxt)
+{
+       this->txt = txt;
+       this->len = len;
+       this->nxt = nxt;
+       line[va] = line[vb] = -1;
+       occurs[va] = occurs[vb] = 0;
+}
+UndoHash::~UndoHash()
 {
-       return data_size ? 1 : 0;
 }
 
+UndoHashTable::UndoHashTable()
+{
+       for( int i=0; i<hash_sz; ++i ) table[i] = 0;
+       bof = new UndoHash(0, 0, 0);
+       eof = new UndoHash(0, 0, 0);
+}
+UndoHashTable::~UndoHashTable()
+{
+       delete bof;
+       delete eof;
+       for( int i=0; i<hash_sz; ++i ) {
+               for( UndoHash *nxt=0, *hp=table[i]; hp; hp=nxt ) {
+                       nxt = hp->nxt;  delete hp;
+               }
+       }
+}
+
+UndoHash *UndoHashTable::add(char *txt, int len)
+{
+       uint8_t *bp = (uint8_t *)txt;
+       int v = 0;
+       for( int i=len; --i>=0; ++bp ) {
+               v = (v<<1) + *bp;
+               v = (v + (v>>hash_sz2)) & hash_sz1;
+       }
+       UndoHash *hp = table[v];
+       while( hp && strncmp(hp->txt,txt,len) ) hp = hp->nxt;
+       if( !hp ) {
+               hp = new UndoHash(txt, len, table[v]);
+               table[v] = hp;
+       }
+       return hp;
+}
+
+UndoLine::UndoLine(UndoHash *hash)
+{
+       this->txt = 0;  this->len = 0;
+       this->hash = hash;
+       hash->occurs[va] = hash->occurs[vb] = 1;
+}
+
+UndoLine::UndoLine(UndoHashTable *hash, char *txt, int len)
+{
+       this->txt = txt;  this->len = len;
+       this->hash = hash->add(txt, len);
+}
+UndoLine::~UndoLine()
+{
+}
+
+int UndoLine::eq(UndoLine *ln)
+{
+       return hash == ln->hash ? 1 : 0;
+}
+
+void UndoVersion::scan_lines(UndoHashTable *hash, char *sp, char *ep)
+{
+       append(new UndoLine(hash->bof));
+       for( int line=1; sp<ep; ++line ) {
+               char *txt = sp;
+               while( sp<ep && *sp++ != '\n' );
+               UndoLine *ln = new UndoLine(hash, txt, sp-txt);
+               ln->hash->line[ver] = line;
+               ++ln->hash->occurs[ver];
+               append(ln);
+       }
+       append(new UndoLine(hash->eof));
+}
+
+
 void UndoStackItem::set_data(char *data)
 {
-       delete [] this->data;
-       this->data = 0;
-       this->data_size = 0;
+       delete [] this->data;  this->data = 0;
+       this->data_size = 0;   this->key = 0;
+       int data_size = strlen(data)+1;
 
-// Search for key buffer within interval
-       int need_key = 1;
        UndoStackItem *current = this;
-       this->key = 0;
-       for( int i=0; i<UNDO_KEY_INTERVAL && current; ++i ) {
-               if( current->key && current->has_data() ) {
-                       need_key = 0;
-                       break;
+// Search for key buffer within interval
+       for( int i=UNDO_KEY_INTERVAL; --i>=0 && current && !current->key; current=PREVIOUS );
+
+       int need_key = current ? 0 : 1;
+// setting need_key = 1 forces all undo data items to key (not diffd)
+//     int need_key = 1;
+       char *prev = 0;  int prev_size = 0;
+       if( !need_key ) {
+#ifdef UNDO_INCREMENTAL
+               prev = previous->get_data();
+               prev_size = prev ? strlen(prev)+1 : 0;
+#else
+               prev = current->data;
+               prev_size = current->data_size;
+#endif
+               int min_size = data_size/16;
+// if the previous is a sizable incremental diff
+               if( previous && !previous->key &&
+                   previous->data_size > min_size ) {
+//  and this minimum diff distance is a lot different, need key
+                       int dist = abs(data_size - prev_size);
+                       if( abs(dist - previous->data_size) > min_size )
+                               need_key = 1;
                }
-               else
-                       current = PREVIOUS;
        }
-
-       int new_size = strlen(data) + 1;
-
        if( !need_key ) {
-// Reconstruct previous data for difference
-               char *prev_buffer = previous->get_data();
-               int prev_size = prev_buffer ? strlen(prev_buffer) + 1 : 0;
-// Timer timer;
-// timer.update();
-// printf("UndoStackItem::set_data 1\n");
-               this->data = (char*)get_difference_fast((unsigned char*)prev_buffer,
-                       prev_size,
-                       (unsigned char*)data,
-                       new_size,
-                       &this->data_size,
-                       0);
-//printf("UndoStackItem::set_data 2 %jd\n", timer.get_difference());
-
-// Testing
-// FILE *test1 = fopen("/tmp/undo1", "w");
-// fwrite(prev_buffer, prev_size, 1, test1);
-// fclose(test1);
-// FILE *test2 = fopen("/tmp/undo2", "w");
-// fwrite(data, new_size, 1, test2);
-// fclose(test2);
-// FILE *test3 = fopen("/tmp/undo2.diff", "w");
-// fwrite(this->data, this->data_size, 1, test3);
-// fclose(test3);
-//
-//printf("UndoStackItem::set_data 3 %d %d\n", new_size, this->data_size);
-
-// Diff was bigger than original.
-// Happens if a lot of tiny changes happened and the record headers
-// took more space than the changes.
-               if( this->data_size > new_size ) {
-                       delete [] this->data;
-                       this->data_size = 0;
-                       need_key = 1;
+               int bfr_size = BCTEXTLEN;
+               char *bfr = new char[bfr_size];
+               XMLBuffer xbfr(bfr_size, bfr, 1);
+               xbfr.write((char*)&data_size, sizeof(data_size));
+               UndoHashTable hash;
+               UndoVersion alines(va), blines(vb);
+               char *asp = data, *aep = asp + data_size-1;
+               char *bsp = prev, *bep = bsp + prev_size-1;
+               alines.scan_lines(&hash, asp, aep);
+               blines.scan_lines(&hash, bsp, bep);
+// trim suffix
+               int asz = alines.size(), bsz = blines.size();
+               while( asz > 0 && bsz > 0 && alines[asz-1]->eq(blines[bsz-1]) ) {
+                       --asz;  --bsz;
                }
-               else {
-// Reconstruct current data from difference
-                       int test_size;
-                       char *test_buffer = (char*)apply_difference((unsigned char*)prev_buffer,
-                               prev_size,
-                               (unsigned char*)this->data,
-                               this->data_size,
+
+               int ai = 0, bi = 0;
+               while( ai < asz || bi < bsz ) {
+// skip running match
+                       if( alines[ai]->eq(blines[bi]) ) { ++ai;  ++bi;  continue; }
+                       char *adp = alines[ai]->txt, *bdp = blines[bi]->txt;
+// find best unique match
+                       int ma = asz, mb = bsz;
+                       int mx = ma + mb + 1;
+                       for( int ia=ai; ia<asz && ia-ai<mx; ++ia ) {
+                               UndoHash *ah = alines[ia]->hash;
+                               if( ah->occurs[va] != 1 || ah->occurs[vb] != 1 ) continue;
+                               int m = (ah->line[va] - ai) + (ah->line[vb] - bi);
+                               if( m >= mx ) continue;
+                               ma = ah->line[va];  mb = ah->line[vb];  mx = m;
+                       }
+// trim suffix
+                       while( ma > 0 && mb > 0 && alines[ma-1]->eq(blines[mb-1]) ) { --ma;  --mb; }
+                       char *ap = alines[ma]->txt, *bp = blines[mb]->txt;
+                       ai = ma;  bi = mb;
+                       undo_record urecd;
+                       urecd.new_ofs = adp - data;
+                       urecd.new_len = ap - adp;
+                       urecd.old_len = bp - bdp;
+                       xbfr.write((const char*)&urecd, sizeof(urecd));
+                       xbfr.write((const char*)adp, urecd.new_len);
+               }
+
+               int64_t pos = xbfr.otell();
+               if( abs(data_size-prev_size) < pos/2 ) {
+                       this->data = new char[this->data_size = pos];
+                       xbfr.iseek(0);
+                       xbfr.read(this->data, pos);
+#if 1
+                       int test_size = 0;
+                       char *test_data = (char*)apply_difference(
+                               (unsigned char *)prev, prev_size,
+                               (unsigned char *)this->data, this->data_size,
                                &test_size);
-                       if( test_size != new_size ||
-                               memcmp(test_buffer, data, test_size) ) {
-// FILE *test1 = fopen("/tmp/undo1", "w");
-// fwrite(prev_buffer, prev_size, 1, test1);
-// fclose(test1);
-// FILE *test2 = fopen("/tmp/undo2", "w");
-// fwrite(data, new_size, 1, test2);
-// fclose(test2);
-// FILE *test3 = fopen("/tmp/undo2.diff", "w");
-// fwrite(this->data, this->data_size, 1, test3);
-// fclose(test3);
-// FILE *test4 = fopen("/tmp/undo3", "w");
-// fwrite(test_buffer, test_size, 1, test4);
-// fclose(test4);
-
-                               printf("UndoStackItem::set_data: incremental undo failed!\n");
+                       if( test_size != data_size || memcmp(test_data, data, test_size) ) {
+                               printf("UndoStackItem::set_data: *** incremental undo failed!\n");
+                               delete [] this->data;  this->data = 0;  this->data_size = 0;
                                need_key = 1;
-                               delete [] this->data;
-                               this->data_size = 0;
-                               this->data = 0;
                        }
-                       delete [] test_buffer;
-
+                       delete [] test_data;
+#endif
                }
-
-
-               delete [] prev_buffer;
+               else
+                       need_key = 1;
        }
 
+#ifdef UNDO_INCREMENTAL
+       delete [] prev;
+#endif
        if( need_key ) {
                this->key = 1;
-               this->data_size = new_size;
-               this->data = new char[this->data_size];
+               this->data = new char[this->data_size = data_size];
                memcpy(this->data, data, this->data_size);
        }
-       return;
-}
-
-char* UndoStackItem::get_incremental_data()
-{
-       return data;
-}
-
-int UndoStackItem::get_incremental_size()
-{
-       return data_size;
-}
-
-int UndoStackItem::get_size()
-{
-       return data_size;
 }
 
 char* UndoStackItem::get_data()
@@ -521,32 +419,37 @@ char* UndoStackItem::get_data()
                memcpy(result, data, data_size);
                return result;
        }
-
+#ifdef UNDO_INCREMENTAL
 // Get key buffer
        char *current_data = current->get_data();
-       int current_size = current->get_size();
+       int current_size = current->data_size;
        current = NEXT;
        while( current ) {
 // Do incremental updates
                int new_size;
-               char *new_data = (char*)apply_difference((unsigned char*)current_data, current_size,
-                       (unsigned char*)current->get_incremental_data(),
-                       current->get_incremental_size(), &new_size);
+               char *new_data = (char*)apply_difference(
+                       (unsigned char*)current_data, current_size,
+                       (unsigned char*)current->data, current->data_size,
+                       &new_size);
                delete [] current_data;
 
                if( current == this )
                        return new_data;
-               else {
-                       current_data = new_data;
-                       current_size = new_size;
-                       current = NEXT;
-               }
+               current_data = new_data;
+               current_size = new_size;
+               current = NEXT;
        }
-
 // Never hit this object.
        delete [] current_data;
        printf("UndoStackItem::get_data: lost starting object!\n");
        return 0;
+#else
+       int new_size;
+       char *new_data = (char*)apply_difference(
+               (unsigned char*)current->data, current->data_size,
+               (unsigned char*)this->data, this->data_size, &new_size);
+       return new_data;
+#endif
 }
 
 int UndoStackItem::is_key()
@@ -554,6 +457,11 @@ int UndoStackItem::is_key()
        return key;
 }
 
+int UndoStackItem::get_size()
+{
+       return data_size;
+}
+
 void UndoStackItem::set_flags(uint64_t flags)
 {
        this->load_flags = flags;
index f54cb0da34de7fe9392da8c6cfccf6e274e056ca..e6791d0a27d65441a6d3073272f047b22fb15a63 100644 (file)
 #define UNDOLEVELS 500
 #define UNDO_KEY_INTERVAL 100
 
-// The undo stack is a series of key undo buffers and
-// incremental undo buffers.  The incremental buffers
-// store the differences in the most compact way possible:
-// a series of offsets, sizes and values.  This should allow
-// a huge number of undo updates.
+
+#define hash_sz2 8
+#define hash_sz  (1<<hash_sz2)
+#define hash_sz1 (hash_sz-1)
+#define va 0
+#define vb 1
+
+class UndoHash
+{
+public:
+       UndoHash(char *txt, int len, UndoHash *nxt);
+       ~UndoHash();
+
+       UndoHash *nxt;
+       char *txt;  int len;
+       int line[2], occurs[2];
+};
+
+class UndoHashTable
+{
+public:
+       UndoHashTable();
+       ~UndoHashTable();
+       UndoHash *add(char *txt, int len);
+
+       UndoHash *table[hash_sz];
+       UndoHash *bof, *eof;
+};
+
+class UndoLine
+{
+public:
+       UndoLine(UndoHash *hash);
+       UndoLine(UndoHashTable *hash, char *txt, int len);
+       ~UndoLine();
+       int eq(UndoLine *ln);
+
+       char *txt;  int len;
+       UndoHash *hash;
+};
+
+class UndoVersion : public ArrayList<UndoLine *>
+{
+public:
+       UndoVersion(int v) { ver = v; }
+       ~UndoVersion() { remove_all_objects(); }
+       int ver;
+
+       void scan_lines(UndoHashTable *hash, char *sp, char *ep);
+};
 
 
 class UndoStackItem : public ListItem<UndoStackItem>
@@ -56,16 +101,10 @@ public:
 // The string must be deleted by the user.
        char* get_data();
        char* get_filename();
-       int has_data();
        int get_size();
        int is_key();
        uint64_t get_flags();
 
-
-// Get pointer to incremental data for use in an apply_difference command.
-       char* get_incremental_data();
-       int get_incremental_size();
-
        void set_creator(void *creator);
        void* get_creator();
 
index 2574499bd160fa3f9e98bd2f66357f0f067c8153..451c71320e3be4f4794fb9483b9c89f1abbed6a4 100644 (file)
@@ -414,15 +414,21 @@ void VWindow::unset_inoutpoint()
        }
 }
 
-void VWindow::copy()
+void VWindow::copy(int all)
 {
        EDL *edl = get_edl();
        if(edl)
        {
-               double start = edl->local_session->get_selectionstart();
-               double end = edl->local_session->get_selectionend();
+               double start = all ? 0 :
+                       edl->local_session->get_selectionstart();
+               double end = all ? edl->tracks->total_length() :
+                       edl->local_session->get_selectionend();
+               EDL *copy_edl = new EDL; // no parent or assets wont be copied
+               copy_edl->create_objects();
+               copy_edl->copy_all(edl);
                FileXML file;
-               edl->copy(start, end, 0, &file, "", 1);
+               copy_edl->copy(start, end, 0, &file, "", 1);
+               copy_edl->remove_user();
                const char *file_string = file.string();
                long file_length = strlen(file_string);
                mwindow->gui->lock_window();
index a858d5c0e22ab3b095bb65379fdb7ceffca94cad..d2a789a4e01659635376560ffcc06a5cdb51f9bf 100644 (file)
@@ -70,7 +70,7 @@ public:
        void set_inpoint();
        void set_outpoint();
        void unset_inoutpoint();
-       void copy();
+       void copy(int all);
        void splice_selection();
        void overwrite_selection();
        void delete_source(int do_main_edl, int update_gui);
index 58f0ef946fd806c64061ae4219f5834f51348614..f446a0f475f758b70bae1c8e72cdc7ff7e10846d 100644 (file)
@@ -567,7 +567,7 @@ VWindowEditing::~VWindowEditing()
 
 void VWindowEditing::copy_selection()
 {
-       vwindow->copy();
+       vwindow->copy(subwindow->shift_down());
 }
 
 void VWindowEditing::splice_selection()
@@ -575,7 +575,7 @@ void VWindowEditing::splice_selection()
        if(vwindow->get_edl())
        {
                mwindow->gui->lock_window("VWindowEditing::splice_selection");
-               mwindow->splice(vwindow->get_edl());
+               mwindow->splice(vwindow->get_edl(), subwindow->shift_down());
                mwindow->gui->unlock_window();
        }
 }
@@ -585,7 +585,7 @@ void VWindowEditing::overwrite_selection()
        if(vwindow->get_edl())
        {
                mwindow->gui->lock_window("VWindowEditing::overwrite_selection");
-               mwindow->overwrite(vwindow->get_edl());
+               mwindow->overwrite(vwindow->get_edl(), subwindow->shift_down());
                mwindow->gui->unlock_window();
        }
 }
@@ -703,7 +703,7 @@ void VWindowEditing::to_clip()
 {
        EDL *edl = vwindow->get_edl();
        if( !edl ) return;
-       mwindow->to_clip(edl, _("viewer window: "));
+       mwindow->to_clip(edl, _("viewer window: "), subwindow->shift_down());
 }
 
 VWindowSource::VWindowSource(MWindow *mwindow, VWindowGUI *vwindow, int x, int y)
index d28e3e9e1c098a83e4b780400bb25476b287fe8f..bada1625e68e7fff709ccd62edaa0ead71857325 100644 (file)
@@ -142,7 +142,7 @@ int ZWindowGUI::button_press_event()
                        local_session->set_selectionend(end);
                        local_session->set_inpoint(start);
                        local_session->set_outpoint(end);
-                       mwindow->overwrite(zwindow->edl);
+                       mwindow->overwrite(zwindow->edl, 0);
                        local_session->set_inpoint(orig_inpoint);
                        local_session->set_outpoint(orig_outpoint);
                        mwindow->gui->update_timebar(1);