├── .gitignore
├── css
├── icon.png
├── next.png
├── prev.png
├── trash.png
├── throbber.gif
├── arrow_down.png
├── icon_thumb.png
├── icon_tiny.png
├── throbber-2.gif
├── normalize.css
└── history.css
├── release.sh
├── .jshintrc
├── manifest.json
├── tests
├── tests.html
├── mocha.css
├── query_parser.js
├── assert.js
└── expect.js
├── LICENSE.txt
├── lib
├── query_parser.js
├── util.js
├── visitsbyday.js
├── view.js
├── model.js
├── ehistory.js
└── controller.js
├── README.md
├── history.html
└── vendor
├── spin.js
└── mustache.js
/.gitignore:
--------------------------------------------------------------------------------
1 | .DS_STORE
--------------------------------------------------------------------------------
/css/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/amasad/eHistory/HEAD/css/icon.png
--------------------------------------------------------------------------------
/css/next.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/amasad/eHistory/HEAD/css/next.png
--------------------------------------------------------------------------------
/css/prev.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/amasad/eHistory/HEAD/css/prev.png
--------------------------------------------------------------------------------
/css/trash.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/amasad/eHistory/HEAD/css/trash.png
--------------------------------------------------------------------------------
/css/throbber.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/amasad/eHistory/HEAD/css/throbber.gif
--------------------------------------------------------------------------------
/css/arrow_down.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/amasad/eHistory/HEAD/css/arrow_down.png
--------------------------------------------------------------------------------
/css/icon_thumb.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/amasad/eHistory/HEAD/css/icon_thumb.png
--------------------------------------------------------------------------------
/css/icon_tiny.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/amasad/eHistory/HEAD/css/icon_tiny.png
--------------------------------------------------------------------------------
/css/throbber-2.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/amasad/eHistory/HEAD/css/throbber-2.gif
--------------------------------------------------------------------------------
/release.sh:
--------------------------------------------------------------------------------
1 | cur=`pwd`
2 | tmp=ehistory_release_tmp
3 | cd ..
4 | rm -rf $tmp
5 | cp -r $cur $tmp
6 | cd $tmp
7 | rm -rf .git
8 | rm release.sh
9 | cd ..
10 | zip -r ehistory_release.zip $tmp/*
11 | rm -rf $tmp
--------------------------------------------------------------------------------
/.jshintrc:
--------------------------------------------------------------------------------
1 | { "eqeqeq": true
2 | , "latedef": true
3 | , "newcap": true
4 | , "noarg": true
5 | , "sub": true
6 | , "undef": true
7 | , "boss": true
8 | , "eqnull": true
9 | , "browser": true
10 | , "laxcomma": true
11 | , "laxbreak": true
12 | , "strict": false
13 | , "asi": true
14 | , "unused": true
15 | , "predef": {
16 | "$": false
17 | }
18 | }
19 |
--------------------------------------------------------------------------------
/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "eHistory",
3 | "version": "1.14",
4 | "description": "Enhanced History",
5 | "permissions": [
6 | "history",
7 | "chrome://favicon/"
8 | ],
9 | "chrome_url_overrides" : {
10 | "history": "history.html"
11 | },
12 | "icons": {
13 | "128": "css/icon.png"
14 | },
15 | "manifest_version": 2
16 | }
17 |
--------------------------------------------------------------------------------
/tests/tests.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Mocha
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
19 |
20 |
21 |
--------------------------------------------------------------------------------
/LICENSE.txt:
--------------------------------------------------------------------------------
1 | Copyright (C) <2011> by Amjad Masad
2 |
3 | Permission is hereby granted, free of charge, to any person obtaining a copy
4 | of this software and associated documentation files (the "Software"), to deal
5 | in the Software without restriction, including without limitation the rights
6 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
7 | copies of the Software, and to permit persons to whom the Software is
8 | furnished to do so, subject to the following conditions:
9 |
10 | The above copyright notice and this permission notice shall be included in
11 | all copies or substantial portions of the Software.
12 |
13 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
19 | THE SOFTWARE.
--------------------------------------------------------------------------------
/lib/query_parser.js:
--------------------------------------------------------------------------------
1 | /* parseQuery: Parses the search query
2 | * @arg (String) input: The search query
3 | * @returns (Array) [settings, filters, text]
4 | */
5 | this.parseQuery = function (query) {
6 | // Custom filters.
7 | var filters = {
8 | inurl: null,
9 | intitle: null,
10 | site: null
11 | };
12 |
13 | // Chrome search object.
14 | var settings = {
15 | startTime: null,
16 | endTime: null
17 | };
18 |
19 | var combined = '';
20 |
21 | // Assumes search query is a space delimted key/value pairs.
22 | query.split(/\s/).forEach(function (pair) {
23 | if (!pair) return;
24 | // Assume key:value
25 | pair = pair.split(':');
26 | if (settings.hasOwnProperty(pair[0])) {
27 | settings[pair[0]] = pair[1];
28 | } else if (filters.hasOwnProperty(pair[0])) {
29 | filters[pair[0]] = pair[1]
30 | combined += ' ' + (pair[1]);
31 | } else {
32 | combined += ' ' + pair.join(':');
33 | }
34 | });
35 |
36 | settings.text = combined.trim();
37 |
38 | // TODO: is this needed?
39 | // delete all empty filters
40 | for (var prop in filters) {
41 | if (filters[prop] === null) {
42 | delete filters[prop];
43 | }
44 | }
45 |
46 | return {
47 | settings: settings,
48 | filters: filters
49 | };
50 | };
51 |
--------------------------------------------------------------------------------
/lib/util.js:
--------------------------------------------------------------------------------
1 | this.Util = (function () {
2 | var rProtocol = /^([a-z0-9+]+:)/i,
3 | rPort = /:[0-9]+$/,
4 | // Charecters that are not allowed in a hostname.
5 | nonHostChars = ['<', '>', '"', '`', ' ', '\r', '\n', '\t', '{', '}', '|',
6 | '\\', '^', '~', '[', ']', '`', '%', '/', '?', ';', '#'],
7 | // Protocols that don't have a hostname.
8 | hostlessProtocols = {
9 | 'javascript': true,
10 | 'file': true
11 | };
12 |
13 | // Gets the hostname without the port.
14 | return {
15 | getHostname: function (url) {
16 | var proto = rProtocol.exec(url),
17 | rest = url,
18 | host = '';
19 |
20 | proto = proto ? proto[0] : null;
21 |
22 | if (proto && !hostlessProtocols[proto]) {
23 | // We should have a host.
24 | rest = rest.substr(proto.length + 2);
25 |
26 | // Find the first non host character index in the url string left.
27 | var firstNonHost = -1;
28 | for (var i = 0; i < nonHostChars.length; i++) {
29 | var index = rest.indexOf(nonHostChars[i]);
30 | if (index !== -1 &&
31 | (index < firstNonHost || firstNonHost === -1)) firstNonHost = index;
32 | }
33 |
34 | // Get the actual hostName.
35 | if (firstNonHost !== -1) {
36 | host = rest.substr(0, firstNonHost);
37 | // Remove port.
38 | host = host.replace(rPort, '');
39 | }
40 | }
41 | return host;
42 | }
43 | };
44 |
45 | })();
46 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # eHistory
2 |
3 | This takes the current chrome history to the next level by adding a set of very useful search operators which is inspired by google's search operators.
4 | It also works around many Google Chrome history bugs and makes it easy to use.
5 |
6 | If you are the kind of person spends a lot of time on the web, sooner or later you'll run into a situation where you want to pull up a page you've visited in the past that you can remember so little about. Maybe you remember that it was an article in the New York times. And maybe you remember you saw it two weeks ago. Normally, this kind of information won't be of any help to you. Enter eHistory: It allows you to power search your browser history with a nice and slick interface. It also adds advanced deletion features.
7 |
8 |
9 | ## Filters
10 |
11 | 1. URL.
12 | 2. Title.
13 | 3. Site.
14 | 4. Time of visit.
15 | 5. Content.
16 |
17 | ## Other features:
18 |
19 | * Search and delete individual or multiple Items.
20 | * Easily select full days to delete.
21 | * Clear history.
22 | * Clear all search results.
23 | * Works around some Chrome history bugs.
24 |
25 | ## Example:
26 | I remember reading a NY times article about coffee:
27 | site:nytimes.com intitle:coffee
28 |
29 |
30 | ## Short comings
31 |
32 | Some queries may run slower than others, and this is because all filteration is done in JavaScript. The Chrome History API only takes text field search, nonetheless eHistory is highly optimized and implements certain hacks to get the results as fast as possible.
33 | Another pain point would be that Chrome's history API doesn't expose the snippet which can reveal which part of the page your search string actually matched.
34 |
35 |
--------------------------------------------------------------------------------
/tests/mocha.css:
--------------------------------------------------------------------------------
1 | @charset "utf-8";
2 |
3 | body {
4 | font: 20px/1.5 "Helvetica Neue", Helvetica, Arial, sans-serif;
5 | padding: 60px 50px;
6 | }
7 |
8 | #mocha ul, #mocha li {
9 | margin: 0;
10 | padding: 0;
11 | }
12 |
13 | #mocha ul {
14 | list-style: none;
15 | }
16 |
17 | #mocha h1, #mocha h2 {
18 | margin: 0;
19 | }
20 |
21 | #mocha h1 {
22 | margin-top: 15px;
23 | font-size: 1em;
24 | font-weight: 200;
25 | }
26 |
27 | #mocha h1 a {
28 | text-decoration: none;
29 | color: inherit;
30 | }
31 |
32 | #mocha h1 a:hover {
33 | text-decoration: underline;
34 | }
35 |
36 | #mocha .suite .suite h1 {
37 | margin-top: 0;
38 | font-size: .8em;
39 | }
40 |
41 | .hidden {
42 | display: none;
43 | }
44 |
45 | #mocha h2 {
46 | font-size: 12px;
47 | font-weight: normal;
48 | cursor: pointer;
49 | }
50 |
51 | #mocha .suite {
52 | margin-left: 15px;
53 | }
54 |
55 | #mocha .test {
56 | margin-left: 15px;
57 | overflow: hidden;
58 | }
59 |
60 | #mocha .test.pending:hover h2::after {
61 | content: '(pending)';
62 | font-family: arial;
63 | }
64 |
65 | #mocha .test.pass.medium .duration {
66 | background: #C09853;
67 | }
68 |
69 | #mocha .test.pass.slow .duration {
70 | background: #B94A48;
71 | }
72 |
73 | #mocha .test.pass::before {
74 | content: '✓';
75 | font-size: 12px;
76 | display: block;
77 | float: left;
78 | margin-right: 5px;
79 | color: #00d6b2;
80 | }
81 |
82 | #mocha .test.pass .duration {
83 | font-size: 9px;
84 | margin-left: 5px;
85 | padding: 2px 5px;
86 | color: white;
87 | -webkit-box-shadow: inset 0 1px 1px rgba(0,0,0,.2);
88 | -moz-box-shadow: inset 0 1px 1px rgba(0,0,0,.2);
89 | box-shadow: inset 0 1px 1px rgba(0,0,0,.2);
90 | -webkit-border-radius: 5px;
91 | -moz-border-radius: 5px;
92 | -ms-border-radius: 5px;
93 | -o-border-radius: 5px;
94 | border-radius: 5px;
95 | }
96 |
97 | #mocha .test.pass.fast .duration {
98 | display: none;
99 | }
100 |
101 | #mocha .test.pending {
102 | color: #0b97c4;
103 | }
104 |
105 | #mocha .test.pending::before {
106 | content: '◦';
107 | color: #0b97c4;
108 | }
109 |
110 | #mocha .test.fail {
111 | color: #c00;
112 | }
113 |
114 | #mocha .test.fail pre {
115 | color: black;
116 | }
117 |
118 | #mocha .test.fail::before {
119 | content: '✖';
120 | font-size: 12px;
121 | display: block;
122 | float: left;
123 | margin-right: 5px;
124 | color: #c00;
125 | }
126 |
127 | #mocha .test pre.error {
128 | color: #c00;
129 | max-height: 300px;
130 | overflow: auto;
131 | }
132 |
133 | #mocha .test pre {
134 | display: block;
135 | float: left;
136 | clear: left;
137 | font: 12px/1.5 monaco, monospace;
138 | margin: 5px;
139 | padding: 15px;
140 | border: 1px solid #eee;
141 | border-bottom-color: #ddd;
142 | -webkit-border-radius: 3px;
143 | -webkit-box-shadow: 0 1px 3px #eee;
144 | -moz-border-radius: 3px;
145 | -moz-box-shadow: 0 1px 3px #eee;
146 | }
147 |
148 | #mocha .test h2 {
149 | position: relative;
150 | }
151 |
152 | #mocha .test a.replay {
153 | position: absolute;
154 | top: 3px;
155 | right: 0;
156 | text-decoration: none;
157 | vertical-align: middle;
158 | display: block;
159 | width: 15px;
160 | height: 15px;
161 | line-height: 15px;
162 | text-align: center;
163 | background: #eee;
164 | font-size: 15px;
165 | -moz-border-radius: 15px;
166 | border-radius: 15px;
167 | -webkit-transition: opacity 200ms;
168 | -moz-transition: opacity 200ms;
169 | transition: opacity 200ms;
170 | opacity: 0.3;
171 | color: #888;
172 | }
173 |
174 | #mocha .test:hover a.replay {
175 | opacity: 1;
176 | }
177 |
178 | #mocha-report.pass .test.fail {
179 | display: none;
180 | }
181 |
182 | #mocha-report.fail .test.pass {
183 | display: none;
184 | }
185 |
186 | #mocha-error {
187 | color: #c00;
188 | font-size: 1.5 em;
189 | font-weight: 100;
190 | letter-spacing: 1px;
191 | }
192 |
193 | #mocha-stats {
194 | position: fixed;
195 | top: 15px;
196 | right: 10px;
197 | font-size: 12px;
198 | margin: 0;
199 | color: #888;
200 | }
201 |
202 | #mocha-stats .progress {
203 | float: right;
204 | padding-top: 0;
205 | }
206 |
207 | #mocha-stats em {
208 | color: black;
209 | }
210 |
211 | #mocha-stats a {
212 | text-decoration: none;
213 | color: inherit;
214 | }
215 |
216 | #mocha-stats a:hover {
217 | border-bottom: 1px solid #eee;
218 | }
219 |
220 | #mocha-stats li {
221 | display: inline-block;
222 | margin: 0 5px;
223 | list-style: none;
224 | padding-top: 11px;
225 | }
226 |
227 | code .comment { color: #ddd }
228 | code .init { color: #2F6FAD }
229 | code .string { color: #5890AD }
230 | code .keyword { color: #8A6343 }
231 | code .number { color: #2F6FAD }
232 |
233 | @media screen and (max-device-width: 480px) {
234 | body {
235 | padding: 60px 0px;
236 | }
237 |
238 | #stats {
239 | position: absolute;
240 | }
241 | }
242 |
--------------------------------------------------------------------------------
/tests/query_parser.js:
--------------------------------------------------------------------------------
1 | describe('query parser', function () {
2 |
3 | describe('plain text', function () {
4 |
5 | it('should handle plain text', function () {
6 | var options = parseQuery('foo');
7 | expect(options).to.eql({
8 | settings: {
9 | startTime: null
10 | , endTime: null
11 | , text: 'foo'
12 | }
13 | , filters: {}
14 | });
15 | });
16 |
17 | it('should handle plain text with special chars', function () {
18 | var options = parseQuery('foo + bar');
19 | expect(options).to.eql({
20 | settings: {
21 | startTime: null
22 | , endTime: null
23 | , text: 'foo + bar'
24 | }
25 | , filters: {}
26 | });
27 | });
28 |
29 | });
30 |
31 | describe('filters', function () {
32 |
33 | describe('simple filtes', function () {
34 |
35 | it('should handle title filter', function () {
36 | var options = parseQuery('intitle:wat');
37 | expect(options).to.eql({
38 | settings: {
39 | startTime: null
40 | , endTime: null
41 | , text: 'wat'
42 | }
43 | , filters: {
44 | intitle: 'wat'
45 | }
46 | });
47 | });
48 |
49 | it('should handle site filter', function () {
50 | var options = parseQuery('site:google.com');
51 | expect(options).to.eql({
52 | settings: {
53 | startTime: null
54 | , endTime: null
55 | , text: 'google.com'
56 | }
57 | , filters: {
58 | site: 'google.com'
59 | }
60 | });
61 | });
62 |
63 | it('should handle url filter', function () {
64 | var options = parseQuery('inurl:foobar');
65 | expect(options).to.eql({
66 | settings: {
67 | startTime: null
68 | , endTime: null
69 | , text: 'foobar'
70 | }
71 | , filters: {
72 | inurl: 'foobar'
73 | }
74 | });
75 | });
76 |
77 | it('should handle startTime filter', function () {
78 | var options = parseQuery('startTime:13-10-20');
79 | expect(options).to.eql({
80 | settings: {
81 | startTime: '13-10-20'
82 | , endTime: null
83 | , text: ''
84 | }
85 | , filters: {}
86 | });
87 | });
88 |
89 | it('should handle endTime filter', function () {
90 | var options = parseQuery('endTime:13-10-20');
91 | expect(options).to.eql({
92 | settings: {
93 | startTime: null
94 | , endTime: '13-10-20'
95 | , text: ''
96 | }
97 | , filters: {}
98 | });
99 | });
100 |
101 | it('should ignore unknown filter', function () {
102 | var options = parseQuery('foo:bar');
103 | expect(options).to.eql({
104 | settings: {
105 | startTime: null
106 | , endTime: null
107 | , text: 'foo:bar'
108 | }
109 | , filters: {}
110 | });
111 | });
112 | });
113 |
114 | describe('multi filters', function () {
115 |
116 | it('should handle time filters', function () {
117 | var options = parseQuery('endTime:13-10-20 startTime:1/1/1');
118 | expect(options).to.eql({
119 | settings: {
120 | startTime: '1/1/1'
121 | , endTime: '13-10-20'
122 | , text: ''
123 | }
124 | , filters: {}
125 | });
126 | });
127 |
128 | it('should handle time filters and site', function () {
129 | var options = parseQuery('endTime:13-10-20 startTime:1/1/1 site:wat.com');
130 | expect(options).to.eql({
131 | settings: {
132 | startTime: '1/1/1'
133 | , endTime: '13-10-20'
134 | , text: 'wat.com'
135 | }
136 | , filters: {
137 | site: 'wat.com'
138 | }
139 | });
140 | });
141 |
142 | it('should handle time filters and site', function () {
143 | var options = parseQuery('endTime:13-10-20 startTime:1/1/1 site:wat.com inurl:shitmang');
144 | expect(options).to.eql({
145 | settings: {
146 | startTime: '1/1/1'
147 | , endTime: '13-10-20'
148 | , text: 'wat.com shitmang'
149 | }
150 | , filters: {
151 | site: 'wat.com'
152 | , inurl: 'shitmang'
153 | }
154 | });
155 | });
156 |
157 | });
158 |
159 | });
160 |
161 | describe('plain text + filters', function () {
162 |
163 | it('should handle time filters and text', function () {
164 | var options = parseQuery('endTime:13-10-20 startTime:1/1/1 shitmang');
165 | expect(options).to.eql({
166 | settings: {
167 | startTime: '1/1/1'
168 | , endTime: '13-10-20'
169 | , text: 'shitmang'
170 | }
171 | , filters: {
172 | }
173 | });
174 | });
175 |
176 | it('should handle inurl + text', function () {
177 | var options = parseQuery('inurl:hah shitmang');
178 | expect(options).to.eql({
179 | settings: {
180 | startTime: null
181 | , endTime: null
182 | , text: 'hah shitmang'
183 | }
184 | , filters: {
185 | inurl: 'hah'
186 | }
187 | });
188 | });
189 |
190 | it('should handle inurl + ignored filter', function () {
191 | var options = parseQuery('inurl:hah shit:mang');
192 | expect(options).to.eql({
193 | settings: {
194 | startTime: null
195 | , endTime: null
196 | , text: 'hah shit:mang'
197 | }
198 | , filters: {
199 | inurl: 'hah'
200 | }
201 | });
202 |
203 | });
204 |
205 | });
206 |
207 | });
208 |
--------------------------------------------------------------------------------
/lib/visitsbyday.js:
--------------------------------------------------------------------------------
1 | /*
2 | * eHistory Chrome Extension
3 | * https://chrome.google.com/webstore/detail/hiiknjobjfknoghbeelhfilaaikffopb
4 | *
5 | * Copyright 2011, Amjad Masad
6 | * Licensed under the MIT license
7 | * https://github.com/amasad/eHistory/blob/master/LICENSE.txt
8 | *
9 | * Date: Mon May 9
10 | */
11 |
12 | // TODO: Implement linked list
13 | // Class DaysVisits
14 | // Holds visit items on one day
15 |
16 | // Constructor
17 | // @arg firstItem: usually when constructed the first item to put in a particular day is available.
18 | function DaysVisits (firstItem) {
19 | // create the items array
20 | this.items = [firstItem];
21 | // a hash containing the id of the visit item and its index in the items array for faster access
22 | this.id_map = {};
23 | this.id_map[firstItem.id] = firstItem;
24 | // The particular day timestamp
25 | this.day = firstItem.day;
26 | }
27 |
28 | DaysVisits.prototype = {
29 | // Method insert : inserts a single visit item, if the another visit
30 | // exists with the same parent history item, replace it
31 | // @arg item: visit item
32 | insert: function (item) {
33 | if (item.day != this.day) throw new Error("Invalid Day");
34 | // pull up the index of the visit item that corresponds to the same history item if it exists
35 | var currentItem = this.id_map[item.id],
36 | index = this.items.indexOf(currentItem),
37 | mid, spliceInd;
38 | // if the index is valid and the item to be added is newer in time then replace the old one
39 | if (this.items[index]) {
40 | if (currentItem.visitTime < item.visitTime) {
41 | this.items.splice(index, 1);
42 | } else {
43 | return;
44 | }
45 | }
46 | if (!this.items.length) {
47 | this.items.push(item);
48 | } else {
49 | spliceInd = this._binsearch(item);
50 | this.items.splice(spliceInd + 1, 0, item);
51 | }
52 | /*
53 |
54 | console.log(this.binsearch(item));
55 | } catch (e) {console.log("err",this.items.length); throw e}
56 | for (var i = 0; i < this.items.length; i++) {
57 | if (this.items[i].visitTime < item.visitTime){
58 | this.items.splice(i, 0, item);
59 | console.log("really ", i);
60 | break;
61 | } else if (i == this.items.length - 1) {
62 | this.items.push(item);
63 | console.log("pushed ", i);
64 | break;
65 | }
66 | }
67 | if (!this.items.length) this.items.push(item);*/
68 | this.id_map[item.id] = item;
69 | },
70 | _binsearch: function(item) {
71 | var i = 0,
72 | j = this.items.length -1 ,
73 | mid;
74 |
75 | while (true) {
76 | mid = Math.floor((j - i) / 2) + i;
77 |
78 | if (this.items[mid].visitTime < item.visitTime) {
79 | j = mid - 1;
80 | if (i > j) return j;
81 | } else {
82 | i = mid + 1;
83 | if (i > j) return mid;
84 | }
85 | }
86 | },
87 | // Method dequeue: Gets the newest items up to a limited number, usually called after sort.
88 | // @arg length: max length of the number of items to splice off.
89 | // @return Array
90 | dequeue: function (length) {
91 |
92 | return this.items.splice(0, length);
93 | },
94 |
95 | clear: function () {
96 | this.items = [];
97 | this.id_map = {};
98 | }
99 | };
100 |
101 | // Class VisitsByDay, wrapper class that holds as much DaysVisits as there is days
102 | // in the current search instance.
103 |
104 | // Constructor
105 | function VisitsByDay () {
106 | // the newest time allowable in the days this class hosts.
107 | this.latestDay = Date.now();
108 | // hash containing all DaysVisits instances
109 | // @key: Day time stamp.
110 | // @value: DaysVisits instance.
111 | this.items_day = {};
112 | // list of days this class items on
113 | this.days = [];
114 | }
115 |
116 | // public methods
117 | VisitsByDay.prototype = {
118 | // Method insert, inserts one single visit item into the appropriate place.
119 | // @arg item: visit Item.
120 | insert: function (item) {
121 | //array of items on one day
122 | if (item.day > this.latestDay) return;
123 | if (!this.items_day[item.day]) {
124 | this.items_day[item.day] = new DaysVisits(item);
125 | } else {
126 | this.items_day[item.day].insert(item);
127 | }
128 | },
129 | // Method sort, sorts all DaysVisits childs
130 | // @TODO: Merge with dequeue
131 | // @chainable
132 | sort: function () {
133 | this.days = Object.keys(this.items_day);
134 | this.days.sort(function (a,b) {return parseInt(b)-parseInt(a);});
135 | // for (var i=0; i < this.days.length; i++){
136 | // this.items_day[this.days[i]].sort();
137 | // }
138 | return this;
139 | },
140 | // Method dequeue
141 | // @arg length: Maximum number of items to return
142 | dequeue: function (length) {
143 | // helper function
144 | // @arg day: A day timestamp
145 | // @arg index: The day's index in the days array.
146 | var that = this,
147 | ret = [],
148 | daysResults;
149 |
150 | var deleteDay = function(index) {
151 | delete that.items_day[that.days[index]];
152 | that.days.splice(index, 1);
153 | }
154 | for (var i = 0; i < this.days.length && length > 0; i++){
155 | // remove days no longer needed, i.e. garbage visits, generated from getting all visits
156 | // @TODO: is this necessary since we have this check on insertion?
157 | if (this.days[i] > this.latestDay) {
158 | deleteDay(i--);
159 | continue;
160 | }
161 | daysResults = this.items_day[this.days[i]].dequeue(length);
162 | length -= daysResults.length;
163 | ret = ret.concat(daysResults);
164 | if (!this.items_day[this.days[i]].items.length) deleteDay(i--);
165 | }
166 | // record that last date of the last visit handed over, being the latest.
167 | if (ret.length) this.latestDay = ret[ret.length - 1].day;
168 | return ret;
169 | }
170 |
171 | };
172 |
--------------------------------------------------------------------------------
/lib/view.js:
--------------------------------------------------------------------------------
1 | /*
2 | * eHistory Chrome Extension
3 | * https://chrome.google.com/webstore/detail/hiiknjobjfknoghbeelhfilaaikffopb
4 | *
5 | * Copyright 2011, Amjad Masad
6 | * Licensed under the MIT license
7 | * https://github.com/amasad/eHistory/blob/master/LICENSE.txt
8 | *
9 | */
10 | (function () {
11 | /* global Mustache, historyModel, EHistory, Spinner */
12 | 'use strict';
13 | // History View: responsible for populating results in the current page view
14 | // Holds current page state, and updates page controls accordingly
15 | // Direct communication with history model.
16 | // @exports historyView
17 | this.historyView = (function () {
18 | // Initial DOM (jQuery) variables
19 | var $table, $olderPage, $newerPage, $allNav, $throbber, $pageNo, $divMain;
20 | // Current page in the history view
21 | var currentPage = 0;
22 |
23 | var templates = {
24 | 'row': Mustache.compile($('#tmpl-entry-row').html().trim()),
25 | 'day-row': Mustache.compile($('#tmpl-day-row').html().trim())
26 | };
27 |
28 | var spinner = new Spinner({
29 | lines: 13, // The number of lines to draw
30 | length: 10, // The length of each line
31 | width: 2, // The line thickness
32 | radius: 10, // The radius of the inner circle
33 | corners: 1, // Corner roundness (0..1)
34 | rotate: 0, // The rotation offset
35 | direction: 1, // 1: clockwise, -1: counterclockwise
36 | color: '#333', // #rgb or #rrggbb or array of colors
37 | speed: 1, // Rounds per second
38 | trail: 60, // Afterglow percentage
39 | shadow: false, // Whether to render a shadow
40 | hwaccel: true, // Whether to use hardware acceleration
41 | className: 'spinner', // The CSS class to assign to the spinner
42 | zIndex: 2e9, // The z-index (defaults to 2000000000)
43 | top: '0', // Top position relative to parent in px
44 | left: 'auto' // Left position relative to parent in px
45 | });
46 |
47 | //init
48 | //On domReady get the DOM elements
49 | $(function(){
50 | // Main table that holds results
51 | $table = $('#tbl-main');
52 | // Button responsible for getting older results, i.e. previous page
53 | $olderPage = $('.next-page');
54 | // Button for getting newer results, i.e. next page
55 | $newerPage = $('.prev-page');
56 | // jQuery instance holding all the navigation controls
57 | $allNav = $olderPage.add($newerPage);
58 | $throbber = $('#throbber');
59 | $pageNo = $('.page-no');
60 | // Division that holds the scroll value for the overflow
61 | $divMain = $('#div-main');
62 | // Scroll to top of results
63 | function navTop(){
64 | $divMain.animate({
65 | scrollTop: $divMain.position().top
66 | }, 'slow');
67 | }
68 | // Bind buttons functionalities
69 | $olderPage.click(function () {
70 | currentPage++;
71 | $(historyModel).trigger('modelrefresh');
72 | navTop();
73 | });
74 | $newerPage.click(function () {
75 | if (currentPage !== 0) currentPage--;
76 | $(historyModel).trigger('modelrefresh');
77 | navTop();
78 | });
79 | // Navigation controls disabled onload
80 | $allNav.attr('disabled', true);
81 | });
82 | // when the view is on the first page disable first, newer buttons
83 | function updateControls () {
84 | $allNav.attr('disabled', false);
85 | if (currentPage === 0){
86 | $newerPage.attr('disabled', true);
87 | }
88 | }
89 | // Listen to history model refresh event
90 | // and get the page required from the model
91 | $(historyModel).bind('modelrefresh', function () {
92 | updateControls();
93 | var page = this.getPage(currentPage);
94 | if (page === -1) return;
95 | $pageNo.text(currentPage + 1);
96 | // results per day hash
97 | var results_day = {};
98 | $.each(page, function (i, visit) {
99 | if (!results_day[visit.day]) results_day[visit.day] = [];
100 | results_day[visit.day].push(visit);
101 | });
102 | // empty the view table
103 | $table.empty();
104 | // for each day create a day row (holds day info and a checkbox allows selection of all day items)
105 | $.each(results_day, function (day, items) {
106 | $(templates['day-row']({date: new Date(parseInt(day, 10)).toDateString()})).appendTo($table);
107 | // on each day populate the results that corresponds to that day.
108 | $.each(items, function (i, visit) {
109 | var row = $(templates['row'](visit));
110 | // let the elem data hold info of the corrosponding visit
111 | row.data('id', visit.id);
112 | row.data('day', visit.day);
113 | // check if the current item to be populated was selected in earlier navigations
114 | if (historyModel.isSelected(visit.id, visit.day)) {
115 | row.children().children('input').attr('checked', true);
116 | }
117 | $table.append(row);
118 | });
119 | });
120 | });
121 | // listens for lastPage event from the historyModel update controls accordingly
122 | $(historyModel).bind('lastPage', function () {
123 | $olderPage.attr('disabled', true);
124 | });
125 | $(EHistory).bind('done', function () {
126 | $throbber.removeClass('active');
127 | });
128 | $(EHistory).bind('finished', function () {
129 | if ($table.is(':empty')) {
130 | $table.append('| No results :( |
');
131 | }
132 | $newerPage.attr('disabled', false);
133 | });
134 | // Public functions
135 | return {
136 | clear: function () {
137 | currentPage = 0;
138 | $table.empty();
139 | },
140 | disableControls: function disableControls() {
141 | $allNav.attr('disabled', true);
142 | },
143 | displayThrobber: function () {
144 | $throbber.addClass('active');
145 | spinner.spin($throbber[0]);
146 | }
147 | };
148 | })();
149 |
150 | }).call(this);
151 |
--------------------------------------------------------------------------------
/history.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | eHistory
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
50 |
58 |
59 |
60 |
64 |
65 |
76 |
77 |
83 |
84 |
85 |
86 |
87 |
88 |
89 |
90 |
91 |
92 |
93 |
94 |
95 |
96 |
--------------------------------------------------------------------------------
/lib/model.js:
--------------------------------------------------------------------------------
1 | /*
2 | * eHistory Chrome Extension
3 | * https://chrome.google.com/webstore/detail/hiiknjobjfknoghbeelhfilaaikffopb
4 | *
5 | * Copyright 2011, Amjad Masad
6 | * Licensed under the MIT license
7 | * https://github.com/amasad/eHistory/blob/master/LICENSE.txt
8 | *
9 | */
10 |
11 | /* Cleanup TODO:
12 | * Why the hell was I mixing module pattern and classes?
13 | * refactor append.
14 | * remove reference to view!
15 | */
16 |
17 | (function (){
18 | 'use strict';
19 | /* global EHistory, historyView, chrome, alert */
20 | // Holds the current history search state, direct communication with the EHistory library.
21 | // Eventsource, triggers:
22 | // 'modelreferesh': when new items are appended to the model.
23 | // 'lastPage': when there is no more items.
24 | this.historyModel = (function(){
25 | // Private variables
26 | // results, holding all the current results available
27 | var results = [];
28 | // boolean stating whether the EHistory system has delivered all of the current query results
29 | var finished = false;
30 | // hash holding all the selected items for deletion
31 | // key: Timestamp for day, value: Array of history Items
32 | // TODO: Make at an array of ids
33 | var selected = {};
34 | var prevPage = -1;
35 | // Constructor for History Model Singleton
36 | function HistoryModel(){
37 | // hash of items, Key: id, Value: Item
38 | this.item_map = {};
39 | // Current results per page
40 | this.pageSize = 150;
41 | }
42 | // Receives data from the EHistory and appends the results to the current model
43 | // Used as a callback for the EHistory to be called when the search is done
44 | // @arg data: hash containing:
45 | // items: history Items
46 | // visits: visits found corresponding to history items.
47 | HistoryModel.prototype = {
48 | append: function (data) {
49 | var item_map = this.item_map;
50 | // Populate the item_map hash with items coming from EHistory
51 | for (var j = 0, item; item = data.items[j]; j++){
52 | item_map[item.id] = item;
53 | }
54 | var visit, resultItem, timeStr, hours, which;
55 | for (var i = 0; visit = data.visits[i]; i++){
56 | // Create a 'resultItem' that will contain all the information about the visit
57 | resultItem = Object.create(item_map[visit.id]);
58 | resultItem.visitTime = visit.visitTime || 0;
59 | resultItem.day = visit.day || 0;
60 |
61 | timeStr = new Date(visit.visitTime).toLocaleTimeString();
62 | // If we know the format of the locale time string then we'll try to
63 | // end up with HH:MM [PERIOD] otherwise we'll just use it as is.
64 | if (/^\d{1,2}:\d{1,2}:\d{1,2}\s[AP]M$/.test(timeStr)) {
65 | var parts = timeStr.split(':');
66 | var period = parts.pop().substr(-2);
67 | resultItem.date = parts.join(':') + ' ' + period;
68 | } else {
69 | resultItem.date = timeStr;
70 | }
71 |
72 | resultItem.domain = Util.getHostname(resultItem.url)
73 |
74 | results.push(resultItem);
75 | }
76 | // Trigger an event stating that the model has new additions
77 | $(this).trigger('modelrefresh');
78 | },
79 | // Gives model results according to the page number requested
80 | // if the page requested was not found, the model will make a new page request
81 | // to the EHistory system requesting a new page, and returns -1 to the caller
82 | // stating that there is no result to be found.
83 | // @arg page: # of the page requested
84 | getPage: function (page) {
85 | var pageSize = this.pageSize;
86 | // pages don't start from index 0
87 | page++;
88 | var uBound = page * pageSize;
89 | var lBound = uBound - pageSize;
90 | // check if the results requested are available and the EHistory has not finished
91 | // if not request from EHistory
92 | var ret = results.slice(lBound, lBound + pageSize);
93 | if (results.length < uBound && !finished){
94 | if (page === prevPage)
95 | return ret;
96 | prevPage = page;
97 | // TODO combine into one function call
98 | historyView.disableControls();
99 | historyView.displayThrobber();
100 | EHistory.getPage(page,$.proxy(this.append, this));
101 | return -1;
102 | }
103 | // TODO: Verify the following
104 | // if only some of the results found then make others know this is the last page.
105 | if (ret.length < pageSize) {
106 | $(this).trigger('lastPage');
107 | }
108 | return ret;
109 | },
110 | // called to make sure that a current item in a current day is to be deleted
111 | // @arg id: item id
112 | // @arg day: the day where the item was selected
113 | // @arg elem: the DOM elem, row corresponding to the item.
114 | // TODO: Is elem necessary?
115 | select: function (id, day, elem) {
116 | if (!selected[day]) selected[day] = [];
117 | this.item_map.elem = elem;
118 | selected[day].push(this.item_map[id]);
119 | },
120 | // remove selection of a specific item from a specific day
121 | // @arg id: item id
122 | // @arg day: the day where the item supposed to be.
123 | unselect: function (id, day) {
124 | if (!selected[day]) return;
125 | selected[day].splice(selected[day].indexOf(this.item_map[id]), 1);
126 | },
127 | //check to see if a specific item in a specific day is selected.
128 | isSelected: function (id, day) {
129 | if (!selected[day]) return;
130 | return selected[day].indexOf(this.item_map[id]) > -1
131 | },
132 | // Deletes all selected items
133 | // loops over all items in all days and make a call to the EHistory
134 | // for each item to be deleted
135 | removeSelected: function () {
136 | EHistory.deleteUrls(selected, function () {
137 | //TODO: Don't reload
138 | window.location.reload();
139 | });
140 | },
141 |
142 | clearHistory: function () {
143 | chrome.history.deleteAll(function () {
144 | window.location.reload();
145 | });
146 | },
147 |
148 | clearResults: function () {
149 | EHistory.deleteAllresults(function () {
150 | alert('The page will reload now\nIt may take Chrome several minutes before making the history available again.');
151 | window.location.reload();
152 | });
153 | },
154 | // Clears the current model state
155 | // usually called when a new search is taking place.
156 | clear: function(){
157 | results = [];
158 | finished = false;
159 | selected = {};
160 | this.item_map = {};
161 | prevPage = -1;
162 | },
163 |
164 | getDomain: function (id) {
165 | var item = this.item_map[id];
166 | if (item) {
167 | return Util.getHostname(item.url);
168 | }
169 | },
170 |
171 | deleteItem: function (id) {
172 | var item = this.item_map[id];
173 | if (item) {
174 | chrome.history.deleteUrl({url: item.url});
175 | }
176 | }
177 | };
178 | // an event listener for when the EHistory has got all its results
179 | $(EHistory).bind('finished', function () {
180 | finished = true;
181 | });
182 | // Instantiate the history model
183 | var historyModel = new HistoryModel();
184 | return historyModel;
185 | })();
186 |
187 | }).call(this);
188 |
--------------------------------------------------------------------------------
/lib/ehistory.js:
--------------------------------------------------------------------------------
1 | /*
2 | * eHistory Chrome Extension
3 | * https://chrome.google.com/webstore/detail/hiiknjobjfknoghbeelhfilaaikffopb
4 | *
5 | * Copyright 2011, Amjad Masad
6 | * Licensed under the MIT license
7 | * https://github.com/amasad/eHistory/blob/master/LICENSE.txt
8 | *
9 | * Date: Mon May 9
10 | */
11 | //EHistory container
12 | var EHistory = (function ($) {
13 | //CONSTANTS
14 | var MAX = 2147483647;
15 | //constructor
16 | function EHistory() {/*fdsfsd*/}
17 | // Extend date class to get some nice features
18 | (function () {
19 | // milliseconds in one day
20 | var msDay = 24 * 60 * 60 * 1000;
21 |
22 | Date.prototype.start = function () {
23 | return new Date(this.toDateString());
24 | };
25 |
26 | Date.prototype.next = function () {
27 | return this.start().getTime() + msDay;
28 | };
29 |
30 | Date.prototype.prev = function () {
31 | return this.start().getTime() - msDay;
32 | };
33 | }());
34 |
35 | var arrayUnique = function (arr) {
36 | var ids = {};
37 | for (var i=0; i < arr.length; i++) {
38 | if (ids[arr[i].id]) {
39 | arr.splice(i--, 1);
40 | } else {
41 | ids[arr[i].id] = true;
42 | }
43 | }
44 | return arr;
45 | }
46 |
47 | // simple memoization
48 | var memoizeKeys = (function () {
49 | var lastKeys;
50 | var lastObj;
51 | return function (obj) {
52 | if (lastObj === obj) {
53 | return lastKeys;
54 | } else {
55 | lastObj = obj;
56 | return (lastKeys = Object.keys(obj));
57 | }
58 | }
59 | })();
60 |
61 | //methods
62 | EHistory.prototype = {
63 |
64 |
65 | search: function (settings, filters, cb) {
66 | this.offset = 0;
67 | this.filters = filters;
68 | this.visits_day = new VisitsByDay();
69 | this.pageSize = settings.maxResults;
70 | this.cb = cb;
71 | this.query = settings.text;
72 | this.settings = {
73 | text: "",
74 | startTime: 0,
75 | endTime: Date.now(),
76 | maxResults : 150
77 | };
78 | $.extend(this.settings, settings);
79 | this.getPage(1);
80 | },
81 |
82 |
83 | getPage: function (pageNo, cb, callback) {
84 | this.cb = cb || this.cb;
85 | var settings = this.settings,
86 | filtered = [],
87 | ids = {},
88 | that = this,
89 | filter = Object.keys(this.filters).length;
90 |
91 | settings.maxResults = this.offset + this._pageLimit(pageNo);
92 |
93 | function search () {
94 | chrome.history.search(settings, function (result) {
95 | var resultItem;
96 | result = result.slice(settings.maxResults - that.pageSize, settings.maxResults);
97 | for (var i=0; i < result.length && filtered.length < that.pageSize; i++) {
98 | resultItem = result[i];
99 | if (ids[resultItem.id] || (filter && !that.filter(resultItem))) continue;
100 | ids[resultItem.id] = true;
101 | filtered.push(resultItem);
102 | }
103 | if (result.length && filtered.length < that.pageSize) {
104 | settings.maxResults += that.pageSize;
105 | that.offset = settings.maxResults - that._pageLimit(pageNo) - i;
106 | search();
107 | } else if (filtered.length) {
108 | if (filtered.length < that.pageSize) $(that).trigger("finished");
109 | if (callback) callback(filtered);
110 | else that.getVisits(filtered);
111 | } else {
112 | $(that).trigger("finished");
113 | $(that).trigger("done");
114 | }
115 | });
116 | }
117 | search();
118 | },
119 |
120 | _pageLimit: function (pageNo) {
121 | return pageNo * this.pageSize;
122 | },
123 |
124 | getVisits: function (items) {
125 | var visits = [],
126 | that = this,
127 | items_length = items.length,
128 | days = [],
129 | visits_day = this.visits_day,
130 | visitItem, day;
131 |
132 | for (var i = 0; i < items_length; i++) {
133 | chrome.history.getVisits({url: items[i].url}, function (res_visits) {
134 | items_length--;
135 | for(var j = 0; j < res_visits.length; j++) {
136 | visitItem = res_visits[j];
137 | if (visitItem.visitTime > that.settings.endTime ||
138 | visitItem.visitTime < that.settings.startTime) continue;
139 | visitItem.day = day = new Date(visitItem.visitTime).start().getTime();
140 | visits_day.insert(visitItem);
141 | }
142 |
143 | if (items_length === 0) {
144 | that.cb({
145 | items: items,
146 | visits: visits_day.sort().dequeue(that.pageSize)
147 | });
148 | $(that).trigger("done");
149 | }
150 | });
151 | }
152 | },
153 |
154 | deleteUrls: function (urlsByDay, callback) {
155 | var days = memoizeKeys(urlsByDay),
156 | count = 0;
157 |
158 | (function deleteUrls() {
159 | if (count === days.length) return callback();
160 |
161 | var urls = urlsByDay[days[count++]];
162 | for (var i = 0; i < urls.length; i++)
163 | chrome.history.deleteUrl({url: urls[i].url});
164 | setTimeout(deleteUrls, 100);
165 | })();
166 | },
167 |
168 | deleteAllresults: function (callback) {
169 | var settings = $.extend(null, this.settings),
170 | finished = false,
171 | that = this;
172 |
173 | settings.maxResults = MAX;
174 | this.offset = 0;
175 | function finish () {
176 | finished = true;
177 | $(that).unbind("finished", finish);
178 | callback();
179 | }
180 | $(this).bind("finished", finish);
181 | (function deleteAll () {
182 | if (finished) return;
183 | that.getPage(1, undefined, function (page) {
184 | for (var i = 0; i < page.length; i++) {
185 | chrome.history.deleteUrl({url: page[i].url});
186 | }
187 | deleteAll();
188 | });
189 | })();
190 |
191 | },
192 | /*
193 | deleteUrlOnDay: function (url, day, callback) {
194 | var nextDay = new Date(parseFloat(day)).next();
195 | var toDelete = [];
196 | var that = this;
197 | chrome.history.getVisits({url:url}, function (visits) {
198 | var visitTime;
199 | for (var i=0, visit; visit = visits[i]; i++) {
200 | visitTime = visit.visitTime;
201 | if (visitTime >= day && visitTime <= nextDay) {
202 | toDelete.push(visitTime);
203 | }
204 | }
205 | that.removeVisits(toDelete, callback);
206 | });
207 | },
208 |
209 | removeVisits: function (visitTimes, callback) {
210 | var length = visitTimes.length;
211 | for (var i = 0, visitTime; visitTime = visitTimes[i]; i++) {
212 | chrome.history.deleteRange({
213 | startTime: visitTime - 0.1,
214 | endTime: visitTime + 0.1
215 | }, function () {
216 | length--;
217 | if (length === 0) {
218 | callback();
219 | }
220 | });
221 | }
222 | },*/
223 |
224 | filter: function (item) {
225 | var operators = memoizeKeys(this.filters);
226 | if (!operators.length) return true;
227 | for (var i=0; i < operators.length; i++) {
228 | var res = Filters[operators[i]](item, this.filters[operators[i]]);
229 | if (!res) return false;
230 | }
231 | return true;
232 | }
233 | };
234 |
235 | var Filters = (function () {
236 | function isIn(prop) {
237 | return function (item, str) {
238 | return item[prop].toLowerCase().indexOf(str.toLowerCase()) > -1;
239 | }
240 | };
241 |
242 | return {
243 | intitle: isIn('title'),
244 | inurl: isIn('url'),
245 | site: function (item, str) {
246 | // amjad.a.com a.com
247 | var hostDomains = Util.getHostname(item.url).split('.'),
248 | // handle extra dots (e.g. .jo.).
249 | siteDomains = str.replace(/^\.*|\.*$/g, '').split('.');
250 | for (var i = siteDomains.length - 1, j = hostDomains.length - 1; j >= 0 && i >= 0; i--, j--) {
251 | if (hostDomains[j] !== siteDomains[i]) return false;
252 | }
253 | return true;
254 | }
255 | };
256 | })();
257 |
258 | return new EHistory;
259 | })(jQuery);
--------------------------------------------------------------------------------
/css/normalize.css:
--------------------------------------------------------------------------------
1 | /*! normalize.css v2.1.2 | MIT License | git.io/normalize */
2 |
3 | /* ==========================================================================
4 | HTML5 display definitions
5 | ========================================================================== */
6 |
7 | /**
8 | * Correct `block` display not defined in IE 8/9.
9 | */
10 |
11 | article,
12 | aside,
13 | details,
14 | figcaption,
15 | figure,
16 | footer,
17 | header,
18 | hgroup,
19 | main,
20 | nav,
21 | section,
22 | summary {
23 | display: block;
24 | }
25 |
26 | /**
27 | * Correct `inline-block` display not defined in IE 8/9.
28 | */
29 |
30 | audio,
31 | canvas,
32 | video {
33 | display: inline-block;
34 | }
35 |
36 | /**
37 | * Prevent modern browsers from displaying `audio` without controls.
38 | * Remove excess height in iOS 5 devices.
39 | */
40 |
41 | audio:not([controls]) {
42 | display: none;
43 | height: 0;
44 | }
45 |
46 | /**
47 | * Address styling not present in IE 8/9.
48 | */
49 |
50 | [hidden] {
51 | display: none;
52 | }
53 |
54 | /* ==========================================================================
55 | Base
56 | ========================================================================== */
57 |
58 | /**
59 | * 1. Set default font family to sans-serif.
60 | * 2. Prevent iOS text size adjust after orientation change, without disabling
61 | * user zoom.
62 | */
63 |
64 | html {
65 | font-family: sans-serif; /* 1 */
66 | -ms-text-size-adjust: 100%; /* 2 */
67 | -webkit-text-size-adjust: 100%; /* 2 */
68 | }
69 |
70 | /**
71 | * Remove default margin.
72 | */
73 |
74 | body {
75 | margin: 0;
76 | }
77 |
78 | /* ==========================================================================
79 | Links
80 | ========================================================================== */
81 |
82 | /**
83 | * Address `outline` inconsistency between Chrome and other browsers.
84 | */
85 |
86 | a:focus {
87 | outline: thin dotted;
88 | }
89 |
90 | /**
91 | * Improve readability when focused and also mouse hovered in all browsers.
92 | */
93 |
94 | a:active,
95 | a:hover {
96 | outline: 0;
97 | }
98 |
99 | /* ==========================================================================
100 | Typography
101 | ========================================================================== */
102 |
103 | /**
104 | * Address variable `h1` font-size and margin within `section` and `article`
105 | * contexts in Firefox 4+, Safari 5, and Chrome.
106 | */
107 |
108 | h1 {
109 | font-size: 2em;
110 | margin: 0.67em 0;
111 | }
112 |
113 | /**
114 | * Address styling not present in IE 8/9, Safari 5, and Chrome.
115 | */
116 |
117 | abbr[title] {
118 | border-bottom: 1px dotted;
119 | }
120 |
121 | /**
122 | * Address style set to `bolder` in Firefox 4+, Safari 5, and Chrome.
123 | */
124 |
125 | b,
126 | strong {
127 | font-weight: bold;
128 | }
129 |
130 | /**
131 | * Address styling not present in Safari 5 and Chrome.
132 | */
133 |
134 | dfn {
135 | font-style: italic;
136 | }
137 |
138 | /**
139 | * Address differences between Firefox and other browsers.
140 | */
141 |
142 | hr {
143 | -moz-box-sizing: content-box;
144 | box-sizing: content-box;
145 | height: 0;
146 | }
147 |
148 | /**
149 | * Address styling not present in IE 8/9.
150 | */
151 |
152 | mark {
153 | background: #ff0;
154 | color: #000;
155 | }
156 |
157 | /**
158 | * Correct font family set oddly in Safari 5 and Chrome.
159 | */
160 |
161 | code,
162 | kbd,
163 | pre,
164 | samp {
165 | font-family: monospace, serif;
166 | font-size: 1em;
167 | }
168 |
169 | /**
170 | * Improve readability of pre-formatted text in all browsers.
171 | */
172 |
173 | pre {
174 | white-space: pre-wrap;
175 | }
176 |
177 | /**
178 | * Set consistent quote types.
179 | */
180 |
181 | q {
182 | quotes: "\201C" "\201D" "\2018" "\2019";
183 | }
184 |
185 | /**
186 | * Address inconsistent and variable font size in all browsers.
187 | */
188 |
189 | small {
190 | font-size: 80%;
191 | }
192 |
193 | /**
194 | * Prevent `sub` and `sup` affecting `line-height` in all browsers.
195 | */
196 |
197 | sub,
198 | sup {
199 | font-size: 75%;
200 | line-height: 0;
201 | position: relative;
202 | vertical-align: baseline;
203 | }
204 |
205 | sup {
206 | top: -0.5em;
207 | }
208 |
209 | sub {
210 | bottom: -0.25em;
211 | }
212 |
213 | /* ==========================================================================
214 | Embedded content
215 | ========================================================================== */
216 |
217 | /**
218 | * Remove border when inside `a` element in IE 8/9.
219 | */
220 |
221 | img {
222 | border: 0;
223 | }
224 |
225 | /**
226 | * Correct overflow displayed oddly in IE 9.
227 | */
228 |
229 | svg:not(:root) {
230 | overflow: hidden;
231 | }
232 |
233 | /* ==========================================================================
234 | Figures
235 | ========================================================================== */
236 |
237 | /**
238 | * Address margin not present in IE 8/9 and Safari 5.
239 | */
240 |
241 | figure {
242 | margin: 0;
243 | }
244 |
245 | /* ==========================================================================
246 | Forms
247 | ========================================================================== */
248 |
249 | /**
250 | * Define consistent border, margin, and padding.
251 | */
252 |
253 | fieldset {
254 | border: 1px solid #c0c0c0;
255 | margin: 0 2px;
256 | padding: 0.35em 0.625em 0.75em;
257 | }
258 |
259 | /**
260 | * 1. Correct `color` not being inherited in IE 8/9.
261 | * 2. Remove padding so people aren't caught out if they zero out fieldsets.
262 | */
263 |
264 | legend {
265 | border: 0; /* 1 */
266 | padding: 0; /* 2 */
267 | }
268 |
269 | /**
270 | * 1. Correct font family not being inherited in all browsers.
271 | * 2. Correct font size not being inherited in all browsers.
272 | * 3. Address margins set differently in Firefox 4+, Safari 5, and Chrome.
273 | */
274 |
275 | button,
276 | input,
277 | select,
278 | textarea {
279 | font-family: inherit; /* 1 */
280 | font-size: 100%; /* 2 */
281 | margin: 0; /* 3 */
282 | }
283 |
284 | /**
285 | * Address Firefox 4+ setting `line-height` on `input` using `!important` in
286 | * the UA stylesheet.
287 | */
288 |
289 | button,
290 | input {
291 | line-height: normal;
292 | }
293 |
294 | /**
295 | * Address inconsistent `text-transform` inheritance for `button` and `select`.
296 | * All other form control elements do not inherit `text-transform` values.
297 | * Correct `button` style inheritance in Chrome, Safari 5+, and IE 8+.
298 | * Correct `select` style inheritance in Firefox 4+ and Opera.
299 | */
300 |
301 | button,
302 | select {
303 | text-transform: none;
304 | }
305 |
306 | /**
307 | * 1. Avoid the WebKit bug in Android 4.0.* where (2) destroys native `audio`
308 | * and `video` controls.
309 | * 2. Correct inability to style clickable `input` types in iOS.
310 | * 3. Improve usability and consistency of cursor style between image-type
311 | * `input` and others.
312 | */
313 |
314 | button,
315 | html input[type="button"], /* 1 */
316 | input[type="reset"],
317 | input[type="submit"] {
318 | -webkit-appearance: button; /* 2 */
319 | cursor: pointer; /* 3 */
320 | }
321 |
322 | /**
323 | * Re-set default cursor for disabled elements.
324 | */
325 |
326 | button[disabled],
327 | html input[disabled] {
328 | cursor: default;
329 | }
330 |
331 | /**
332 | * 1. Address box sizing set to `content-box` in IE 8/9.
333 | * 2. Remove excess padding in IE 8/9.
334 | */
335 |
336 | input[type="checkbox"],
337 | input[type="radio"] {
338 | box-sizing: border-box; /* 1 */
339 | padding: 0; /* 2 */
340 | }
341 |
342 | /**
343 | * 1. Address `appearance` set to `searchfield` in Safari 5 and Chrome.
344 | * 2. Address `box-sizing` set to `border-box` in Safari 5 and Chrome
345 | * (include `-moz` to future-proof).
346 | */
347 |
348 | input[type="search"] {
349 | -webkit-appearance: textfield; /* 1 */
350 | -moz-box-sizing: content-box;
351 | -webkit-box-sizing: content-box; /* 2 */
352 | box-sizing: content-box;
353 | }
354 |
355 | /**
356 | * Remove inner padding and search cancel button in Safari 5 and Chrome
357 | * on OS X.
358 | */
359 |
360 | input[type="search"]::-webkit-search-cancel-button,
361 | input[type="search"]::-webkit-search-decoration {
362 | -webkit-appearance: none;
363 | }
364 |
365 | /**
366 | * Remove inner padding and border in Firefox 4+.
367 | */
368 |
369 | button::-moz-focus-inner,
370 | input::-moz-focus-inner {
371 | border: 0;
372 | padding: 0;
373 | }
374 |
375 | /**
376 | * 1. Remove default vertical scrollbar in IE 8/9.
377 | * 2. Improve readability and alignment in all browsers.
378 | */
379 |
380 | textarea {
381 | overflow: auto; /* 1 */
382 | vertical-align: top; /* 2 */
383 | }
384 |
385 | /* ==========================================================================
386 | Tables
387 | ========================================================================== */
388 |
389 | /**
390 | * Remove most spacing between table cells.
391 | */
392 |
393 | table {
394 | border-collapse: collapse;
395 | border-spacing: 0;
396 | }
--------------------------------------------------------------------------------
/lib/controller.js:
--------------------------------------------------------------------------------
1 | /*
2 | * eHistory Chrome Extension
3 | * https://chrome.google.com/webstore/detail/hiiknjobjfknoghbeelhfilaaikffopb
4 | *
5 | * Copyright 2011, Amjad Masad
6 | * Licensed under the MIT license
7 | * https://github.com/amasad/eHistory/blob/master/LICENSE.txt
8 | *
9 | */
10 | (function(){
11 | /* global parseQuery, historyModel, historyView, EHistory, confirm */
12 | 'use strict';
13 | /* parseForm: Parses the html form into text format
14 | * @arg (jQueryObject) $form: jQuery object containing the form element
15 | * @returns (String) text query equivalent to the form
16 | * intitle:title inurl:url site:site startTime:startime endTime:endtime searchquery
17 | */
18 | function parseForm ($form) {
19 | var query = '',
20 | text = '';
21 | // loop over all input elements
22 | $form.find('input').each(function (i, elem) {
23 | elem = $(elem);
24 | if (elem.attr('id') === 'pure-text') {
25 | // just text
26 | text += elem.val();
27 | } else {
28 | // filter/setting
29 | query += elem.val() ? ' ' + elem.data('settings-item') + ':' + elem.val() : '';
30 | }
31 | });
32 | // return filter/setting text format key:value followed by regular text
33 | return $.trim(query + ' ' + text);
34 | }
35 |
36 | // Check version number
37 | $(function() {
38 | $.getJSON('manifest.json', function (manifest) {
39 | var version = manifest.version;
40 | if (localStorage['version'] !== version) {
41 | localStorage.clear();
42 | localStorage['version'] = version;
43 | /* global console */
44 | console.log('Version Updated!');
45 | }
46 | });
47 | });
48 |
49 | /*************** Controller ***************/
50 | /* Collection of functions and event handlers
51 | * Interacts with UI, historyModel and historyView
52 | */
53 | $(function(){
54 | // DOM ready
55 | // search box
56 | var $query = $('#query');
57 | // advanced search form
58 | var $pnlAdvanced = $('.advanced-search');
59 | // history items table
60 | var $resultsTable = $('#tbl-main');
61 |
62 | $('.open-advanced').click(function () {
63 | var config = parseQuery($query.val());
64 | var operators = $.extend(config.settings, config.filters);
65 | $pnlAdvanced.find('input').each(function (i, elem) {
66 | elem = $(elem);
67 | if (elem.attr('id') === 'pure-text'){
68 | elem.val(config[2]);
69 | } else {
70 | // in the elements data contains type of settings/filter
71 | elem.val(operators[elem.data('settings-item')] || '');
72 | }
73 | });
74 | });
75 |
76 | // results day headers check-boxs handler
77 | $resultsTable.delegate('.chk-day', 'change', function () {
78 | // check all results until the next day header
79 | $(this).parents('tr').nextUntil('.hdr-day')
80 | .children(':nth-child(1)').children()
81 | .attr('checked', $(this).is(':checked')).trigger('change');
82 | });
83 |
84 | var shiftDown = false;
85 | $(document).bind('keydown keyup', function (e) {
86 | shiftDown = e.shiftKey;
87 | });
88 |
89 | $resultsTable.delegate('.chk-entry', 'click', function () {
90 | if (shiftDown) {
91 | shiftClick($(this));
92 | }
93 | });
94 |
95 | var shiftClick = (function () {
96 |
97 | function getPath($firstChecked, dir, isChecked) {
98 | var $path = $();
99 | while ($firstChecked.length) {
100 | $firstChecked = $firstChecked[dir]();
101 | if (!$firstChecked.is('.entry')) {
102 | continue;
103 | } else if ((isChecked && $firstChecked.find(':checked').length) ||
104 | (!isChecked && !$firstChecked.find(':checked').length)) {
105 | break;
106 | } else {
107 | $path = $path.add($firstChecked);
108 | }
109 | }
110 |
111 | // If we reached the end and we're looking for checked boxes then
112 | // we haven't found any, however, if you we reached the end while
113 | // looking for checked boxes, we may have something.
114 | if ($firstChecked.length || !isChecked) {
115 | return $path;
116 | } else {
117 | return $();
118 | }
119 | }
120 |
121 | return function ($input) {
122 | var $row = $input.parents('tr');
123 | var isChecked = $input.is(':checked');
124 | // Go up until we find the first checked input
125 | var $path = getPath($row, 'prev', isChecked);
126 | // If we couldn't find anything going up then go down.
127 | if (!$path.length) {
128 | $path = getPath($row, 'next', isChecked);
129 | }
130 | $path.each(function (i, row) {
131 | if ($(row).is('.entry')) {
132 | $(row).find('input[type=checkbox]').attr('checked', isChecked);
133 | }
134 | });
135 | };
136 |
137 | })();
138 |
139 | // result item checkbox handler
140 | $resultsTable.delegate('.chk-entry', 'change', function () {
141 | var val = $(this).attr('checked'),
142 | $row = $(this).parents('tr'),
143 | // decides what function to call, select/unselect
144 | fn = val ? $.proxy(historyModel.select, historyModel) : $.proxy(historyModel.unselect, historyModel);
145 | fn($row.data('id'),$row.data('day'));
146 | });
147 |
148 | $(document).delegate('.hdr-day, .entry', 'click', function (e) {
149 | if ($(e.target).is('input') || $(e.target).is('a')) return;
150 | var $input = $(this).find('input[type=checkbox]');
151 | $input
152 | .attr('checked', !$input.is(':checked'))
153 | .trigger('change');
154 | if (shiftDown) {
155 | shiftClick($input);
156 | }
157 | });
158 |
159 | // Update the main search box whenever advanced settings are changed.
160 | var updateMainSearchBox = function () {
161 | // Delay until the keypress is handled by the browser.
162 | setTimeout(function() {
163 | $query.val(parseForm($pnlAdvanced) || '');
164 | }, 0);
165 | };
166 |
167 | $('input', $pnlAdvanced).change(updateMainSearchBox)
168 | .keypress(updateMainSearchBox)
169 | .keydown(updateMainSearchBox);
170 |
171 | $('body').bind('click', function (e) {
172 | if ($(e.srcElement).parents('.advanced-search').length) return;
173 | $('.open-advanced').show();
174 | $('.advanced-search').hide();
175 | });
176 | $('.open-advanced').click(function () {
177 | $('.advanced-search').show();
178 | $('.open-advanced').hide();
179 | return false;
180 | });
181 |
182 | // called to initiate the search
183 | function search(config) {
184 | var settings = config.settings,
185 | filters = config.filters;
186 |
187 | historyView.displayThrobber();
188 |
189 | EHistory.search({
190 | text: settings.text || '',
191 | startTime: new Date(settings.startTime || 0).getTime() ,
192 | endTime: new Date(settings.endTime || Date.now()).getTime(),
193 | maxResults: historyModel.pageSize
194 | }, filters, function(results){
195 | historyModel.append(results);
196 | });
197 |
198 | // If the user isn't searching then 'delete resutls' is the same
199 | // as 'clear history'.
200 | if (!settings.text) {
201 | $('.delete-menu .results').hide();
202 | } else {
203 | $('.delete-menu .results').show();
204 | }
205 | }
206 |
207 | // form submit handler
208 | $('#frm-search').submit(function (e) {
209 | var text;
210 | e.preventDefault();
211 | //clear everything
212 | historyModel.clear();
213 | historyView.clear();
214 | historyView.disableControls();
215 |
216 | if ($pnlAdvanced.is(':visible')){
217 | text = parseForm($pnlAdvanced);
218 | search(parseQuery(text));
219 | } else {
220 | search(parseQuery($query.val()));
221 | }
222 | //return false;
223 | });
224 |
225 | $('#btn-clear-history').click(function () {
226 | if (confirm('Delete all items from history?')) {
227 | historyModel.clearHistory();
228 | }
229 | });
230 |
231 | $('.delete.dropdown').click(function () {
232 | $(this).toggleClass('open');
233 | $(this).next().toggle($(this).is('.open'));
234 | });
235 |
236 | $('body').click(function (e) {
237 | if (!$(e.target).is('.delete.dropdown')
238 | && !$(e.target).closest('.delete.dropdown').length) {
239 | $('.delete-menu').hide();
240 | $('.delete.dropdown').removeClass('open');
241 | }
242 | });
243 |
244 | $('.delete-menu .selected').click(function () {
245 | if (confirm('Delete selected items?')) {
246 | historyModel.removeSelected();
247 | }
248 | });
249 |
250 | $('.delete-menu .results').click(function () {
251 | if (confirm('Delete all search results?')) {
252 | historyModel.clearResults();
253 | }
254 | });
255 |
256 | $('.query').focus(function () {
257 | $('.query-wrapper').addClass('active');
258 | });
259 | $('.query').blur(function () {
260 | $('.query-wrapper').removeClass('active');
261 | });
262 |
263 | $resultsTable.delegate('a', 'click', function (e) {
264 | if ($(this).attr('href').match(/^file/)) {
265 | alert(
266 | 'For security concerns we cannot open local files. ' +
267 | 'You have to manually open the link by right clicking ' +
268 | 'on it and selecting "Open Link in New Tab"'
269 | );
270 | }
271 | });
272 |
273 | var $menu = $('.options-menu');
274 | $menu.delegate('button', 'click', function () {
275 | var $row = $menu.data('row');
276 | if ($row) {
277 | if ($(this).is('.delete')) {
278 | historyModel.deleteItem($row.data('id'));
279 | $row.fadeOut('fast', function () {
280 | $row.remove();
281 | });
282 | } else if ($(this).is('.more')) {
283 | $query.val('site:' + historyModel.getDomain($row.data('id')));
284 | $('#frm-search').submit();
285 | }
286 | }
287 | });
288 |
289 | $resultsTable.delegate('.entry .options', 'click', function () {
290 | var $this = $(this);
291 | if (!$this.is('.open')) {
292 | var pos = $(this).offset();
293 | $('.options-menu')
294 | .css({
295 | left: pos.left,
296 | top: pos.top + $this.outerHeight()
297 | })
298 | .show()
299 | .data('row', $this.closest('tr'));
300 | $('.entry .options.open').removeClass('open');
301 | $this.addClass('open');
302 | $('body').bind('click.menu', function () {
303 | $menu.hide().data('id', null);
304 | $('body').unbind('.menu');
305 | $this.removeClass('open');
306 | });
307 | } else {
308 | $menu.hide().data('id', null);
309 | $(this).removeClass('open');
310 | $menu.hide();
311 | }
312 | return false;
313 | });
314 |
315 | // Focus query box by default.
316 | $query.focus();
317 | });
318 |
319 | $(function () {
320 |
321 | $(window).resize(function () {
322 | $('#div-main').css('height', $(window).height() - 85);
323 | }).resize();
324 | $('#frm-search').submit();
325 | });
326 |
327 | })();
328 |
--------------------------------------------------------------------------------
/vendor/spin.js:
--------------------------------------------------------------------------------
1 | //fgnass.github.com/spin.js#v1.3.2
2 |
3 | /**
4 | * Copyright (c) 2011-2013 Felix Gnass
5 | * Licensed under the MIT license
6 | */
7 | (function(root, factory) {
8 |
9 | /* CommonJS */
10 | if (typeof exports == 'object') module.exports = factory()
11 |
12 | /* AMD module */
13 | else if (typeof define == 'function' && define.amd) define(factory)
14 |
15 | /* Browser global */
16 | else root.Spinner = factory()
17 | }
18 | (this, function() {
19 | "use strict";
20 |
21 | var prefixes = ['webkit', 'Moz', 'ms', 'O'] /* Vendor prefixes */
22 | , animations = {} /* Animation rules keyed by their name */
23 | , useCssAnimations /* Whether to use CSS animations or setTimeout */
24 |
25 | /**
26 | * Utility function to create elements. If no tag name is given,
27 | * a DIV is created. Optionally properties can be passed.
28 | */
29 | function createEl(tag, prop) {
30 | var el = document.createElement(tag || 'div')
31 | , n
32 |
33 | for(n in prop) el[n] = prop[n]
34 | return el
35 | }
36 |
37 | /**
38 | * Appends children and returns the parent.
39 | */
40 | function ins(parent /* child1, child2, ...*/) {
41 | for (var i=1, n=arguments.length; i> 1) : parseInt(o.left, 10) + mid) + 'px',
194 | top: (o.top == 'auto' ? tp.y-ep.y + (target.offsetHeight >> 1) : parseInt(o.top, 10) + mid) + 'px'
195 | })
196 | }
197 |
198 | el.setAttribute('role', 'progressbar')
199 | self.lines(el, self.opts)
200 |
201 | if (!useCssAnimations) {
202 | // No CSS animation support, use setTimeout() instead
203 | var i = 0
204 | , start = (o.lines - 1) * (1 - o.direction) / 2
205 | , alpha
206 | , fps = o.fps
207 | , f = fps/o.speed
208 | , ostep = (1-o.opacity) / (f*o.trail / 100)
209 | , astep = f/o.lines
210 |
211 | ;(function anim() {
212 | i++;
213 | for (var j = 0; j < o.lines; j++) {
214 | alpha = Math.max(1 - (i + (o.lines - j) * astep) % f * ostep, o.opacity)
215 |
216 | self.opacity(el, j * o.direction + start, alpha, o)
217 | }
218 | self.timeout = self.el && setTimeout(anim, ~~(1000/fps))
219 | })()
220 | }
221 | return self
222 | },
223 |
224 | /**
225 | * Stops and removes the Spinner.
226 | */
227 | stop: function() {
228 | var el = this.el
229 | if (el) {
230 | clearTimeout(this.timeout)
231 | if (el.parentNode) el.parentNode.removeChild(el)
232 | this.el = undefined
233 | }
234 | return this
235 | },
236 |
237 | /**
238 | * Internal method that draws the individual lines. Will be overwritten
239 | * in VML fallback mode below.
240 | */
241 | lines: function(el, o) {
242 | var i = 0
243 | , start = (o.lines - 1) * (1 - o.direction) / 2
244 | , seg
245 |
246 | function fill(color, shadow) {
247 | return css(createEl(), {
248 | position: 'absolute',
249 | width: (o.length+o.width) + 'px',
250 | height: o.width + 'px',
251 | background: color,
252 | boxShadow: shadow,
253 | transformOrigin: 'left',
254 | transform: 'rotate(' + ~~(360/o.lines*i+o.rotate) + 'deg) translate(' + o.radius+'px' +',0)',
255 | borderRadius: (o.corners * o.width>>1) + 'px'
256 | })
257 | }
258 |
259 | for (; i < o.lines; i++) {
260 | seg = css(createEl(), {
261 | position: 'absolute',
262 | top: 1+~(o.width/2) + 'px',
263 | transform: o.hwaccel ? 'translate3d(0,0,0)' : '',
264 | opacity: o.opacity,
265 | animation: useCssAnimations && addAnimation(o.opacity, o.trail, start + i * o.direction, o.lines) + ' ' + 1/o.speed + 's linear infinite'
266 | })
267 |
268 | if (o.shadow) ins(seg, css(fill('#000', '0 0 4px ' + '#000'), {top: 2+'px'}))
269 | ins(el, ins(seg, fill(getColor(o.color, i), '0 0 1px rgba(0,0,0,.1)')))
270 | }
271 | return el
272 | },
273 |
274 | /**
275 | * Internal method that adjusts the opacity of a single line.
276 | * Will be overwritten in VML fallback mode below.
277 | */
278 | opacity: function(el, i, val) {
279 | if (i < el.childNodes.length) el.childNodes[i].style.opacity = val
280 | }
281 |
282 | })
283 |
284 |
285 | function initVML() {
286 |
287 | /* Utility function to create a VML tag */
288 | function vml(tag, attr) {
289 | return createEl('<' + tag + ' xmlns="urn:schemas-microsoft.com:vml" class="spin-vml">', attr)
290 | }
291 |
292 | // No CSS transforms but VML support, add a CSS rule for VML elements:
293 | sheet.addRule('.spin-vml', 'behavior:url(#default#VML)')
294 |
295 | Spinner.prototype.lines = function(el, o) {
296 | var r = o.length+o.width
297 | , s = 2*r
298 |
299 | function grp() {
300 | return css(
301 | vml('group', {
302 | coordsize: s + ' ' + s,
303 | coordorigin: -r + ' ' + -r
304 | }),
305 | { width: s, height: s }
306 | )
307 | }
308 |
309 | var margin = -(o.width+o.length)*2 + 'px'
310 | , g = css(grp(), {position: 'absolute', top: margin, left: margin})
311 | , i
312 |
313 | function seg(i, dx, filter) {
314 | ins(g,
315 | ins(css(grp(), {rotation: 360 / o.lines * i + 'deg', left: ~~dx}),
316 | ins(css(vml('roundrect', {arcsize: o.corners}), {
317 | width: r,
318 | height: o.width,
319 | left: o.radius,
320 | top: -o.width>>1,
321 | filter: filter
322 | }),
323 | vml('fill', {color: getColor(o.color, i), opacity: o.opacity}),
324 | vml('stroke', {opacity: 0}) // transparent stroke to fix color bleeding upon opacity change
325 | )
326 | )
327 | )
328 | }
329 |
330 | if (o.shadow)
331 | for (i = 1; i <= o.lines; i++)
332 | seg(i, -2, 'progid:DXImageTransform.Microsoft.Blur(pixelradius=2,makeshadow=1,shadowopacity=.3)')
333 |
334 | for (i = 1; i <= o.lines; i++) seg(i)
335 | return ins(el, g)
336 | }
337 |
338 | Spinner.prototype.opacity = function(el, i, val, o) {
339 | var c = el.firstChild
340 | o = o.shadow && o.lines || 0
341 | if (c && i+o < c.childNodes.length) {
342 | c = c.childNodes[i+o]; c = c && c.firstChild; c = c && c.firstChild
343 | if (c) c.opacity = val
344 | }
345 | }
346 | }
347 |
348 | var probe = css(createEl('group'), {behavior: 'url(#default#VML)'})
349 |
350 | if (!vendor(probe, 'transform') && probe.adj) initVML()
351 | else useCssAnimations = vendor(probe, 'animation')
352 |
353 | return Spinner
354 |
355 | }));
356 |
--------------------------------------------------------------------------------
/css/history.css:
--------------------------------------------------------------------------------
1 | html {
2 | font-size: 62.5%;
3 | }
4 | body {
5 | background-color: white;
6 | color: black;
7 | font-family: "HelveticaNeue-Light", "Helvetica Neue Light", "Helvetica Neue", Helvetica, Arial, "Lucida Grande", sans-serif;
8 | font-size: 1.5rem;
9 | height: 100%;
10 | overflow: hidden;
11 | -webkit-user-select: none;
12 | }
13 |
14 | * { box-sizing: border-box; }
15 |
16 | header {
17 | background: #eee;
18 | padding: 22px 20px 0px 20px;
19 | height: 85px;
20 | box-shadow: 0 2px 4px rgba(0,0,0,0.2);
21 | }
22 |
23 | .logo {
24 | -webkit-transform: scale(0.7885);
25 | margin-top: -10px;
26 | }
27 |
28 | .bttn {
29 | height: 40px;
30 | padding: 10px 20px;
31 | font-size: 1em;
32 | color: white;
33 | border: 1px solid;
34 | border-radius: 3px;
35 | }
36 |
37 | .bttn:active {
38 | background-color: #eee;
39 | opacity: 1;
40 | box-shadow: inset 0 1px 2px rgba(0,0,0,0.1);
41 | }
42 | .bttn.search {
43 | background: #3261c3;
44 | border: 1px solid #3b2eca;
45 | }
46 |
47 | .query-wrapper {
48 | background: #fff;
49 | display: inline-block;
50 | padding: 7px;
51 | border: 1px solid #ceced6;
52 | width: 400px;
53 | font-size: 1.13em;
54 | height: 40px;
55 | position: relative;
56 | transition: border 0.25s linear;
57 | }
58 | .query-wrapper.active {
59 | border: 1px solid #7590f5!important;
60 | outline:none;
61 | }
62 |
63 | .query {
64 | border: 0;
65 | width: -webkit-calc(100% - 16px);
66 | width: calc(100% - 16px);
67 | outline: none;
68 | }
69 |
70 | .open-advanced {
71 | width: 16px;
72 | height: 16px;
73 | position: absolute;
74 | right: 5px;
75 | top: 12px;
76 | background: url(arrow_down.png);
77 | cursor: pointer;
78 | opacity: 0.2;
79 | transition: opacity 0.25s linear;
80 | }
81 |
82 | .query-wrapper:hover {
83 | border: 1px solid rgb(147, 147, 148);
84 | }
85 | .open-advanced:hover {
86 | opacity: 1;
87 | }
88 |
89 | .advanced-search {
90 | background: #fff;
91 | width: 400px;
92 | color: grey;
93 | padding: 10px;
94 | border: 1px solid #ceced6;
95 | box-shadow: 0 2px 4px rgba(0,0,0,0.2);
96 | position: absolute;
97 | top: 60px;
98 | display: none;
99 | z-index: 1;
100 | }
101 | .advanced-search fieldset {
102 | border: none;
103 | }
104 | .advanced-search label {
105 | width: 50px;
106 | display: inline-block;
107 | }
108 | .advanced-search input {
109 | border: 1px solid #ceced6;
110 | width: calc(100% - 50px);
111 | height: 30px;
112 | }
113 | .advanced-search input:focus {
114 | border:1px solid #4d90fe!important;
115 | outline: none;
116 | }
117 |
118 | @media all and (max-width:805px) {
119 | .query-wrapper, .advanced-search {
120 | width: 300px;
121 | }
122 | }
123 |
124 | @media all and (max-width: 710px) {
125 | .query-wrapper, .advanced-search {
126 | width: 230px;
127 | }
128 | }
129 |
130 | @media all and (max-width:632px) {
131 | .top-bar {
132 | display: none;
133 | }
134 | }
135 |
136 | @media all and (max-width:460px) {
137 | .query-wrapper, .advanced-search {
138 | width: 150px;
139 | }
140 | }
141 | @media all and (max-width:365) {
142 | .query-wrapper, .advanced-search {
143 | width: 80px;
144 | }
145 | }
146 | section.content {
147 | position: relative;
148 | }
149 |
150 | header .logo {
151 | float: left;
152 | }
153 |
154 | .top-bar {
155 | margin: 0;
156 | height: 38px;
157 | padding-left: 30px;
158 | float: left;
159 | position: relative;
160 | width: 200px;
161 | }
162 |
163 | .top-bar > * {
164 | float:right;
165 | }
166 |
167 | #div-main {
168 | height: 500px;
169 | overflow-y: scroll;
170 | width: 100%;
171 | clear: both;
172 | }
173 |
174 | .page-buttons {
175 | width: 100px;
176 | height: 32px;
177 | float: right;
178 | font-weight: bold;
179 | margin-right: 5px;
180 | margin-top: 5px;
181 | display: -webkit-flex;
182 | }
183 |
184 | .prev-page,.next-page {
185 | width: 32px;
186 | height: 32px;
187 | display: inline-block;
188 | border: 1px solid #ceced6;
189 | cursor: pointer;
190 | transition: border 0.25s linear;
191 | padding: 0;
192 | position: relative;
193 | opacity: 0.9;
194 | }
195 |
196 | .bttn {
197 | background: #F3F3F3;
198 | }
199 |
200 | .bttn[disabled], .bttn[disabled]:hover {
201 | opacity: 0.4;
202 | }
203 |
204 | .bttn:focus {
205 | outline: none;
206 | -webkit-transition: border-color 200ms;
207 | border-color: rgb(77, 144, 254);
208 | }
209 |
210 | .prev-page .img, .next-page .img {
211 | position: absolute;
212 | top:0;
213 | left:0;
214 | right:0;
215 | bottom:0;
216 | transition: opacity 0.25s linear;
217 | }
218 | .prev-page .img {
219 | background: url(prev.png);
220 | }
221 | .next-page .img {
222 | background: url(next.png)
223 | }
224 | .prev-page:hover .img, .next-page:hover .img {
225 | opacity: 1;
226 | }
227 | .prev-page:hover:not([disabled]),.next-page:hover:not([disabled]) {
228 | border: 1px solid black;
229 | }
230 | .page-no {
231 | display: inline-block;
232 | height: 32px;
233 | width: 26px;
234 | padding: 0;
235 | margin: 0;
236 | display: -webkit-flex;
237 | -webkit-justify-content: center;
238 | -webkit-align-items: center;
239 | }
240 |
241 | #tbl-main {
242 | width: 100%;
243 | }
244 | .hdr-day {
245 | width: 100%;
246 | height: 35px;
247 | border-bottom: 1px #e5e5e5 solid;
248 | background-color: rgb(248, 248, 248);
249 | }
250 | .hdr-day td {
251 | margin: 0;
252 | padding: 0;
253 | border-spacing: 0;
254 | }
255 | .hdr-day .date {
256 | font-size: 1.2em;
257 | font-family: "HelveticaNeue-Light", "Helvetica Neue Light", "Helvetica Neue", Helvetica, Arial, "Lucida Grande", sans-serif;
258 | font-weight: 300;
259 | -webkit-user-select: initial;
260 | }
261 |
262 | .hdr-day, .entry {
263 | cursor: pointer;
264 | }
265 |
266 | #tbl-main td {
267 | padding: 8px 10px;
268 | }
269 |
270 | .entry .time {
271 | color: #888;
272 | min-width: 56px;
273 | white-space: nowrap;
274 | }
275 |
276 | .entry {
277 | border-bottom: 1px #e5e5e5 solid;
278 | }
279 |
280 | .entry:hover {
281 | background-color: rgb(252, 252, 252)
282 | }
283 |
284 | #tbl-main input[type="checkbox"]{
285 | -webkit-appearance: none;
286 | height: 13px;
287 | width: 13px;
288 | border: 1px solid rgba(0, 0, 0, 0.25);
289 | border-radius: 2px;
290 | color: #444;
291 | text-shadow: 0 1px 0 rgb(240, 240, 240);
292 | }
293 | #tbl-main input[type="checkbox"]:checked {
294 | background-image:-webkit-image-set(url('') 1x, url('') 2x);
295 | }
296 | #tbl-main input[type="checkbox"]:focus {
297 | -webkit-transition: border-color 200ms;
298 | border-color: rgb(77, 144, 254);
299 | outline: none;
300 | }
301 |
302 | .entry a {
303 | background-position: 0px 1px;
304 | background-repeat: no-repeat;
305 | background-size: 16px;
306 | box-sizing: border-box;
307 | display: inline-block;
308 | max-width: 500px;
309 | overflow: hidden;
310 | padding: 1px 0px 4px 22px;
311 | text-overflow: ellipsis;
312 | white-space:nowrap;
313 | color: rgb(48, 57, 66);
314 | text-decoration: none;
315 | margin-right: 12px;
316 | -webkit-user-select: initial;
317 | }
318 |
319 | .entry a:hover {
320 | text-decoration: underline;
321 | }
322 |
323 | .entry .domain {
324 | color: rgb(151, 156, 160);
325 | font-size: 13px;
326 | -webkit-user-select: initial;
327 | }
328 |
329 | .dwn-arrow {
330 | width: 16px;
331 | height: 16px;
332 | display: inline-block;
333 | background: url(arrow_down.png);
334 | -webkit-transform: scale(0.6);
335 | margin-left: -1px;
336 | margin-top: -1px;
337 | }
338 |
339 | .entry .options {
340 | border: 1px solid rgb(192, 195, 198);
341 | opacity: 0.5;
342 | border-radius: 3px;
343 | align-items: flex-start;
344 | text-align: center;
345 | cursor: default;
346 | width: 16px;
347 | height: 16px;
348 | display: inline-block;
349 | margin-left: 10px;
350 | }
351 |
352 | .entry .options:hover {
353 | opacity: 1;
354 | }
355 |
356 | .entry .options.open {
357 | background: black;
358 | opacity: 1;
359 | }
360 |
361 | .entry .options.open .dwn-arrow {
362 | -webkit-filter: invert(100%);
363 | }
364 |
365 | .entry .item-content {
366 | display: flex;
367 | align-items: center;
368 | }
369 |
370 | menu {
371 | background: white;
372 | display: none;
373 | -webkit-box-shadow: 0 2px 4px rgba(0, 0, 0, .50);
374 | background: white;
375 | border-radius: 2px;
376 | color: black;
377 | cursor: default;
378 | margin: 0;
379 | outline: 1px solid rgba(0, 0, 0, 0.2);
380 | padding: 8px 0;
381 | position: fixed;
382 | white-space: nowrap;
383 | z-index: 99;
384 | line-height: 29px;
385 | }
386 |
387 | menu button {
388 | line-height: 29px;
389 | box-sizing: border-box;
390 | display: block;
391 | margin: 0;
392 | text-align: start;
393 | width: 100%;
394 | -webkit-appearance: none;
395 | background: transparent;
396 | border: 0;
397 | color: black;
398 | font: inherit;
399 | font-size: 13px;
400 | outline: none;
401 | overflow: hidden;
402 | padding: 0 19px;
403 | text-overflow: ellipsis;
404 | }
405 |
406 | menu button:hover {
407 | background: #d84938;
408 | color: white;
409 | }
410 |
411 | .bttn.delete {
412 | color: black;
413 | height: 32px;
414 | font-size: 0.9em;
415 | padding: 0 10px;
416 | margin-right: 5px;
417 | border: 1px solid #ceced6;
418 | transition: border 0.25s linear;
419 | margin-top: 5px;
420 | padding-bottom: 2px;
421 | }
422 |
423 | .bttn.delete:hover {
424 | border: 1px solid black;
425 | }
426 |
427 | .no-results td {
428 | padding: 50px;
429 | }
430 |
431 | #tbl-main input[type=checkbox] {
432 | margin-left: 10px;
433 | }
434 |
435 | #frm-search {
436 | -webkit-margin-start: 12px;
437 | float: left;
438 | }
439 |
440 | .feedback {
441 | position: fixed;
442 | bottom: 0;
443 | right: 30px;
444 | background: #eee;
445 | padding: 12px;
446 | border-radius: 4px;
447 | }
448 |
449 | #throbber {
450 | position: absolute;
451 | left: 0;
452 | right: 0;
453 | top: 0;
454 | bottom: 0;
455 | z-index: 2;
456 | background: #eee;
457 | display: none;
458 | }
459 |
460 | #throbber.active {
461 | display: block;
462 | }
463 |
464 | .dropdown.delete {
465 | width: 51px;
466 | opacity: 0.9;
467 | transition: opacity 0.25s linear;
468 | padding: 0;
469 | display: flex;
470 | align-items: center;
471 | vertical-align: middle;
472 | cursor: inherit;
473 | }
474 |
475 | .dropdown.delete:hover {
476 | opacity: 1;
477 | }
478 |
479 | .dropdown.delete:focus {
480 | outline: none;
481 | }
482 |
483 | .dropdown.delete .trash {
484 | background-image: url(trash.png);
485 | background-repeat: no-repeat;
486 | width: 32px;
487 | height: 32px;
488 | display: inline-block;
489 | /* Laziness */
490 | -webkit-transform: scale(0.7);
491 | }
492 |
493 | .dropdown.delete .dwn-arrow {
494 | margin-top: 7px;
495 | align-items: flex-start;
496 | text-align: center;
497 | margin-left: -4px;
498 | }
499 |
500 |
501 | .dropdown.delete.open {
502 | background-color: #eee;
503 | opacity: 1;
504 | box-shadow: inset 0 1px 2px rgba(0,0,0,0.1);
505 | }
506 |
507 | .delete-menu {
508 | display: none;
509 | position: absolute;
510 | right: 6px;
511 | top: 38px;
512 | }
513 |
--------------------------------------------------------------------------------
/vendor/mustache.js:
--------------------------------------------------------------------------------
1 | /*!
2 | * mustache.js - Logic-less {{mustache}} templates with JavaScript
3 | * http://github.com/janl/mustache.js
4 | */
5 |
6 | /*global define: false*/
7 |
8 | (function (root, factory) {
9 | if (typeof exports === "object" && exports) {
10 | factory(exports); // CommonJS
11 | } else {
12 | var mustache = {};
13 | factory(mustache);
14 | if (typeof define === "function" && define.amd) {
15 | define(mustache); // AMD
16 | } else {
17 | root.Mustache = mustache; //