hakto 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/LICENSE +7 -0
- data/README.md +91 -0
- data/README_jp.md +92 -0
- data/Rakefile +18 -0
- data/doc/Hakto/SafeSDBM.html +318 -0
- data/doc/Hakto.html +139 -0
- data/doc/Object.html +143 -0
- data/doc/Rakefile.html +96 -0
- data/doc/SafeSDBMTest.html +591 -0
- data/doc/created.rid +4 -0
- data/doc/images/add.png +0 -0
- data/doc/images/brick.png +0 -0
- data/doc/images/brick_link.png +0 -0
- data/doc/images/bug.png +0 -0
- data/doc/images/bullet_black.png +0 -0
- data/doc/images/bullet_toggle_minus.png +0 -0
- data/doc/images/bullet_toggle_plus.png +0 -0
- data/doc/images/date.png +0 -0
- data/doc/images/delete.png +0 -0
- data/doc/images/find.png +0 -0
- data/doc/images/loadingAnimation.gif +0 -0
- data/doc/images/macFFBgHack.png +0 -0
- data/doc/images/package.png +0 -0
- data/doc/images/page_green.png +0 -0
- data/doc/images/page_white_text.png +0 -0
- data/doc/images/page_white_width.png +0 -0
- data/doc/images/plugin.png +0 -0
- data/doc/images/ruby.png +0 -0
- data/doc/images/tag_blue.png +0 -0
- data/doc/images/tag_green.png +0 -0
- data/doc/images/transparent.png +0 -0
- data/doc/images/wrench.png +0 -0
- data/doc/images/wrench_orange.png +0 -0
- data/doc/images/zoom.png +0 -0
- data/doc/index.html +84 -0
- data/doc/js/darkfish.js +153 -0
- data/doc/js/jquery.js +18 -0
- data/doc/js/navigation.js +142 -0
- data/doc/js/search.js +94 -0
- data/doc/js/search_index.js +1 -0
- data/doc/js/searcher.js +228 -0
- data/doc/rdoc.css +543 -0
- data/doc/table_of_contents.html +96 -0
- data/lib/hakto/safe_sdbm.rb +75 -0
- data/test/tb_safe_sdbm.rb +94 -0
- data/test/test.sdb.dir +0 -0
- data/test/test.sdb.pag +0 -0
- 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 — 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 — 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 = "hakto"\ns.summary ...</pre>\n"]]}}
|
data/doc/js/searcher.js
ADDED
@@ -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
|
+
|