hakto 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (48) hide show
  1. data/LICENSE +7 -0
  2. data/README.md +91 -0
  3. data/README_jp.md +92 -0
  4. data/Rakefile +18 -0
  5. data/doc/Hakto/SafeSDBM.html +318 -0
  6. data/doc/Hakto.html +139 -0
  7. data/doc/Object.html +143 -0
  8. data/doc/Rakefile.html +96 -0
  9. data/doc/SafeSDBMTest.html +591 -0
  10. data/doc/created.rid +4 -0
  11. data/doc/images/add.png +0 -0
  12. data/doc/images/brick.png +0 -0
  13. data/doc/images/brick_link.png +0 -0
  14. data/doc/images/bug.png +0 -0
  15. data/doc/images/bullet_black.png +0 -0
  16. data/doc/images/bullet_toggle_minus.png +0 -0
  17. data/doc/images/bullet_toggle_plus.png +0 -0
  18. data/doc/images/date.png +0 -0
  19. data/doc/images/delete.png +0 -0
  20. data/doc/images/find.png +0 -0
  21. data/doc/images/loadingAnimation.gif +0 -0
  22. data/doc/images/macFFBgHack.png +0 -0
  23. data/doc/images/package.png +0 -0
  24. data/doc/images/page_green.png +0 -0
  25. data/doc/images/page_white_text.png +0 -0
  26. data/doc/images/page_white_width.png +0 -0
  27. data/doc/images/plugin.png +0 -0
  28. data/doc/images/ruby.png +0 -0
  29. data/doc/images/tag_blue.png +0 -0
  30. data/doc/images/tag_green.png +0 -0
  31. data/doc/images/transparent.png +0 -0
  32. data/doc/images/wrench.png +0 -0
  33. data/doc/images/wrench_orange.png +0 -0
  34. data/doc/images/zoom.png +0 -0
  35. data/doc/index.html +84 -0
  36. data/doc/js/darkfish.js +153 -0
  37. data/doc/js/jquery.js +18 -0
  38. data/doc/js/navigation.js +142 -0
  39. data/doc/js/search.js +94 -0
  40. data/doc/js/search_index.js +1 -0
  41. data/doc/js/searcher.js +228 -0
  42. data/doc/rdoc.css +543 -0
  43. data/doc/table_of_contents.html +96 -0
  44. data/lib/hakto/safe_sdbm.rb +75 -0
  45. data/test/tb_safe_sdbm.rb +94 -0
  46. data/test/test.sdb.dir +0 -0
  47. data/test/test.sdb.pag +0 -0
  48. metadata +113 -0
