1 | // =============================================================== // |
---|
2 | // // |
---|
3 | // File : NT_sync_scroll.cxx // |
---|
4 | // Purpose : Sync tree scrolling // |
---|
5 | // // |
---|
6 | // Coded by Ralf Westram (coder@reallysoft.de) in October 2016 // |
---|
7 | // http://www.arb-home.de/ // |
---|
8 | // // |
---|
9 | // =============================================================== // |
---|
10 | |
---|
11 | #include "ScrollSynchronizer.h" |
---|
12 | #include "NT_sync_scroll.h" |
---|
13 | |
---|
14 | #include <TreeDisplay.hxx> |
---|
15 | |
---|
16 | #include <aw_msg.hxx> |
---|
17 | #include <aw_root.hxx> |
---|
18 | #include <aw_awar.hxx> |
---|
19 | #include <aw_select.hxx> |
---|
20 | |
---|
21 | #define AWAR_TEMPL_SYNCED_WITH_WINDOW "tmp/sync%i/with" |
---|
22 | #define AWAR_TEMPL_AUTO_SYNCED "tmp/sync%i/auto" |
---|
23 | |
---|
24 | #define MAX_AWARNAME_LENGTH (3+1+5+1+4) |
---|
25 | |
---|
26 | static ScrollSynchronizer synchronizer; |
---|
27 | |
---|
28 | inline const char *awarname(const char *awarname_template, int idx) { |
---|
29 | nt_assert(idx>=0 && idx<MAX_NT_WINDOWS); |
---|
30 | static char buffer[MAX_AWARNAME_LENGTH+1]; |
---|
31 | |
---|
32 | #if defined(ASSERTION_USED) |
---|
33 | int printed = |
---|
34 | #endif |
---|
35 | sprintf(buffer, awarname_template, idx); |
---|
36 | nt_assert(printed<=MAX_AWARNAME_LENGTH); |
---|
37 | |
---|
38 | return buffer; |
---|
39 | } |
---|
40 | |
---|
41 | static void refill_syncWithList_cb(AW_root *, AW_selection_list *sellst, TREE_canvas *ntw) { |
---|
42 | sellst->clear(); |
---|
43 | NT_fill_canvas_selection_list(sellst, ntw); |
---|
44 | sellst->insert_default("<don't sync>", NO_SCROLL_SYNC); |
---|
45 | sellst->update(); |
---|
46 | } |
---|
47 | |
---|
48 | static unsigned auto_refresh_cb(AW_root*) { |
---|
49 | synchronizer.auto_update(); |
---|
50 | return 0; // do not call again |
---|
51 | } |
---|
52 | |
---|
53 | static void canvas_updated_cb(AWT_canvas *ntw, AW_CL /*cd*/) { |
---|
54 | TREE_canvas *tw = DOWNCAST(TREE_canvas*, ntw); |
---|
55 | synchronizer.announce_update(tw->get_index()); |
---|
56 | |
---|
57 | AW_root *awr = ntw->awr; |
---|
58 | awr->add_timed_callback(250, makeTimedCallback(auto_refresh_cb)); |
---|
59 | } |
---|
60 | |
---|
61 | static void sync_changed_cb(AW_root *awr, int slave_idx) { |
---|
62 | AW_awar *awar_sync_with = awr->awar(awarname(AWAR_TEMPL_SYNCED_WITH_WINDOW, slave_idx)); |
---|
63 | AW_awar *awar_autosync = awr->awar(awarname(AWAR_TEMPL_AUTO_SYNCED, slave_idx)); |
---|
64 | |
---|
65 | int master_idx = awar_sync_with->read_int(); |
---|
66 | bool autosync = awar_autosync->read_int(); |
---|
67 | |
---|
68 | if (valid_canvas_index(master_idx)) { |
---|
69 | TREE_canvas *master = NT_get_canvas_by_index(master_idx); |
---|
70 | master->at_screen_update_call(canvas_updated_cb, 0); |
---|
71 | } |
---|
72 | |
---|
73 | GB_ERROR error = synchronizer.define_dependency(slave_idx, master_idx, autosync); |
---|
74 | nt_assert(implicated(error, autosync)); // error may only occur if autosync is ON |
---|
75 | |
---|
76 | if (error) { |
---|
77 | aw_message(GBS_global_string("Auto-sync turned off (%s)", error)); |
---|
78 | awar_autosync->write_int(0); |
---|
79 | } |
---|
80 | else { |
---|
81 | synchronizer.update_implicit(slave_idx); |
---|
82 | } |
---|
83 | } |
---|
84 | |
---|
85 | static void explicit_scroll_sync_cb(AW_window*, int tgt_idx) { |
---|
86 | aw_message_if(synchronizer.update_explicit(tgt_idx)); |
---|
87 | } |
---|
88 | |
---|
89 | AW_window *NT_create_syncScroll_window(AW_root *awr, TREE_canvas *ntw) { |
---|
90 | AW_window_simple *aws = new AW_window_simple; |
---|
91 | |
---|
92 | int ntw_idx = ntw->get_index(); |
---|
93 | nt_assert(ntw_idx>=0); |
---|
94 | |
---|
95 | AW_awar *awar_sync_with = awr->awar_int(awarname(AWAR_TEMPL_SYNCED_WITH_WINDOW, ntw_idx), NO_SCROLL_SYNC, AW_ROOT_DEFAULT); |
---|
96 | AW_awar *awar_autosync = awr->awar_int(awarname(AWAR_TEMPL_AUTO_SYNCED, ntw_idx), 0, AW_ROOT_DEFAULT); |
---|
97 | |
---|
98 | { |
---|
99 | char *wid = GBS_global_string_copy("SYNC_TREE_SCROLL_%i", ntw_idx); |
---|
100 | aws->init(awr, wid, |
---|
101 | ntw_idx>0 |
---|
102 | ? GBS_global_string("Synchronize tree scrolling (%i)", ntw_idx) |
---|
103 | : "Synchronize tree scrolling"); |
---|
104 | free(wid); |
---|
105 | } |
---|
106 | aws->load_xfig("syncscroll.fig"); |
---|
107 | |
---|
108 | aws->at("close"); |
---|
109 | aws->callback(AW_POPDOWN); |
---|
110 | aws->create_button("CLOSE", "CLOSE", "C"); |
---|
111 | |
---|
112 | aws->at("help"); |
---|
113 | aws->callback(makeHelpCallback("syncscroll.hlp")); |
---|
114 | aws->create_button("HELP", "HELP", "H"); |
---|
115 | |
---|
116 | RootCallback sync_changed_rcb = makeRootCallback(sync_changed_cb, ntw_idx); |
---|
117 | { |
---|
118 | aws->at("box"); |
---|
119 | AW_selection_list *sellist = aws->create_selection_list(awar_sync_with->awar_name, true); |
---|
120 | |
---|
121 | // refill list when new main-windows are created: |
---|
122 | RootCallback refill_sellist_cb = makeRootCallback(refill_syncWithList_cb, sellist, ntw); |
---|
123 | awr->awar(AWAR_NTREE_MAIN_WINDOW_COUNT)->add_callback(refill_sellist_cb); |
---|
124 | refill_sellist_cb(awr); |
---|
125 | |
---|
126 | awar_sync_with->add_callback(sync_changed_rcb); // resync when source changes |
---|
127 | } |
---|
128 | |
---|
129 | aws->at("dosync"); |
---|
130 | aws->callback(makeWindowCallback(explicit_scroll_sync_cb, ntw_idx)); |
---|
131 | aws->create_autosize_button("SYNC_SCROLL", "Sync scroll"); |
---|
132 | |
---|
133 | aws->at("autosync"); |
---|
134 | aws->label("Auto-sync?"); |
---|
135 | aws->create_toggle(awar_autosync->awar_name); |
---|
136 | |
---|
137 | awar_autosync->add_callback(sync_changed_rcb); // resync when auto-sync changes (master may have been scrolled since last sync) |
---|
138 | |
---|
139 | return aws; |
---|
140 | } |
---|
141 | |
---|
142 | |
---|