data/doc/js/search.js ADDED
@@ -0,0 +1,94 @@
1
+ Search = function(data, input, result) {
2
+ this.data = data;
3
+ this.$input = $(input);
4
+ this.$result = $(result);
5
+
6
+ this.$current = null;
7
+ this.$view = this.$result.parent();
8
+ this.searcher = new Searcher(data.index);
9
+ this.init();
10
+ }
11
+
12
+ Search.prototype = $.extend({}, Navigation, new function() {
13
+ var suid = 1;
14
+
15
+ this.init = function() {
16
+ var _this = this;
17
+ var observer = function() {
18
+ _this.search(_this.$input[0].value);
19
+ };
20
+ this.$input.keyup(observer);
21
+ this.$input.click(observer); // mac's clear field
22
+
23
+ this.searcher.ready(function(results, isLast) {
24
+ _this.addResults(results, isLast);
25
+ })
26
+
27
+ this.initNavigation();
28
+ this.setNavigationActive(false);
29
+ }
30
+
31
+ this.search = function(value, selectFirstMatch) {
32
+ value = jQuery.trim(value).toLowerCase();
33
+ if (value) {
34
+ this.setNavigationActive(true);
35
+ } else {
36
+ this.setNavigationActive(false);
37
+ }
38
+
39
+ if (value == '') {
40
+ this.lastQuery = value;
41
+ this.$result.empty();
42
+ this.setNavigationActive(false);
43
+ } else if (value != this.lastQuery) {
44
+ this.lastQuery = value;
45
+ this.firstRun = true;
46
+ this.searcher.find(value);
47
+ }
48
+ }
49
+
50
+ this.addResults = function(results, isLast) {
51
+ var target = this.$result.get(0);
52
+ if (this.firstRun && (results.length > 0 || isLast)) {
53
+ this.$current = null;
54
+ this.$result.empty();
55
+ }
56
+
57
+ for (var i=0, l = results.length; i < l; i++) {
58
+ target.appendChild(this.renderItem.call(this, results[i]));
59
+ };
60
+
61
+ if (this.firstRun && results.length > 0) {
62
+ this.firstRun = false;
63
+ this.$current = $(target.firstChild);
64
+ this.$current.addClass('current');
65
+ }
66
+ if (jQuery.browser.msie) this.$element[0].className += '';
67
+ }
68
+
69
+ this.move = function(isDown) {
70
+ if (!this.$current) return;
71
+ var $next = this.$current[isDown ? 'next' : 'prev']();
72
+ if ($next.length) {
73
+ this.$current.removeClass('current');
74
+ $next.addClass('current');
75
+ this.scrollIntoView($next[0], this.$view[0]);
76
+ this.$current = $next;
77
+ }
78
+ return true;
79
+ }
80
+
81
+ this.hlt = function(html) {
82
+ return this.escapeHTML(html).
83
+ replace(/\u0001/g, '<em>').
84
+ replace(/\u0002/g, '</em>');
85
+ }
86
+
87
+ this.escapeHTML = function(html) {
88
+ return html.replace(/[&<>]/g, function(c) {
89
+ return '&#' + c.charCodeAt(0) + ';';
90
+ });
91
+ }
92
+
93
+ });
94
+
@@ -0,0 +1 @@
1
+ var search_data = {"index":{"searchIndex":["hakto","safesdbm","object","safesdbmtest","capture_stdout()","new()","print_each()","print_keys()","print_values()","setup()","test_add_get()","test_clear()","test_db_path()","test_delete()","test_each()","test_keys()","test_length()","test_not_have_close_method()","test_print_each()","test_print_keys()","test_print_values()","rakefile"],"longSearchIndex":["hakto","hakto::safesdbm","object","safesdbmtest","safesdbmtest#capture_stdout()","hakto::safesdbm::new()","hakto::safesdbm#print_each()","hakto::safesdbm#print_keys()","hakto::safesdbm#print_values()","safesdbmtest#setup()","safesdbmtest#test_add_get()","safesdbmtest#test_clear()","safesdbmtest#test_db_path()","safesdbmtest#test_delete()","safesdbmtest#test_each()","safesdbmtest#test_keys()","safesdbmtest#test_length()","safesdbmtest#test_not_have_close_method()","safesdbmtest#test_print_each()","safesdbmtest#test_print_keys()","safesdbmtest#test_print_values()",""],"info":[["Hakto","","Hakto.html","","<p>Hakto Safe SDBM Wrapper is a safe wrapper of SDBM class.\n<p>Author &mdash; Moza USANE (mozamimy@quellencode.org) …\n"],["Hakto::SafeSDBM","","Hakto/SafeSDBM.html","","<p>Wrapper of SDBM class. It provides safe access to SDBM database. You can\nuse almost instance methods …\n"],["Object","","Object.html","",""],["SafeSDBMTest","","SafeSDBMTest.html","",""],["capture_stdout","SafeSDBMTest","SafeSDBMTest.html#method-i-capture_stdout","(&block)",""],["new","Hakto::SafeSDBM","Hakto/SafeSDBM.html#method-c-new","(db_path)","<p>Initialize SafeSDBM.\n<p>Args\n<p>db_path &mdash; A filepath of DB file\n"],["print_each","Hakto::SafeSDBM","Hakto/SafeSDBM.html#method-i-print_each","()","<p>Print key-value pair with following format.\n<p>[key1]:value1\n<p>[key2]:value2\n"],["print_keys","Hakto::SafeSDBM","Hakto/SafeSDBM.html#method-i-print_keys","()","<p>Print keys with following format. key1, key2, key3, …\n"],["print_values","Hakto::SafeSDBM","Hakto/SafeSDBM.html#method-i-print_values","()","<p>Print values with following format. value1, value2, value3, …\n"],["setup","SafeSDBMTest","SafeSDBMTest.html#method-i-setup","()",""],["test_add_get","SafeSDBMTest","SafeSDBMTest.html#method-i-test_add_get","()",""],["test_clear","SafeSDBMTest","SafeSDBMTest.html#method-i-test_clear","()",""],["test_db_path","SafeSDBMTest","SafeSDBMTest.html#method-i-test_db_path","()",""],["test_delete","SafeSDBMTest","SafeSDBMTest.html#method-i-test_delete","()",""],["test_each","SafeSDBMTest","SafeSDBMTest.html#method-i-test_each","()",""],["test_keys","SafeSDBMTest","SafeSDBMTest.html#method-i-test_keys","()",""],["test_length","SafeSDBMTest","SafeSDBMTest.html#method-i-test_length","()",""],["test_not_have_close_method","SafeSDBMTest","SafeSDBMTest.html#method-i-test_not_have_close_method","()",""],["test_print_each","SafeSDBMTest","SafeSDBMTest.html#method-i-test_print_each","()",""],["test_print_keys","SafeSDBMTest","SafeSDBMTest.html#method-i-test_print_keys","()",""],["test_print_values","SafeSDBMTest","SafeSDBMTest.html#method-i-test_print_values","()",""],["Rakefile","","Rakefile.html","","<p>require “rake/gempackagetask”\n<p>spec = Gem::Specification.new do |s|\n\n<pre>s.name = &quot;hakto&quot;\ns.summary ...</pre>\n"]]}}
@@ -0,0 +1,228 @@
1
+ Searcher = function(data) {
2
+ this.data = data;
3
+ this.handlers = [];
4
+ }
5
+
6
+ Searcher.prototype = new function() {
7
+ // search is performed in chunks of 1000 for non-blocking user input
8
+ var CHUNK_SIZE = 1000;
9
+ // do not try to find more than 100 results
10
+ var MAX_RESULTS = 100;
11
+ var huid = 1;
12
+ var suid = 1;
13
+ var runs = 0;
14
+
15
+ this.find = function(query) {
16
+ var queries = splitQuery(query);
17
+ var regexps = buildRegexps(queries);
18
+ var highlighters = buildHilighters(queries);
19
+ var state = { from: 0, pass: 0, limit: MAX_RESULTS, n: suid++};
20
+ var _this = this;
21
+
22
+ this.currentSuid = state.n;
23
+
24
+ if (!query) return;
25
+
26
+ var run = function() {
27
+ // stop current search thread if new search started
28
+ if (state.n != _this.currentSuid) return;
29
+
30
+ var results =
31
+ performSearch(_this.data, regexps, queries, highlighters, state);
32
+ var hasMore = (state.limit > 0 && state.pass < 4);
33
+
34
+ triggerResults.call(_this, results, !hasMore);
35
+ if (hasMore) {
36
+ setTimeout(run, 2);
37
+ }
38
+ runs++;
39
+ };
40
+ runs = 0;
41
+
42
+ // start search thread
43
+ run();
44
+ }
45
+
46
+ /* ----- Events ------ */
47
+ this.ready = function(fn) {
48
+ fn.huid = huid;
49
+ this.handlers.push(fn);
50
+ }
51
+
52
+ /* ----- Utilities ------ */
53
+ function splitQuery(query) {
54
+ return jQuery.grep(query.split(/(\s+|::?|\(\)?)/), function(string) {
55
+ return string.match(/\S/)
56
+ });
57
+ }
58
+
59
+ function buildRegexps(queries) {
60
+ return jQuery.map(queries, function(query) {
61
+ return new RegExp(query.replace(/(.)/g, '([$1])([^$1]*?)'), 'i')
62
+ });
63
+ }
64
+
65
+ function buildHilighters(queries) {
66
+ return jQuery.map(queries, function(query) {
67
+ return jQuery.map(query.split(''), function(l, i) {
68
+ return '\u0001$' + (i*2+1) + '\u0002$' + (i*2+2);
69
+ }).join('');
70
+ });
71
+ }
72
+
73
+ // function longMatchRegexp(index, longIndex, regexps) {
74
+ // for (var i = regexps.length - 1; i >= 0; i--){
75
+ // if (!index.match(regexps[i]) && !longIndex.match(regexps[i])) return false;
76
+ // };
77
+ // return true;
78
+ // }
79
+
80
+
81
+ /* ----- Mathchers ------ */
82
+
83
+ /*
84
+ * This record matches if the index starts with queries[0] and the record
85
+ * matches all of the regexps
86
+ */
87
+ function matchPassBeginning(index, longIndex, queries, regexps) {
88
+ if (index.indexOf(queries[0]) != 0) return false;
89
+ for (var i=1, l = regexps.length; i < l; i++) {
90
+ if (!index.match(regexps[i]) && !longIndex.match(regexps[i]))
91
+ return false;
92
+ };
93
+ return true;
94
+ }
95
+
96
+ /*
97
+ * This record matches if the longIndex starts with queries[0] and the
98
+ * longIndex matches all of the regexps
99
+ */
100
+ function matchPassLongIndex(index, longIndex, queries, regexps) {
101
+ if (longIndex.indexOf(queries[0]) != 0) return false;
102
+ for (var i=1, l = regexps.length; i < l; i++) {
103
+ if (!longIndex.match(regexps[i]))
104
+ return false;
105
+ };
106
+ return true;
107
+ }
108
+
109
+ /*
110
+ * This record matches if the index contains queries[0] and the record
111
+ * matches all of the regexps
112
+ */
113
+ function matchPassContains(index, longIndex, queries, regexps) {
114
+ if (index.indexOf(queries[0]) == -1) return false;
115
+ for (var i=1, l = regexps.length; i < l; i++) {
116
+ if (!index.match(regexps[i]) && !longIndex.match(regexps[i]))
117
+ return false;
118
+ };
119
+ return true;
120
+ }
121
+
122
+ /*
123
+ * This record matches if regexps[0] matches the index and the record
124
+ * matches all of the regexps
125
+ */
126
+ function matchPassRegexp(index, longIndex, queries, regexps) {
127
+ if (!index.match(regexps[0])) return false;
128
+ for (var i=1, l = regexps.length; i < l; i++) {
129
+ if (!index.match(regexps[i]) && !longIndex.match(regexps[i]))
130
+ return false;
131
+ };
132
+ return true;
133
+ }
134
+
135
+
136
+ /* ----- Highlighters ------ */
137
+ function highlightRegexp(info, queries, regexps, highlighters) {
138
+ var result = createResult(info);
139
+ for (var i=0, l = regexps.length; i < l; i++) {
140
+ result.title = result.title.replace(regexps[i], highlighters[i]);
141
+ result.namespace = result.namespace.replace(regexps[i], highlighters[i]);
142
+ };
143
+ return result;
144
+ }
145
+
146
+ function hltSubstring(string, pos, length) {
147
+ return string.substring(0, pos) + '\u0001' + string.substring(pos, pos + length) + '\u0002' + string.substring(pos + length);
148
+ }
149
+
150
+ function highlightQuery(info, queries, regexps, highlighters) {
151
+ var result = createResult(info);
152
+ var pos = 0;
153
+ var lcTitle = result.title.toLowerCase();
154
+
155
+ pos = lcTitle.indexOf(queries[0]);
156
+ if (pos != -1) {
157
+ result.title = hltSubstring(result.title, pos, queries[0].length);
158
+ }
159
+
160
+ result.namespace = result.namespace.replace(regexps[0], highlighters[0]);
161
+ for (var i=1, l = regexps.length; i < l; i++) {
162
+ result.title = result.title.replace(regexps[i], highlighters[i]);
163
+ result.namespace = result.namespace.replace(regexps[i], highlighters[i]);
164
+ };
165
+ return result;
166
+ }
167
+
168
+ function createResult(info) {
169
+ var result = {};
170
+ result.title = info[0];
171
+ result.namespace = info[1];
172
+ result.path = info[2];
173
+ result.params = info[3];
174
+ result.snippet = info[4];
175
+ return result;
176
+ }
177
+
178
+ /* ----- Searching ------ */
179
+ function performSearch(data, regexps, queries, highlighters, state) {
180
+ var searchIndex = data.searchIndex;
181
+ var longSearchIndex = data.longSearchIndex;
182
+ var info = data.info;
183
+ var result = [];
184
+ var i = state.from;
185
+ var l = searchIndex.length;
186
+ var togo = CHUNK_SIZE;
187
+ var matchFunc, hltFunc;
188
+
189
+ while (state.pass < 4 && state.limit > 0 && togo > 0) {
190
+ if (state.pass == 0) {
191
+ matchFunc = matchPassBeginning;
192
+ hltFunc = highlightQuery;
193
+ } else if (state.pass == 1) {
194
+ matchFunc = matchPassLongIndex;
195
+ hltFunc = highlightQuery;
196
+ } else if (state.pass == 2) {
197
+ matchFunc = matchPassContains;
198
+ hltFunc = highlightQuery;
199
+ } else if (state.pass == 3) {
200
+ matchFunc = matchPassRegexp;
201
+ hltFunc = highlightRegexp;
202
+ }
203
+
204
+ for (; togo > 0 && i < l && state.limit > 0; i++, togo--) {
205
+ if (info[i].n == state.n) continue;
206
+ if (matchFunc(searchIndex[i], longSearchIndex[i], queries, regexps)) {
207
+ info[i].n = state.n;
208
+ result.push(hltFunc(info[i], queries, regexps, highlighters));
209
+ state.limit--;
210
+ }
211
+ };
212
+ if (searchIndex.length <= i) {
213
+ state.pass++;
214
+ i = state.from = 0;
215
+ } else {
216
+ state.from = i;
217
+ }
218
+ }
219
+ return result;
220
+ }
221
+
222
+ function triggerResults(results, isLast) {
223
+ jQuery.each(this.handlers, function(i, fn) {
224
+ fn.call(this, results, isLast)
225
+ })
226
+ }
227
+ }
228
+