Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
Y
yii2
Project
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
PSDI Army
yii2
Commits
69f0e5be
Commit
69f0e5be
authored
Feb 06, 2014
by
Alexander Makarov
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #2329 from tonydspaniard/1094-modal-generator-autocomplete-fix
1904 modal generator autocomplete fix
parents
d438913e
333630ae
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
1212 additions
and
698 deletions
+1212
-698
CHANGELOG.md
extensions/gii/CHANGELOG.md
+1
-0
gii.js
extensions/gii/assets/gii.js
+9
-0
typeahead.js
extensions/gii/assets/typeahead.js
+1198
-697
ActiveField.php
extensions/gii/components/ActiveField.php
+4
-1
No files found.
extensions/gii/CHANGELOG.md
View file @
69f0e5be
...
@@ -5,6 +5,7 @@ Yii Framework 2 gii extension Change Log
...
@@ -5,6 +5,7 @@ Yii Framework 2 gii extension Change Log
----------------------------
----------------------------
-
Bug #1405: fixed disambiguation of relation names generated by gii (qiangxue)
-
Bug #1405: fixed disambiguation of relation names generated by gii (qiangxue)
-
Bug #1904: Fixed autocomplete to work with underscore inputs "_" (tonydspaniard)
-
Bug #2298: Fixed the bug that Gii controller generator did not allow digit in the controller ID (qiangxue)
-
Bug #2298: Fixed the bug that Gii controller generator did not allow digit in the controller ID (qiangxue)
-
Bug: fixed controller in crud template to avoid returning query in findModel() (cebe)
-
Bug: fixed controller in crud template to avoid returning query in findModel() (cebe)
-
Enh #1624: generate rules for unique indexes (lucianobaraglia)
-
Enh #1624: generate rules for unique indexes (lucianobaraglia)
...
...
extensions/gii/assets/gii.js
View file @
69f0e5be
...
@@ -71,6 +71,15 @@ yii.gii = (function ($) {
...
@@ -71,6 +71,15 @@ yii.gii = (function ($) {
};
};
return
{
return
{
autocomplete
:
function
(
counter
,
data
)
{
var
datum
=
new
Bloodhound
({
datumTokenizer
:
function
(
d
){
return
Bloodhound
.
tokenizers
.
whitespace
(
d
.
word
);},
queryTokenizer
:
Bloodhound
.
tokenizers
.
whitespace
,
local
:
data
});
datum
.
initialize
();
jQuery
(
'.typeahead-'
+
counter
).
typeahead
(
null
,{
displayKey
:
'word'
,
source
:
datum
.
ttAdapter
()});
},
init
:
function
()
{
init
:
function
()
{
initHintBlocks
();
initHintBlocks
();
initStickyInputs
();
initStickyInputs
();
...
...
extensions/gii/assets/typeahead.js
View file @
69f0e5be
/*!
/*!
* typeahead.js 0.
9.3
* typeahead.js 0.
10.0
* https://github.com/twitter/typeahead
* https://github.com/twitter/typeahead
.js
* Copyright 2013 Twitter, Inc. and other contributors; Licensed MIT
* Copyright 2013 Twitter, Inc. and other contributors; Licensed MIT
*/
*/
(
function
(
$
)
{
(
function
(
$
)
{
var
VERSION
=
"0.9.3"
;
var
_
=
{
var
utils
=
{
isMsie
:
function
()
{
isMsie
:
function
()
{
var
match
=
/
(
msie
)
([\w
.
]
+
)
/i
.
exec
(
navigator
.
userAgent
);
return
/
(
msie|trident
)
/i
.
test
(
navigator
.
userAgent
)
?
navigator
.
userAgent
.
match
(
/
(
msie |rv:
)(\d
+
(
.
\d
+
)?)
/i
)[
2
]
:
false
;
return
match
?
parseInt
(
match
[
2
],
10
)
:
false
;
},
},
isBlankString
:
function
(
str
)
{
isBlankString
:
function
(
str
)
{
return
!
str
||
/^
\s
*$/
.
test
(
str
);
return
!
str
||
/^
\s
*$/
.
test
(
str
);
...
@@ -30,21 +28,12 @@
...
@@ -30,21 +28,12 @@
return
typeof
obj
===
"undefined"
;
return
typeof
obj
===
"undefined"
;
},
},
bind
:
$
.
proxy
,
bind
:
$
.
proxy
,
bindAll
:
function
(
obj
)
{
each
:
function
(
collection
,
cb
)
{
var
val
;
$
.
each
(
collection
,
reverseArgs
)
;
f
or
(
var
key
in
obj
)
{
f
unction
reverseArgs
(
index
,
value
)
{
$
.
isFunction
(
val
=
obj
[
key
])
&&
(
obj
[
key
]
=
$
.
proxy
(
val
,
obj
)
);
return
cb
(
value
,
index
);
}
}
},
},
indexOf
:
function
(
haystack
,
needle
)
{
for
(
var
i
=
0
;
i
<
haystack
.
length
;
i
++
)
{
if
(
haystack
[
i
]
===
needle
)
{
return
i
;
}
}
return
-
1
;
},
each
:
$
.
each
,
map
:
$
.
map
,
map
:
$
.
map
,
filter
:
$
.
grep
,
filter
:
$
.
grep
,
every
:
function
(
obj
,
test
)
{
every
:
function
(
obj
,
test
)
{
...
@@ -78,6 +67,12 @@
...
@@ -78,6 +67,12 @@
return
counter
++
;
return
counter
++
;
};
};
}(),
}(),
templatify
:
function
templatify
(
obj
)
{
return
$
.
isFunction
(
obj
)
?
obj
:
template
;
function
template
()
{
return
String
(
obj
);
}
},
defer
:
function
(
fn
)
{
defer
:
function
(
fn
)
{
setTimeout
(
fn
,
0
);
setTimeout
(
fn
,
0
);
},
},
...
@@ -123,69 +118,69 @@
...
@@ -123,69 +118,69 @@
return
result
;
return
result
;
};
};
},
},
tokenizeQuery
:
function
(
str
)
{
return
$
.
trim
(
str
).
toLowerCase
().
split
(
/
[\s]
+/
);
},
tokenizeText
:
function
(
str
)
{
return
$
.
trim
(
str
).
toLowerCase
().
split
(
/
[\s\-
_
]
+/
);
},
getProtocol
:
function
()
{
return
location
.
protocol
;
},
noop
:
function
()
{}
noop
:
function
()
{}
};
};
var
EventTarget
=
function
()
{
var
VERSION
=
"0.10.0"
;
var
eventSplitter
=
/
\s
+/
;
var
LruCache
=
function
(
root
,
undefined
)
{
return
{
function
LruCache
(
maxSize
)
{
on
:
function
(
events
,
callback
)
{
this
.
maxSize
=
maxSize
||
100
;
var
event
;
this
.
size
=
0
;
if
(
!
callback
)
{
this
.
hash
=
{};
return
this
;
this
.
list
=
new
List
();
}
}
this
.
_callbacks
=
this
.
_callbacks
||
{};
_
.
mixin
(
LruCache
.
prototype
,
{
events
=
events
.
split
(
eventSplitter
);
set
:
function
set
(
key
,
val
)
{
while
(
event
=
events
.
shift
())
{
var
tailItem
=
this
.
list
.
tail
,
node
;
this
.
_callbacks
[
event
]
=
this
.
_callbacks
[
event
]
||
[];
if
(
this
.
size
>=
this
.
maxSize
)
{
this
.
_callbacks
[
event
].
push
(
callback
);
this
.
list
.
remove
(
tailItem
);
delete
this
.
hash
[
tailItem
.
key
];
}
if
(
node
=
this
.
hash
[
key
])
{
node
.
val
=
val
;
this
.
list
.
moveToFront
(
node
);
}
else
{
node
=
new
Node
(
key
,
val
);
this
.
list
.
add
(
node
);
this
.
hash
[
key
]
=
node
;
this
.
size
++
;
}
}
return
this
;
},
},
trigger
:
function
(
events
,
data
)
{
get
:
function
get
(
key
)
{
var
event
,
callbacks
;
var
node
=
this
.
hash
[
key
];
if
(
!
this
.
_callbacks
)
{
if
(
node
)
{
return
this
;
this
.
list
.
moveToFront
(
node
);
}
return
node
.
val
;
events
=
events
.
split
(
eventSplitter
);
while
(
event
=
events
.
shift
())
{
if
(
callbacks
=
this
.
_callbacks
[
event
])
{
for
(
var
i
=
0
;
i
<
callbacks
.
length
;
i
+=
1
)
{
callbacks
[
i
].
call
(
this
,
{
type
:
event
,
data
:
data
});
}
}
}
}
});
function
List
()
{
this
.
head
=
this
.
tail
=
null
;
}
}
return
this
;
_
.
mixin
(
List
.
prototype
,
{
}
add
:
function
add
(
node
)
{
};
if
(
this
.
head
)
{
}();
node
.
next
=
this
.
head
;
var
EventBus
=
function
()
{
this
.
head
.
prev
=
node
;
var
namespace
=
"typeahead:"
;
function
EventBus
(
o
)
{
if
(
!
o
||
!
o
.
el
)
{
$
.
error
(
"EventBus initialized without el"
);
}
this
.
$el
=
$
(
o
.
el
);
}
}
utils
.
mixin
(
EventBus
.
prototype
,
{
this
.
head
=
node
;
trigger
:
function
(
type
)
{
this
.
tail
=
this
.
tail
||
node
;
var
args
=
[].
slice
.
call
(
arguments
,
1
);
},
this
.
$el
.
trigger
(
namespace
+
type
,
args
);
remove
:
function
remove
(
node
)
{
node
.
prev
?
node
.
prev
.
next
=
node
.
next
:
this
.
head
=
node
.
next
;
node
.
next
?
node
.
next
.
prev
=
node
.
prev
:
this
.
tail
=
node
.
prev
;
},
moveToFront
:
function
(
node
)
{
this
.
remove
(
node
);
this
.
add
(
node
);
}
}
});
});
return
EventBus
;
function
Node
(
key
,
val
)
{
}();
this
.
key
=
key
;
this
.
val
=
val
;
this
.
prev
=
this
.
next
=
null
;
}
return
LruCache
;
}(
this
);
var
PersistentStorage
=
function
()
{
var
PersistentStorage
=
function
()
{
var
ls
,
methods
;
var
ls
,
methods
;
try
{
try
{
...
@@ -215,7 +210,7 @@
...
@@ -215,7 +210,7 @@
return
decode
(
ls
.
getItem
(
this
.
_prefix
(
key
)));
return
decode
(
ls
.
getItem
(
this
.
_prefix
(
key
)));
},
},
set
:
function
(
key
,
val
,
ttl
)
{
set
:
function
(
key
,
val
,
ttl
)
{
if
(
utils
.
isNumber
(
ttl
))
{
if
(
_
.
isNumber
(
ttl
))
{
ls
.
setItem
(
this
.
_ttlKey
(
key
),
encode
(
now
()
+
ttl
));
ls
.
setItem
(
this
.
_ttlKey
(
key
),
encode
(
now
()
+
ttl
));
}
else
{
}
else
{
ls
.
removeItem
(
this
.
_ttlKey
(
key
));
ls
.
removeItem
(
this
.
_ttlKey
(
key
));
...
@@ -241,330 +236,630 @@
...
@@ -241,330 +236,630 @@
},
},
isExpired
:
function
(
key
)
{
isExpired
:
function
(
key
)
{
var
ttl
=
decode
(
ls
.
getItem
(
this
.
_ttlKey
(
key
)));
var
ttl
=
decode
(
ls
.
getItem
(
this
.
_ttlKey
(
key
)));
return
utils
.
isNumber
(
ttl
)
&&
now
()
>
ttl
?
true
:
false
;
return
_
.
isNumber
(
ttl
)
&&
now
()
>
ttl
?
true
:
false
;
}
}
};
};
}
else
{
}
else
{
methods
=
{
methods
=
{
get
:
utils
.
noop
,
get
:
_
.
noop
,
set
:
utils
.
noop
,
set
:
_
.
noop
,
remove
:
utils
.
noop
,
remove
:
_
.
noop
,
clear
:
utils
.
noop
,
clear
:
_
.
noop
,
isExpired
:
utils
.
noop
isExpired
:
_
.
noop
};
};
}
}
utils
.
mixin
(
PersistentStorage
.
prototype
,
methods
);
_
.
mixin
(
PersistentStorage
.
prototype
,
methods
);
return
PersistentStorage
;
return
PersistentStorage
;
function
now
()
{
function
now
()
{
return
new
Date
().
getTime
();
return
new
Date
().
getTime
();
}
}
function
encode
(
val
)
{
function
encode
(
val
)
{
return
JSON
.
stringify
(
utils
.
isUndefined
(
val
)
?
null
:
val
);
return
JSON
.
stringify
(
_
.
isUndefined
(
val
)
?
null
:
val
);
}
}
function
decode
(
val
)
{
function
decode
(
val
)
{
return
JSON
.
parse
(
val
);
return
JSON
.
parse
(
val
);
}
}
}();
}();
var
RequestCache
=
function
()
{
function
RequestCache
(
o
)
{
utils
.
bindAll
(
this
);
o
=
o
||
{};
this
.
sizeLimit
=
o
.
sizeLimit
||
10
;
this
.
cache
=
{};
this
.
cachedKeysByAge
=
[];
}
utils
.
mixin
(
RequestCache
.
prototype
,
{
get
:
function
(
url
)
{
return
this
.
cache
[
url
];
},
set
:
function
(
url
,
resp
)
{
var
requestToEvict
;
if
(
this
.
cachedKeysByAge
.
length
===
this
.
sizeLimit
)
{
requestToEvict
=
this
.
cachedKeysByAge
.
shift
();
delete
this
.
cache
[
requestToEvict
];
}
this
.
cache
[
url
]
=
resp
;
this
.
cachedKeysByAge
.
push
(
url
);
}
});
return
RequestCache
;
}();
var
Transport
=
function
()
{
var
Transport
=
function
()
{
var
pendingRequestsCount
=
0
,
pendingRequests
=
{},
maxPendingRequests
,
requestCache
;
var
pendingRequestsCount
=
0
,
pendingRequests
=
{},
maxPendingRequests
=
6
,
requestCache
=
new
LruCache
(
10
)
;
function
Transport
(
o
)
{
function
Transport
(
o
)
{
utils
.
bindAll
(
this
);
o
=
o
||
{};
o
=
utils
.
isString
(
o
)
?
{
this
.
_send
=
o
.
send
?
callbackToDeferred
(
o
.
send
)
:
$
.
ajax
;
url
:
o
this
.
_get
=
o
.
rateLimiter
?
o
.
rateLimiter
(
this
.
_get
)
:
this
.
_get
;
}
:
o
;
requestCache
=
requestCache
||
new
RequestCache
();
maxPendingRequests
=
utils
.
isNumber
(
o
.
maxParallelRequests
)
?
o
.
maxParallelRequests
:
maxPendingRequests
||
6
;
this
.
url
=
o
.
url
;
this
.
wildcard
=
o
.
wildcard
||
"%QUERY"
;
this
.
filter
=
o
.
filter
;
this
.
replace
=
o
.
replace
;
this
.
ajaxSettings
=
{
type
:
"get"
,
cache
:
o
.
cache
,
timeout
:
o
.
timeout
,
dataType
:
o
.
dataType
||
"json"
,
beforeSend
:
o
.
beforeSend
};
this
.
_get
=
(
/^throttle$/i
.
test
(
o
.
rateLimitFn
)
?
utils
.
throttle
:
utils
.
debounce
)(
this
.
_get
,
o
.
rateLimitWait
||
300
);
}
}
utils
.
mixin
(
Transport
.
prototype
,
{
Transport
.
setMaxPendingRequests
=
function
setMaxPendingRequests
(
num
)
{
_get
:
function
(
url
,
cb
)
{
maxPendingRequests
=
num
;
var
that
=
this
;
};
if
(
belowPendingRequestsThreshold
())
{
Transport
.
resetCache
=
function
clearCache
()
{
this
.
_sendRequest
(
url
).
done
(
done
);
requestCache
=
new
LruCache
(
10
);
};
_
.
mixin
(
Transport
.
prototype
,
{
_get
:
function
(
url
,
o
,
cb
)
{
var
that
=
this
,
jqXhr
;
if
(
jqXhr
=
pendingRequests
[
url
])
{
jqXhr
.
done
(
done
);
}
else
if
(
pendingRequestsCount
<
maxPendingRequests
)
{
pendingRequestsCount
++
;
pendingRequests
[
url
]
=
this
.
_send
(
url
,
o
).
done
(
done
).
always
(
always
);
}
else
{
}
else
{
this
.
onDeckRequestArgs
=
[].
slice
.
call
(
arguments
,
0
);
this
.
onDeckRequestArgs
=
[].
slice
.
call
(
arguments
,
0
);
}
}
function
done
(
resp
)
{
function
done
(
resp
)
{
var
data
=
that
.
filter
?
that
.
filter
(
resp
)
:
resp
;
cb
&&
cb
(
resp
);
cb
&&
cb
(
data
);
requestCache
.
set
(
url
,
resp
);
requestCache
.
set
(
url
,
resp
);
}
}
},
_sendRequest
:
function
(
url
)
{
var
that
=
this
,
jqXhr
=
pendingRequests
[
url
];
if
(
!
jqXhr
)
{
incrementPendingRequests
();
jqXhr
=
pendingRequests
[
url
]
=
$
.
ajax
(
url
,
this
.
ajaxSettings
).
always
(
always
);
}
return
jqXhr
;
function
always
()
{
function
always
()
{
decrementPendingRequests
()
;
pendingRequestsCount
--
;
pendingRequests
[
url
]
=
null
;
delete
pendingRequests
[
url
]
;
if
(
that
.
onDeckRequestArgs
)
{
if
(
that
.
onDeckRequestArgs
)
{
that
.
_get
.
apply
(
that
,
that
.
onDeckRequestArgs
);
that
.
_get
.
apply
(
that
,
that
.
onDeckRequestArgs
);
that
.
onDeckRequestArgs
=
null
;
that
.
onDeckRequestArgs
=
null
;
}
}
}
}
},
},
get
:
function
(
query
,
cb
)
{
get
:
function
(
url
,
o
,
cb
)
{
var
that
=
this
,
encodedQuery
=
encodeURIComponent
(
query
||
""
),
url
,
resp
;
var
that
=
this
,
resp
;
cb
=
cb
||
utils
.
noop
;
if
(
_
.
isFunction
(
o
))
{
url
=
this
.
replace
?
this
.
replace
(
this
.
url
,
encodedQuery
)
:
this
.
url
.
replace
(
this
.
wildcard
,
encodedQuery
);
cb
=
o
;
o
=
{};
}
if
(
resp
=
requestCache
.
get
(
url
))
{
if
(
resp
=
requestCache
.
get
(
url
))
{
utils
.
defer
(
function
()
{
_
.
defer
(
function
()
{
cb
(
that
.
filter
?
that
.
filter
(
resp
)
:
resp
);
cb
&&
cb
(
resp
);
});
});
}
else
{
}
else
{
this
.
_get
(
url
,
cb
);
this
.
_get
(
url
,
o
,
cb
);
}
}
return
!!
resp
;
return
!!
resp
;
}
}
});
});
return
Transport
;
return
Transport
;
function
incrementPendingRequests
()
{
function
callbackToDeferred
(
fn
)
{
pendingRequestsCount
++
;
return
function
customSendWrapper
(
url
,
o
)
{
var
deferred
=
$
.
Deferred
();
fn
(
url
,
o
,
onSuccess
,
onError
);
return
deferred
;
function
onSuccess
(
resp
)
{
_
.
defer
(
function
()
{
deferred
.
resolve
(
resp
);
});
}
}
function
decrementPendingRequests
()
{
function
onError
(
err
)
{
pendingRequestsCount
--
;
_
.
defer
(
function
()
{
deferred
.
reject
(
err
);
});
}
}
function
belowPendingRequestsThreshold
()
{
};
return
pendingRequestsCount
<
maxPendingRequests
;
}
}
}();
}();
var
Dataset
=
function
()
{
var
SearchIndex
=
function
()
{
var
keys
=
{
function
SearchIndex
(
o
)
{
thumbprint
:
"thumbprint"
,
o
=
o
||
{};
protocol
:
"protocol"
,
if
(
!
o
.
datumTokenizer
||
!
o
.
queryTokenizer
)
{
itemHash
:
"itemHash"
,
$
.
error
(
"datumTokenizer and queryTokenizer are both required"
);
adjacencyList
:
"adjacencyList"
};
function
Dataset
(
o
)
{
utils
.
bindAll
(
this
);
if
(
utils
.
isString
(
o
.
template
)
&&
!
o
.
engine
)
{
$
.
error
(
"no template engine specified"
);
}
}
if
(
!
o
.
local
&&
!
o
.
prefetch
&&
!
o
.
remote
)
{
this
.
datumTokenizer
=
o
.
datumTokenizer
;
$
.
error
(
"one of local, prefetch, or remote is required"
);
this
.
queryTokenizer
=
o
.
queryTokenizer
;
this
.
datums
=
[];
this
.
trie
=
newNode
();
}
_
.
mixin
(
SearchIndex
.
prototype
,
{
bootstrap
:
function
bootstrap
(
o
)
{
this
.
datums
=
o
.
datums
;
this
.
trie
=
o
.
trie
;
},
add
:
function
(
data
)
{
var
that
=
this
;
data
=
_
.
isArray
(
data
)
?
data
:
[
data
];
_
.
each
(
data
,
function
(
datum
)
{
var
id
,
tokens
;
id
=
that
.
datums
.
push
(
datum
)
-
1
;
tokens
=
normalizeTokens
(
that
.
datumTokenizer
(
datum
));
_
.
each
(
tokens
,
function
(
token
)
{
var
node
,
chars
,
ch
,
ids
;
node
=
that
.
trie
;
chars
=
token
.
split
(
""
);
while
(
ch
=
chars
.
shift
())
{
node
=
node
.
children
[
ch
]
||
(
node
.
children
[
ch
]
=
newNode
());
node
.
ids
.
push
(
id
);
}
}
this
.
name
=
o
.
name
||
utils
.
getUniqueId
();
this
.
limit
=
o
.
limit
||
5
;
this
.
minLength
=
o
.
minLength
||
1
;
this
.
header
=
o
.
header
;
this
.
footer
=
o
.
footer
;
this
.
valueKey
=
o
.
valueKey
||
"value"
;
this
.
template
=
compileTemplate
(
o
.
template
,
o
.
engine
,
this
.
valueKey
);
this
.
local
=
o
.
local
;
this
.
prefetch
=
o
.
prefetch
;
this
.
remote
=
o
.
remote
;
this
.
itemHash
=
{};
this
.
adjacencyList
=
{};
this
.
storage
=
o
.
name
?
new
PersistentStorage
(
o
.
name
)
:
null
;
}
utils
.
mixin
(
Dataset
.
prototype
,
{
_processLocalData
:
function
(
data
)
{
this
.
_mergeProcessedData
(
this
.
_processData
(
data
));
},
_loadPrefetchData
:
function
(
o
)
{
var
that
=
this
,
thumbprint
=
VERSION
+
(
o
.
thumbprint
||
""
),
storedThumbprint
,
storedProtocol
,
storedItemHash
,
storedAdjacencyList
,
isExpired
,
deferred
;
if
(
this
.
storage
)
{
storedThumbprint
=
this
.
storage
.
get
(
keys
.
thumbprint
);
storedProtocol
=
this
.
storage
.
get
(
keys
.
protocol
);
storedItemHash
=
this
.
storage
.
get
(
keys
.
itemHash
);
storedAdjacencyList
=
this
.
storage
.
get
(
keys
.
adjacencyList
);
}
isExpired
=
storedThumbprint
!==
thumbprint
||
storedProtocol
!==
utils
.
getProtocol
();
o
=
utils
.
isString
(
o
)
?
{
url
:
o
}
:
o
;
o
.
ttl
=
utils
.
isNumber
(
o
.
ttl
)
?
o
.
ttl
:
24
*
60
*
60
*
1
e3
;
if
(
storedItemHash
&&
storedAdjacencyList
&&
!
isExpired
)
{
this
.
_mergeProcessedData
({
itemHash
:
storedItemHash
,
adjacencyList
:
storedAdjacencyList
});
});
deferred
=
$
.
Deferred
().
resolve
();
});
}
else
{
},
deferred
=
$
.
getJSON
(
o
.
url
).
done
(
processPrefetchData
);
get
:
function
get
(
query
)
{
var
that
=
this
,
tokens
,
matches
;
tokens
=
normalizeTokens
(
this
.
queryTokenizer
(
query
));
_
.
each
(
tokens
,
function
(
token
)
{
var
node
,
chars
,
ch
,
ids
;
if
(
matches
&&
matches
.
length
===
0
)
{
return
false
;
}
}
return
deferred
;
node
=
that
.
trie
;
function
processPrefetchData
(
data
)
{
chars
=
token
.
split
(
""
);
var
filteredData
=
o
.
filter
?
o
.
filter
(
data
)
:
data
,
processedData
=
that
.
_processData
(
filteredData
),
itemHash
=
processedData
.
itemHash
,
adjacencyList
=
processedData
.
adjacencyList
;
while
(
node
&&
(
ch
=
chars
.
shift
()))
{
if
(
that
.
storage
)
{
node
=
node
.
children
[
ch
];
that
.
storage
.
set
(
keys
.
itemHash
,
itemHash
,
o
.
ttl
);
that
.
storage
.
set
(
keys
.
adjacencyList
,
adjacencyList
,
o
.
ttl
);
that
.
storage
.
set
(
keys
.
thumbprint
,
thumbprint
,
o
.
ttl
);
that
.
storage
.
set
(
keys
.
protocol
,
utils
.
getProtocol
(),
o
.
ttl
);
}
}
that
.
_mergeProcessedData
(
processedData
);
if
(
node
&&
chars
.
length
===
0
)
{
ids
=
node
.
ids
.
slice
(
0
);
matches
=
matches
?
getIntersection
(
matches
,
ids
)
:
ids
;
}
else
{
matches
=
[];
return
false
;
}
}
});
return
matches
?
_
.
map
(
unique
(
matches
),
function
(
id
)
{
return
that
.
datums
[
id
];
})
:
[];
},
},
_transformDatum
:
function
(
datum
)
{
serialize
:
function
serialize
(
)
{
var
value
=
utils
.
isString
(
datum
)
?
datum
:
datum
[
this
.
valueKey
],
tokens
=
datum
.
tokens
||
utils
.
tokenizeText
(
value
),
item
=
{
return
{
value
:
value
,
datums
:
this
.
datums
,
t
okens
:
tokens
t
rie
:
this
.
trie
};
};
if
(
utils
.
isString
(
datum
))
{
item
.
datum
=
{};
item
.
datum
[
this
.
valueKey
]
=
datum
;
}
else
{
item
.
datum
=
datum
;
}
}
item
.
tokens
=
utils
.
filter
(
item
.
tokens
,
function
(
token
)
{
return
!
utils
.
isBlankString
(
token
);
});
});
item
.
tokens
=
utils
.
map
(
item
.
tokens
,
function
(
token
)
{
return
SearchIndex
;
return
token
.
toLowerCase
();
function
normalizeTokens
(
tokens
)
{
});
tokens
=
_
.
filter
(
tokens
,
function
(
token
)
{
return
item
;
return
!!
token
;
},
_processData
:
function
(
data
)
{
var
that
=
this
,
itemHash
=
{},
adjacencyList
=
{};
utils
.
each
(
data
,
function
(
i
,
datum
)
{
var
item
=
that
.
_transformDatum
(
datum
),
id
=
utils
.
getUniqueId
(
item
.
value
);
itemHash
[
id
]
=
item
;
utils
.
each
(
item
.
tokens
,
function
(
i
,
token
)
{
var
character
=
token
.
charAt
(
0
),
adjacency
=
adjacencyList
[
character
]
||
(
adjacencyList
[
character
]
=
[
id
]);
!~
utils
.
indexOf
(
adjacency
,
id
)
&&
adjacency
.
push
(
id
);
});
});
tokens
=
_
.
map
(
tokens
,
function
(
token
)
{
return
token
.
toLowerCase
();
});
});
return
tokens
;
}
function
newNode
()
{
return
{
return
{
itemHash
:
itemHash
,
ids
:
[]
,
adjacencyList
:
adjacencyList
children
:
{}
};
};
},
}
_mergeProcessedData
:
function
(
processedData
)
{
function
unique
(
array
)
{
var
that
=
this
;
var
seen
=
{},
uniques
=
[];
utils
.
mixin
(
this
.
itemHash
,
processedData
.
itemHash
);
for
(
var
i
=
0
;
i
<
array
.
length
;
i
++
)
{
utils
.
each
(
processedData
.
adjacencyList
,
function
(
character
,
adjacency
)
{
if
(
!
seen
[
array
[
i
]])
{
var
masterAdjacency
=
that
.
adjacencyList
[
character
];
seen
[
array
[
i
]]
=
true
;
that
.
adjacencyList
[
character
]
=
masterAdjacency
?
masterAdjacency
.
concat
(
adjacency
)
:
adjacency
;
uniques
.
push
(
array
[
i
]);
}
}
return
uniques
;
}
function
getIntersection
(
arrayA
,
arrayB
)
{
var
ai
=
0
,
bi
=
0
,
intersection
=
[];
arrayA
=
arrayA
.
sort
(
compare
);
arrayB
=
arrayB
.
sort
(
compare
);
while
(
ai
<
arrayA
.
length
&&
bi
<
arrayB
.
length
)
{
if
(
arrayA
[
ai
]
<
arrayB
[
bi
])
{
ai
++
;
}
else
if
(
arrayA
[
ai
]
>
arrayB
[
bi
])
{
bi
++
;
}
else
{
intersection
.
push
(
arrayA
[
ai
]);
ai
++
;
bi
++
;
}
}
return
intersection
;
function
compare
(
a
,
b
)
{
return
a
-
b
;
}
}
}();
var
oParser
=
function
()
{
return
{
local
:
getLocal
,
prefetch
:
getPrefetch
,
remote
:
getRemote
};
function
getLocal
(
o
)
{
return
o
.
local
||
null
;
}
function
getPrefetch
(
o
)
{
var
prefetch
,
defaults
;
defaults
=
{
url
:
null
,
thumbprint
:
""
,
ttl
:
24
*
60
*
60
*
1
e3
,
filter
:
null
,
ajax
:
{}
};
if
(
prefetch
=
o
.
prefetch
||
null
)
{
prefetch
=
_
.
isString
(
prefetch
)
?
{
url
:
prefetch
}
:
prefetch
;
prefetch
=
_
.
mixin
(
defaults
,
prefetch
);
prefetch
.
thumbprint
=
VERSION
+
prefetch
.
thumbprint
;
prefetch
.
ajax
.
method
=
prefetch
.
ajax
.
method
||
"get"
;
prefetch
.
ajax
.
dataType
=
prefetch
.
ajax
.
dataType
||
"json"
;
!
prefetch
.
url
&&
$
.
error
(
"prefetch requires url to be set"
);
}
return
prefetch
;
}
function
getRemote
(
o
)
{
var
remote
,
defaults
;
defaults
=
{
url
:
null
,
wildcard
:
"%QUERY"
,
replace
:
null
,
rateLimitBy
:
"debounce"
,
rateLimitWait
:
300
,
send
:
null
,
filter
:
null
,
ajax
:
{}
};
if
(
remote
=
o
.
remote
||
null
)
{
remote
=
_
.
isString
(
remote
)
?
{
url
:
remote
}
:
remote
;
remote
=
_
.
mixin
(
defaults
,
remote
);
remote
.
rateLimiter
=
/^throttle$/i
.
test
(
remote
.
rateLimitBy
)
?
byThrottle
(
remote
.
rateLimitWait
)
:
byDebounce
(
remote
.
rateLimitWait
);
remote
.
ajax
.
method
=
remote
.
ajax
.
method
||
"get"
;
remote
.
ajax
.
dataType
=
remote
.
ajax
.
dataType
||
"json"
;
delete
remote
.
rateLimitBy
;
delete
remote
.
rateLimitWait
;
!
remote
.
url
&&
$
.
error
(
"remote requires url to be set"
);
}
return
remote
;
function
byDebounce
(
wait
)
{
return
function
(
fn
)
{
return
_
.
debounce
(
fn
,
wait
);
};
}
function
byThrottle
(
wait
)
{
return
function
(
fn
)
{
return
_
.
throttle
(
fn
,
wait
);
};
}
}
}();
var
Bloodhound
=
window
.
Bloodhound
=
function
()
{
var
keys
;
keys
=
{
data
:
"data"
,
protocol
:
"protocol"
,
thumbprint
:
"thumbprint"
};
function
Bloodhound
(
o
)
{
if
(
!
o
||
!
o
.
local
&&
!
o
.
prefetch
&&
!
o
.
remote
)
{
$
.
error
(
"one of local, prefetch, or remote is required"
);
}
this
.
limit
=
o
.
limit
||
5
;
this
.
sorter
=
o
.
sorter
||
noSort
;
this
.
dupDetector
=
o
.
dupDetector
||
ignoreDuplicates
;
this
.
local
=
oParser
.
local
(
o
);
this
.
prefetch
=
oParser
.
prefetch
(
o
);
this
.
remote
=
oParser
.
remote
(
o
);
this
.
cacheKey
=
this
.
prefetch
?
this
.
prefetch
.
cacheKey
||
this
.
prefetch
.
url
:
null
;
this
.
index
=
new
SearchIndex
({
datumTokenizer
:
o
.
datumTokenizer
,
queryTokenizer
:
o
.
queryTokenizer
});
});
this
.
storage
=
this
.
cacheKey
?
new
PersistentStorage
(
this
.
cacheKey
)
:
null
;
}
Bloodhound
.
tokenizers
=
{
whitespace
:
function
whitespaceTokenizer
(
s
)
{
return
s
.
split
(
/
\s
+/
);
},
},
_getLocalSuggestions
:
function
(
terms
)
{
nonword
:
function
nonwordTokenizer
(
s
)
{
var
that
=
this
,
firstChars
=
[],
lists
=
[],
shortestList
,
suggestions
=
[];
return
s
.
split
(
/
\W
+/
);
utils
.
each
(
terms
,
function
(
i
,
term
)
{
}
var
firstChar
=
term
.
charAt
(
0
);
};
!~
utils
.
indexOf
(
firstChars
,
firstChar
)
&&
firstChars
.
push
(
firstChar
);
_
.
mixin
(
Bloodhound
.
prototype
,
{
});
_loadPrefetch
:
function
loadPrefetch
(
o
)
{
utils
.
each
(
firstChars
,
function
(
i
,
firstChar
)
{
var
that
=
this
,
serialized
,
deferred
;
var
list
=
that
.
adjacencyList
[
firstChar
];
if
(
serialized
=
this
.
_readFromStorage
(
o
.
thumbprint
))
{
if
(
!
list
)
{
this
.
index
.
bootstrap
(
serialized
);
return
false
;
deferred
=
$
.
Deferred
().
resolve
();
}
else
{
deferred
=
$
.
ajax
(
o
.
url
,
o
.
ajax
).
done
(
handlePrefetchResponse
);
}
}
lists
.
push
(
list
);
return
deferred
;
if
(
!
shortestList
||
list
.
length
<
shortestList
.
length
)
{
function
handlePrefetchResponse
(
resp
)
{
shortestList
=
list
;
var
filtered
;
filtered
=
o
.
filter
?
o
.
filter
(
resp
)
:
resp
;
that
.
add
(
filtered
);
that
.
_saveToStorage
(
that
.
index
.
serialize
(),
o
.
thumbprint
,
o
.
ttl
);
}
}
});
},
if
(
lists
.
length
<
firstChars
.
length
)
{
_getFromRemote
:
function
getFromRemote
(
query
,
cb
)
{
return
[];
var
that
=
this
,
url
,
uriEncodedQuery
;
query
=
query
||
""
;
uriEncodedQuery
=
encodeURIComponent
(
query
);
url
=
this
.
remote
.
replace
?
this
.
remote
.
replace
(
this
.
remote
.
url
,
query
)
:
this
.
remote
.
url
.
replace
(
this
.
remote
.
wildcard
,
uriEncodedQuery
);
return
this
.
transport
.
get
(
url
,
this
.
remote
.
ajax
,
handleRemoteResponse
);
function
handleRemoteResponse
(
resp
)
{
var
filtered
=
that
.
remote
.
filter
?
that
.
remote
.
filter
(
resp
)
:
resp
;
cb
(
filtered
);
}
}
utils
.
each
(
shortestList
,
function
(
i
,
id
)
{
},
var
item
=
that
.
itemHash
[
id
],
isCandidate
,
isMatch
;
_saveToStorage
:
function
saveToStorage
(
data
,
thumbprint
,
ttl
)
{
isCandidate
=
utils
.
every
(
lists
,
function
(
list
)
{
if
(
this
.
storage
)
{
return
~
utils
.
indexOf
(
list
,
id
);
this
.
storage
.
set
(
keys
.
data
,
data
,
ttl
);
});
this
.
storage
.
set
(
keys
.
protocol
,
location
.
protocol
,
ttl
);
isMatch
=
isCandidate
&&
utils
.
every
(
terms
,
function
(
term
)
{
this
.
storage
.
set
(
keys
.
thumbprint
,
thumbprint
,
ttl
);
return
utils
.
some
(
item
.
tokens
,
function
(
token
)
{
}
return
token
.
indexOf
(
term
)
===
0
;
},
});
_readFromStorage
:
function
readFromStorage
(
thumbprint
)
{
var
stored
=
{};
if
(
this
.
storage
)
{
stored
.
data
=
this
.
storage
.
get
(
keys
.
data
);
stored
.
protocol
=
this
.
storage
.
get
(
keys
.
protocol
);
stored
.
thumbprint
=
this
.
storage
.
get
(
keys
.
thumbprint
);
}
isExpired
=
stored
.
thumbprint
!==
thumbprint
||
stored
.
protocol
!==
location
.
protocol
;
return
stored
.
data
&&
!
isExpired
?
stored
.
data
:
null
;
},
initialize
:
function
initialize
()
{
var
that
=
this
,
deferred
;
deferred
=
this
.
prefetch
?
this
.
_loadPrefetch
(
this
.
prefetch
)
:
$
.
Deferred
().
resolve
();
this
.
local
&&
deferred
.
done
(
addLocalToIndex
);
this
.
transport
=
this
.
remote
?
new
Transport
(
this
.
remote
)
:
null
;
this
.
initialize
=
function
initialize
()
{
return
deferred
.
promise
();
};
return
deferred
.
promise
();
function
addLocalToIndex
()
{
that
.
add
(
that
.
local
);
}
},
add
:
function
add
(
data
)
{
this
.
index
.
add
(
data
);
},
get
:
function
get
(
query
,
cb
)
{
var
that
=
this
,
matches
,
cacheHit
=
false
;
matches
=
this
.
index
.
get
(
query
).
sort
(
this
.
sorter
).
slice
(
0
,
this
.
limit
);
if
(
matches
.
length
<
this
.
limit
&&
this
.
transport
)
{
cacheHit
=
this
.
_getFromRemote
(
query
,
returnRemoteMatches
);
}
!
cacheHit
&&
cb
&&
cb
(
matches
);
function
returnRemoteMatches
(
remoteMatches
)
{
var
matchesWithBackfill
=
matches
.
slice
(
0
);
_
.
each
(
remoteMatches
,
function
(
remoteMatch
)
{
var
isDuplicate
;
isDuplicate
=
_
.
some
(
matchesWithBackfill
,
function
(
match
)
{
return
that
.
dupDetector
(
remoteMatch
,
match
);
});
});
isMatch
&&
suggestions
.
push
(
item
);
!
isDuplicate
&&
matchesWithBackfill
.
push
(
remoteMatch
);
return
matchesWithBackfill
.
length
<
that
.
limit
;
});
});
return
suggestions
;
cb
&&
cb
(
matchesWithBackfill
.
sort
(
that
.
sorter
));
}
},
},
initialize
:
function
()
{
ttAdapter
:
function
ttAdapter
()
{
var
deferred
;
return
_
.
bind
(
this
.
get
,
this
);
this
.
local
&&
this
.
_processLocalData
(
this
.
local
);
}
this
.
transport
=
this
.
remote
?
new
Transport
(
this
.
remote
)
:
null
;
});
deferred
=
this
.
prefetch
?
this
.
_loadPrefetchData
(
this
.
prefetch
)
:
$
.
Deferred
().
resolve
();
return
Bloodhound
;
this
.
local
=
this
.
prefetch
=
this
.
remote
=
null
;
function
noSort
()
{
this
.
initialize
=
function
()
{
return
0
;
return
deferred
;
}
function
ignoreDuplicates
()
{
return
false
;
}
}();
var
html
=
{
wrapper
:
'<span class="twitter-typeahead"></span>'
,
dropdown
:
'<span class="tt-dropdown-menu"></span>'
,
dataset
:
'<div class="tt-dataset-%CLASS%"></div>'
,
suggestions
:
'<span class="tt-suggestions"></span>'
,
suggestion
:
'<div class="tt-suggestion">%BODY%</div>'
};
};
return
deferred
;
var
css
=
{
wrapper
:
{
position
:
"relative"
,
display
:
"inline-block"
},
},
getSuggestions
:
function
(
query
,
cb
)
{
hint
:
{
var
that
=
this
,
terms
,
suggestions
,
cacheHit
=
false
;
position
:
"absolute"
,
if
(
query
.
length
<
this
.
minLength
)
{
top
:
"0"
,
return
;
left
:
"0"
,
borderColor
:
"transparent"
,
boxShadow
:
"none"
},
input
:
{
position
:
"relative"
,
verticalAlign
:
"top"
,
backgroundColor
:
"transparent"
},
inputWithNoHint
:
{
position
:
"relative"
,
verticalAlign
:
"top"
},
dropdown
:
{
position
:
"absolute"
,
top
:
"100%"
,
left
:
"0"
,
zIndex
:
"100"
,
display
:
"none"
},
suggestions
:
{
display
:
"block"
},
suggestion
:
{
whiteSpace
:
"nowrap"
,
cursor
:
"pointer"
},
suggestionChild
:
{
whiteSpace
:
"normal"
},
ltr
:
{
left
:
"0"
,
right
:
"auto"
},
rtl
:
{
left
:
"auto"
,
right
:
" 0"
}
}
terms
=
utils
.
tokenizeQuery
(
query
);
};
suggestions
=
this
.
_getLocalSuggestions
(
terms
).
slice
(
0
,
this
.
limit
);
if
(
_
.
isMsie
())
{
if
(
suggestions
.
length
<
this
.
limit
&&
this
.
transport
)
{
_
.
mixin
(
css
.
input
,
{
cacheHit
=
this
.
transport
.
get
(
query
,
processRemoteData
);
backgroundImage
:
"url()"
}
!
cacheHit
&&
cb
&&
cb
(
suggestions
);
function
processRemoteData
(
data
)
{
suggestions
=
suggestions
.
slice
(
0
);
utils
.
each
(
data
,
function
(
i
,
datum
)
{
var
item
=
that
.
_transformDatum
(
datum
),
isDuplicate
;
isDuplicate
=
utils
.
some
(
suggestions
,
function
(
suggestion
)
{
return
item
.
value
===
suggestion
.
value
;
});
});
!
isDuplicate
&&
suggestions
.
push
(
item
);
}
return
suggestions
.
length
<
that
.
limit
;
if
(
_
.
isMsie
()
&&
_
.
isMsie
()
<=
7
)
{
_
.
mixin
(
css
.
input
,
{
marginTop
:
"-1px"
});
});
cb
&&
cb
(
suggestions
);
}
}
var
EventBus
=
function
()
{
var
namespace
=
"typeahead:"
;
function
EventBus
(
o
)
{
if
(
!
o
||
!
o
.
el
)
{
$
.
error
(
"EventBus initialized without el"
);
}
this
.
$el
=
$
(
o
.
el
);
}
_
.
mixin
(
EventBus
.
prototype
,
{
trigger
:
function
(
type
)
{
var
args
=
[].
slice
.
call
(
arguments
,
1
);
this
.
$el
.
trigger
(
namespace
+
type
,
args
);
}
}
});
});
return
Dataset
;
return
EventBus
;
function
compileTemplate
(
template
,
engine
,
valueKey
)
{
}();
var
renderFn
,
compiledTemplate
;
var
EventEmitter
=
function
()
{
if
(
utils
.
isFunction
(
template
))
{
var
splitter
=
/
\s
+/
,
nextTick
=
getNextTick
();
renderFn
=
template
;
return
{
}
else
if
(
utils
.
isString
(
template
))
{
onSync
:
onSync
,
compiledTemplate
=
engine
.
compile
(
template
);
onAsync
:
onAsync
,
renderFn
=
utils
.
bind
(
compiledTemplate
.
render
,
compiledTemplate
);
off
:
off
,
trigger
:
trigger
};
function
on
(
method
,
types
,
cb
,
context
)
{
var
type
;
if
(
!
cb
)
{
return
this
;
}
types
=
types
.
split
(
splitter
);
cb
=
context
?
bindContext
(
cb
,
context
)
:
cb
;
this
.
_callbacks
=
this
.
_callbacks
||
{};
while
(
type
=
types
.
shift
())
{
this
.
_callbacks
[
type
]
=
this
.
_callbacks
[
type
]
||
{
sync
:
[],
async
:
[]
};
this
.
_callbacks
[
type
][
method
].
push
(
cb
);
}
return
this
;
}
function
onAsync
(
types
,
cb
,
context
)
{
return
on
.
call
(
this
,
"async"
,
types
,
cb
,
context
);
}
function
onSync
(
types
,
cb
,
context
)
{
return
on
.
call
(
this
,
"sync"
,
types
,
cb
,
context
);
}
function
off
(
types
)
{
var
type
;
if
(
!
this
.
_callbacks
)
{
return
this
;
}
types
=
types
.
split
(
splitter
);
while
(
type
=
types
.
shift
())
{
delete
this
.
_callbacks
[
type
];
}
return
this
;
}
function
trigger
(
types
)
{
var
that
=
this
,
type
,
callbacks
,
args
,
syncFlush
,
asyncFlush
;
if
(
!
this
.
_callbacks
)
{
return
this
;
}
types
=
types
.
split
(
splitter
);
args
=
[].
slice
.
call
(
arguments
,
1
);
while
((
type
=
types
.
shift
())
&&
(
callbacks
=
this
.
_callbacks
[
type
]))
{
syncFlush
=
getFlush
(
callbacks
.
sync
,
this
,
[
type
].
concat
(
args
));
asyncFlush
=
getFlush
(
callbacks
.
async
,
this
,
[
type
].
concat
(
args
));
syncFlush
()
&&
nextTick
(
asyncFlush
);
}
return
this
;
}
function
getFlush
(
callbacks
,
context
,
args
)
{
return
flush
;
function
flush
()
{
var
cancelled
;
for
(
var
i
=
0
;
!
cancelled
&&
i
<
callbacks
.
length
;
i
+=
1
)
{
cancelled
=
callbacks
[
i
].
apply
(
context
,
args
)
===
false
;
}
return
!
cancelled
;
}
}
function
getNextTick
()
{
var
nextTickFn
,
messageChannel
;
if
(
window
.
setImmediate
)
{
nextTickFn
=
function
nextTickSetImmediate
(
fn
)
{
setImmediate
(
function
()
{
fn
();
});
};
}
else
{
}
else
{
renderFn
=
function
(
context
)
{
nextTickFn
=
function
nextTickSetTimeout
(
fn
)
{
return
"<p>"
+
context
[
valueKey
]
+
"</p>"
;
setTimeout
(
function
()
{
fn
();
},
0
);
};
};
}
}
return
renderFn
;
return
nextTickFn
;
}
function
bindContext
(
fn
,
context
)
{
return
fn
.
bind
?
fn
.
bind
(
context
)
:
function
()
{
fn
.
apply
(
context
,
[].
slice
.
call
(
arguments
,
0
));
};
}
}
}();
}();
var
InputView
=
function
()
{
var
highlight
=
function
(
doc
)
{
function
InputView
(
o
)
{
var
defaults
=
{
var
that
=
this
;
node
:
null
,
utils
.
bindAll
(
this
);
pattern
:
null
,
this
.
specialKeyCodeMap
=
{
tagName
:
"strong"
,
className
:
null
,
wordsOnly
:
false
,
caseSensitive
:
false
};
return
function
hightlight
(
o
)
{
var
regex
;
o
=
_
.
mixin
({},
defaults
,
o
);
if
(
!
o
.
node
||
!
o
.
pattern
)
{
return
;
}
o
.
pattern
=
_
.
isArray
(
o
.
pattern
)
?
o
.
pattern
:
[
o
.
pattern
];
regex
=
getRegex
(
o
.
pattern
,
o
.
caseSensitive
,
o
.
wordsOnly
);
traverse
(
o
.
node
,
hightlightTextNode
);
function
hightlightTextNode
(
textNode
)
{
var
match
,
patternNode
;
if
(
match
=
regex
.
exec
(
textNode
.
data
))
{
wrapperNode
=
doc
.
createElement
(
o
.
tagName
);
o
.
className
&&
(
wrapperNode
.
className
=
o
.
className
);
patternNode
=
textNode
.
splitText
(
match
.
index
);
patternNode
.
splitText
(
match
[
0
].
length
);
wrapperNode
.
appendChild
(
patternNode
.
cloneNode
(
true
));
textNode
.
parentNode
.
replaceChild
(
wrapperNode
,
patternNode
);
}
return
!!
match
;
}
function
traverse
(
el
,
hightlightTextNode
)
{
var
childNode
,
TEXT_NODE_TYPE
=
3
;
for
(
var
i
=
0
;
i
<
el
.
childNodes
.
length
;
i
++
)
{
childNode
=
el
.
childNodes
[
i
];
if
(
childNode
.
nodeType
===
TEXT_NODE_TYPE
)
{
i
+=
hightlightTextNode
(
childNode
)
?
1
:
0
;
}
else
{
traverse
(
childNode
,
hightlightTextNode
);
}
}
}
};
function
getRegex
(
patterns
,
caseSensitive
,
wordsOnly
)
{
var
escapedPatterns
=
[],
regexStr
;
for
(
var
i
=
0
;
i
<
patterns
.
length
;
i
++
)
{
escapedPatterns
.
push
(
_
.
escapeRegExChars
(
patterns
[
i
]));
}
regexStr
=
wordsOnly
?
"
\\
b("
+
escapedPatterns
.
join
(
"|"
)
+
")
\\
b"
:
"("
+
escapedPatterns
.
join
(
"|"
)
+
")"
;
return
caseSensitive
?
new
RegExp
(
regexStr
)
:
new
RegExp
(
regexStr
,
"i"
);
}
}(
window
.
document
);
var
Input
=
function
()
{
var
specialKeyCodeMap
;
specialKeyCodeMap
=
{
9
:
"tab"
,
9
:
"tab"
,
27
:
"esc"
,
27
:
"esc"
,
37
:
"left"
,
37
:
"left"
,
...
@@ -573,84 +868,141 @@
...
@@ -573,84 +868,141 @@
38
:
"up"
,
38
:
"up"
,
40
:
"down"
40
:
"down"
};
};
function
Input
(
o
)
{
var
that
=
this
,
onBlur
,
onFocus
,
onKeydown
,
onInput
;
o
=
o
||
{};
if
(
!
o
.
input
)
{
$
.
error
(
"input is missing"
);
}
onBlur
=
_
.
bind
(
this
.
_onBlur
,
this
);
onFocus
=
_
.
bind
(
this
.
_onFocus
,
this
);
onKeydown
=
_
.
bind
(
this
.
_onKeydown
,
this
);
onInput
=
_
.
bind
(
this
.
_onInput
,
this
);
this
.
$hint
=
$
(
o
.
hint
);
this
.
$hint
=
$
(
o
.
hint
);
this
.
$input
=
$
(
o
.
input
).
on
(
"blur.tt"
,
this
.
_handleBlur
).
on
(
"focus.tt"
,
this
.
_handleFocus
).
on
(
"keydown.tt"
,
this
.
_handleSpecialKeyEvent
);
this
.
$input
=
$
(
o
.
input
).
on
(
"blur.tt"
,
onBlur
).
on
(
"focus.tt"
,
onFocus
).
on
(
"keydown.tt"
,
onKeydown
);
if
(
!
utils
.
isMsie
())
{
if
(
this
.
$hint
.
length
===
0
)
{
this
.
$input
.
on
(
"input.tt"
,
this
.
_compareQueryToInputValue
);
this
.
setHintValue
=
this
.
getHintValue
=
this
.
clearHint
=
_
.
noop
;
}
if
(
!
_
.
isMsie
())
{
this
.
$input
.
on
(
"input.tt"
,
onInput
);
}
else
{
}
else
{
this
.
$input
.
on
(
"keydown.tt keypress.tt cut.tt paste.tt"
,
function
(
$e
)
{
this
.
$input
.
on
(
"keydown.tt keypress.tt cut.tt paste.tt"
,
function
(
$e
)
{
if
(
that
.
specialKeyCodeMap
[
$e
.
which
||
$e
.
keyCode
])
{
if
(
specialKeyCodeMap
[
$e
.
which
||
$e
.
keyCode
])
{
return
;
return
;
}
}
utils
.
defer
(
that
.
_compareQueryToInputValue
);
_
.
defer
(
_
.
bind
(
that
.
_onInput
,
that
,
$e
)
);
});
});
}
}
this
.
query
=
this
.
$input
.
val
();
this
.
query
=
this
.
$input
.
val
();
this
.
$overflowHelper
=
buildOverflowHelper
(
this
.
$input
);
this
.
$overflowHelper
=
buildOverflowHelper
(
this
.
$input
);
}
}
utils
.
mixin
(
InputView
.
prototype
,
EventTarget
,
{
Input
.
normalizeQuery
=
function
(
str
)
{
_handleFocus
:
function
()
{
return
(
str
||
""
).
replace
(
/^
\s
*/g
,
""
).
replace
(
/
\s{2,}
/g
,
" "
);
};
_
.
mixin
(
Input
.
prototype
,
EventEmitter
,
{
_onBlur
:
function
onBlur
(
$e
)
{
this
.
resetInputValue
();
this
.
trigger
(
"blurred"
);
},
_onFocus
:
function
onFocus
(
$e
)
{
this
.
trigger
(
"focused"
);
this
.
trigger
(
"focused"
);
},
},
_handleBlur
:
function
()
{
_onKeydown
:
function
onKeydown
(
$e
)
{
this
.
trigger
(
"blured"
);
var
keyName
=
specialKeyCodeMap
[
$e
.
which
||
$e
.
keyCode
];
this
.
_managePreventDefault
(
keyName
,
$e
);
if
(
keyName
&&
this
.
_shouldTrigger
(
keyName
,
$e
))
{
this
.
trigger
(
keyName
+
"Keyed"
,
$e
);
}
},
},
_handleSpecialKeyEvent
:
function
(
$e
)
{
_onInput
:
function
onInput
(
$e
)
{
var
keyName
=
this
.
specialKeyCodeMap
[
$e
.
which
||
$e
.
keyCode
];
this
.
_checkInputValue
();
keyName
&&
this
.
trigger
(
keyName
+
"Keyed"
,
$e
);
},
},
_compareQueryToInputValue
:
function
()
{
_managePreventDefault
:
function
managePreventDefault
(
keyName
,
$e
)
{
var
inputValue
=
this
.
getInputValue
(),
isSameQuery
=
compareQueries
(
this
.
query
,
inputValue
),
isSameQueryExceptWhitespace
=
isSameQuery
?
this
.
query
.
length
!==
inputValue
.
length
:
false
;
var
preventDefault
,
hintValue
,
inputValue
;
if
(
isSameQueryExceptWhitespace
)
{
switch
(
keyName
)
{
this
.
trigger
(
"whitespaceChanged"
,
{
case
"tab"
:
value
:
this
.
query
hintValue
=
this
.
getHintValue
();
});
inputValue
=
this
.
getInputValue
();
}
else
if
(
!
isSameQuery
)
{
preventDefault
=
hintValue
&&
hintValue
!==
inputValue
&&
!
withModifier
(
$e
);
this
.
trigger
(
"queryChanged"
,
{
break
;
value
:
this
.
query
=
inputValue
});
case
"up"
:
case
"down"
:
preventDefault
=
!
withModifier
(
$e
);
break
;
default
:
preventDefault
=
false
;
}
}
preventDefault
&&
$e
.
preventDefault
();
},
},
destroy
:
function
()
{
_shouldTrigger
:
function
shouldTrigger
(
keyName
,
$e
)
{
this
.
$hint
.
off
(
".tt"
);
var
trigger
;
this
.
$input
.
off
(
".tt"
);
switch
(
keyName
)
{
this
.
$hint
=
this
.
$input
=
this
.
$overflowHelper
=
null
;
case
"tab"
:
trigger
=
!
withModifier
(
$e
);
break
;
default
:
trigger
=
true
;
}
return
trigger
;
},
_checkInputValue
:
function
checkInputValue
()
{
var
inputValue
,
areEquivalent
,
hasDifferentWhitespace
;
inputValue
=
this
.
getInputValue
();
areEquivalent
=
areQueriesEquivalent
(
inputValue
,
this
.
query
);
hasDifferentWhitespace
=
areEquivalent
?
this
.
query
.
length
!==
inputValue
.
length
:
false
;
if
(
!
areEquivalent
)
{
this
.
trigger
(
"queryChanged"
,
this
.
query
=
inputValue
);
}
else
if
(
hasDifferentWhitespace
)
{
this
.
trigger
(
"whitespaceChanged"
,
this
.
query
);
}
},
},
focus
:
function
()
{
focus
:
function
focus
()
{
this
.
$input
.
focus
();
this
.
$input
.
focus
();
},
},
blur
:
function
()
{
blur
:
function
blur
()
{
this
.
$input
.
blur
();
this
.
$input
.
blur
();
},
},
getQuery
:
function
()
{
getQuery
:
function
getQuery
()
{
return
this
.
query
;
return
this
.
query
;
},
},
setQuery
:
function
(
query
)
{
setQuery
:
function
setQuery
(
query
)
{
this
.
query
=
query
;
this
.
query
=
query
;
},
},
getInputValue
:
function
()
{
getInputValue
:
function
getInputValue
()
{
return
this
.
$input
.
val
();
return
this
.
$input
.
val
();
},
},
setInputValue
:
function
(
value
,
silent
)
{
setInputValue
:
function
setInputValue
(
value
,
silent
)
{
this
.
$input
.
val
(
value
);
this
.
$input
.
val
(
value
);
!
silent
&&
this
.
_c
ompareQueryTo
InputValue
();
!
silent
&&
this
.
_c
heck
InputValue
();
},
},
getHintValue
:
function
()
{
getHintValue
:
function
getHintValue
()
{
return
this
.
$hint
.
val
();
return
this
.
$hint
.
val
();
},
},
setHintValue
:
function
(
value
)
{
setHintValue
:
function
setHintValue
(
value
)
{
this
.
$hint
.
val
(
value
);
this
.
$hint
.
val
(
value
);
},
},
getLanguageDirection
:
function
()
{
resetInputValue
:
function
resetInputValue
()
{
this
.
$input
.
val
(
this
.
query
);
},
clearHint
:
function
clearHint
()
{
this
.
$hint
.
val
(
""
);
},
getLanguageDirection
:
function
getLanguageDirection
()
{
return
(
this
.
$input
.
css
(
"direction"
)
||
"ltr"
).
toLowerCase
();
return
(
this
.
$input
.
css
(
"direction"
)
||
"ltr"
).
toLowerCase
();
},
},
isOverflow
:
function
()
{
hasOverflow
:
function
hasOverflow
()
{
var
constraint
=
this
.
$input
.
width
()
-
2
;
this
.
$overflowHelper
.
text
(
this
.
getInputValue
());
this
.
$overflowHelper
.
text
(
this
.
getInputValue
());
return
this
.
$overflowHelper
.
width
()
>
this
.
$input
.
width
()
;
return
this
.
$overflowHelper
.
width
()
>
=
constraint
;
},
},
isCursorAtEnd
:
function
()
{
isCursorAtEnd
:
function
()
{
var
valueLength
=
this
.
$input
.
val
().
length
,
selectionStart
=
this
.
$input
[
0
].
selectionStart
,
range
;
var
valueLength
,
selectionStart
,
range
;
if
(
utils
.
isNumber
(
selectionStart
))
{
valueLength
=
this
.
$input
.
val
().
length
;
selectionStart
=
this
.
$input
[
0
].
selectionStart
;
if
(
_
.
isNumber
(
selectionStart
))
{
return
selectionStart
===
valueLength
;
return
selectionStart
===
valueLength
;
}
else
if
(
document
.
selection
)
{
}
else
if
(
document
.
selection
)
{
range
=
document
.
selection
.
createRange
();
range
=
document
.
selection
.
createRange
();
...
@@ -658,13 +1010,17 @@
...
@@ -658,13 +1010,17 @@
return
valueLength
===
range
.
text
.
length
;
return
valueLength
===
range
.
text
.
length
;
}
}
return
true
;
return
true
;
},
destroy
:
function
destroy
()
{
this
.
$hint
.
off
(
".tt"
);
this
.
$input
.
off
(
".tt"
);
this
.
$hint
=
this
.
$input
=
this
.
$overflowHelper
=
null
;
}
}
});
});
return
Input
View
;
return
Input
;
function
buildOverflowHelper
(
$input
)
{
function
buildOverflowHelper
(
$input
)
{
return
$
(
"<span></span>"
).
css
({
return
$
(
'<pre aria-hidden="true"></pre>'
).
css
({
position
:
"absolute"
,
position
:
"absolute"
,
left
:
"-9999px"
,
visibility
:
"hidden"
,
visibility
:
"hidden"
,
whiteSpace
:
"nowrap"
,
whiteSpace
:
"nowrap"
,
fontFamily
:
$input
.
css
(
"font-family"
),
fontFamily
:
$input
.
css
(
"font-family"
),
...
@@ -679,452 +1035,597 @@
...
@@ -679,452 +1035,597 @@
textTransform
:
$input
.
css
(
"text-transform"
)
textTransform
:
$input
.
css
(
"text-transform"
)
}).
insertAfter
(
$input
);
}).
insertAfter
(
$input
);
}
}
function
compareQueries
(
a
,
b
)
{
function
areQueriesEquivalent
(
a
,
b
)
{
a
=
(
a
||
""
).
replace
(
/^
\s
*/g
,
""
).
replace
(
/
\s{2,}
/g
,
" "
);
return
Input
.
normalizeQuery
(
a
)
===
Input
.
normalizeQuery
(
b
);
b
=
(
b
||
""
).
replace
(
/^
\s
*/g
,
""
).
replace
(
/
\s{2,}
/g
,
" "
);
}
return
a
===
b
;
function
withModifier
(
$e
)
{
return
$e
.
altKey
||
$e
.
ctrlKey
||
$e
.
metaKey
||
$e
.
shiftKey
;
}
}
}();
}();
var
DropdownView
=
function
()
{
var
Dataset
=
function
()
{
var
html
=
{
var
datasetKey
=
"ttDataset"
,
valueKey
=
"ttValue"
,
datumKey
=
"ttDatum"
;
suggestionsList
:
'<span class="tt-suggestions"></span>'
function
Dataset
(
o
)
{
},
css
=
{
o
=
o
||
{};
suggestionsList
:
{
o
.
templates
=
o
.
templates
||
{};
display
:
"block"
if
(
!
o
.
source
)
{
$
.
error
(
"missing source"
);
}
this
.
query
=
null
;
this
.
highlight
=
!!
o
.
highlight
;
this
.
name
=
o
.
name
||
_
.
getUniqueId
();
this
.
source
=
o
.
source
;
this
.
valueKey
=
o
.
displayKey
||
"value"
;
this
.
templates
=
getTemplates
(
o
.
templates
,
this
.
valueKey
);
this
.
$el
=
$
(
html
.
dataset
.
replace
(
"%CLASS%"
,
this
.
name
));
}
Dataset
.
extractDatasetName
=
function
extractDatasetName
(
el
)
{
return
$
(
el
).
data
(
datasetKey
);
};
Dataset
.
extractValue
=
function
extractDatum
(
el
)
{
return
$
(
el
).
data
(
valueKey
);
};
Dataset
.
extractDatum
=
function
extractDatum
(
el
)
{
return
$
(
el
).
data
(
datumKey
);
};
_
.
mixin
(
Dataset
.
prototype
,
EventEmitter
,
{
_render
:
function
render
(
query
,
suggestions
)
{
if
(
!
this
.
$el
)
{
return
;
}
var
that
=
this
,
hasSuggestions
;
this
.
$el
.
empty
();
hasSuggestions
=
suggestions
&&
suggestions
.
length
;
if
(
!
hasSuggestions
&&
this
.
templates
.
empty
)
{
this
.
$el
.
html
(
getEmptyHtml
()).
prepend
(
that
.
templates
.
header
?
getHeaderHtml
()
:
null
).
append
(
that
.
templates
.
footer
?
getFooterHtml
()
:
null
);
}
else
if
(
hasSuggestions
)
{
this
.
$el
.
html
(
getSuggestionsHtml
()).
prepend
(
that
.
templates
.
header
?
getHeaderHtml
()
:
null
).
append
(
that
.
templates
.
footer
?
getFooterHtml
()
:
null
);
}
this
.
trigger
(
"rendered"
);
function
getEmptyHtml
()
{
return
that
.
templates
.
empty
({
query
:
query
});
}
function
getSuggestionsHtml
()
{
var
$suggestions
;
$suggestions
=
$
(
html
.
suggestions
).
css
(
css
.
suggestions
).
append
(
_
.
map
(
suggestions
,
getSuggestionNode
));
that
.
highlight
&&
highlight
({
node
:
$suggestions
[
0
],
pattern
:
query
});
return
$suggestions
;
function
getSuggestionNode
(
suggestion
)
{
var
$el
,
innerHtml
,
outerHtml
;
innerHtml
=
that
.
templates
.
suggestion
(
suggestion
);
outerHtml
=
html
.
suggestion
.
replace
(
"%BODY%"
,
innerHtml
);
$el
=
$
(
outerHtml
).
data
(
datasetKey
,
that
.
name
).
data
(
valueKey
,
suggestion
[
that
.
valueKey
]).
data
(
datumKey
,
suggestion
);
$el
.
children
().
each
(
function
()
{
$
(
this
).
css
(
css
.
suggestionChild
);
});
return
$el
;
}
}
function
getHeaderHtml
()
{
return
that
.
templates
.
header
({
query
:
query
,
isEmpty
:
!
hasSuggestions
});
}
function
getFooterHtml
()
{
return
that
.
templates
.
footer
({
query
:
query
,
isEmpty
:
!
hasSuggestions
});
}
},
},
suggestion
:
{
getRoot
:
function
getRoot
()
{
whiteSpace
:
"nowrap"
,
return
this
.
$el
;
cursor
:
"pointer"
},
},
suggestionChild
:
{
update
:
function
update
(
query
)
{
whiteSpace
:
"normal"
var
that
=
this
;
this
.
query
=
query
;
this
.
source
(
query
,
renderIfQueryIsSame
);
function
renderIfQueryIsSame
(
suggestions
)
{
query
===
that
.
query
&&
that
.
_render
(
query
,
suggestions
);
}
},
clear
:
function
clear
()
{
this
.
_render
(
this
.
query
||
""
);
},
isEmpty
:
function
isEmpty
()
{
return
this
.
$el
.
is
(
":empty"
);
},
destroy
:
function
destroy
()
{
this
.
$el
=
null
;
}
}
});
return
Dataset
;
function
getTemplates
(
templates
,
valueKey
)
{
return
{
empty
:
templates
.
empty
&&
_
.
templatify
(
templates
.
empty
),
header
:
templates
.
header
&&
_
.
templatify
(
templates
.
header
),
footer
:
templates
.
footer
&&
_
.
templatify
(
templates
.
footer
),
suggestion
:
templates
.
suggestion
||
suggestionTemplate
};
};
function
DropdownView
(
o
)
{
function
suggestionTemplate
(
context
)
{
utils
.
bindAll
(
this
);
return
"<p>"
+
context
[
valueKey
]
+
"</p>"
;
}
}
}();
var
Dropdown
=
function
()
{
function
Dropdown
(
o
)
{
var
that
=
this
,
onMouseEnter
,
onMouseLeave
,
onSuggestionClick
,
onSuggestionMouseEnter
,
onSuggestionMouseLeave
;
o
=
o
||
{};
if
(
!
o
.
menu
)
{
$
.
error
(
"menu is required"
);
}
this
.
isOpen
=
false
;
this
.
isOpen
=
false
;
this
.
isEmpty
=
true
;
this
.
isEmpty
=
true
;
this
.
isMouseOverDropdown
=
false
;
this
.
isMouseOverDropdown
=
false
;
this
.
$menu
=
$
(
o
.
menu
).
on
(
"mouseenter.tt"
,
this
.
_handleMouseenter
).
on
(
"mouseleave.tt"
,
this
.
_handleMouseleave
).
on
(
"click.tt"
,
".tt-suggestion"
,
this
.
_handleSelection
).
on
(
"mouseover.tt"
,
".tt-suggestion"
,
this
.
_handleMouseover
);
this
.
datasets
=
_
.
map
(
o
.
datasets
,
initializeDataset
);
onMouseEnter
=
_
.
bind
(
this
.
_onMouseEnter
,
this
);
onMouseLeave
=
_
.
bind
(
this
.
_onMouseLeave
,
this
);
onSuggestionClick
=
_
.
bind
(
this
.
_onSuggestionClick
,
this
);
onSuggestionMouseEnter
=
_
.
bind
(
this
.
_onSuggestionMouseEnter
,
this
);
onSuggestionMouseLeave
=
_
.
bind
(
this
.
_onSuggestionMouseLeave
,
this
);
this
.
$menu
=
$
(
o
.
menu
).
on
(
"mouseenter.tt"
,
onMouseEnter
).
on
(
"mouseleave.tt"
,
onMouseLeave
).
on
(
"click.tt"
,
".tt-suggestion"
,
onSuggestionClick
).
on
(
"mouseenter.tt"
,
".tt-suggestion"
,
onSuggestionMouseEnter
).
on
(
"mouseleave.tt"
,
".tt-suggestion"
,
onSuggestionMouseLeave
);
_
.
each
(
this
.
datasets
,
function
(
dataset
)
{
that
.
$menu
.
append
(
dataset
.
getRoot
());
dataset
.
onSync
(
"rendered"
,
that
.
_onRendered
,
that
);
});
}
}
utils
.
mixin
(
DropdownView
.
prototype
,
EventTarget
,
{
_
.
mixin
(
Dropdown
.
prototype
,
EventEmitter
,
{
_
handleMouseenter
:
function
(
)
{
_
onMouseEnter
:
function
onMouseEnter
(
$e
)
{
this
.
isMouseOverDropdown
=
true
;
this
.
isMouseOverDropdown
=
true
;
},
},
_
handleMouseleave
:
function
(
)
{
_
onMouseLeave
:
function
onMouseLeave
(
$e
)
{
this
.
isMouseOverDropdown
=
false
;
this
.
isMouseOverDropdown
=
false
;
},
},
_handleMouseover
:
function
(
$e
)
{
_onSuggestionClick
:
function
onSuggestionClick
(
$e
)
{
var
$suggestion
=
$
(
$e
.
currentTarget
);
this
.
trigger
(
"suggestionClicked"
,
$
(
$e
.
currentTarget
));
this
.
_getSuggestions
().
removeClass
(
"tt-is-under-cursor"
);
$suggestion
.
addClass
(
"tt-is-under-cursor"
);
},
},
_
handleSelection
:
function
(
$e
)
{
_
onSuggestionMouseEnter
:
function
onSuggestionMouseEnter
(
$e
)
{
var
$suggestion
=
$
(
$e
.
currentTarget
);
this
.
_removeCursor
(
);
this
.
trigger
(
"suggestionSelected"
,
extractSuggestion
(
$suggestion
)
);
this
.
_setCursor
(
$
(
$e
.
currentTarget
),
true
);
},
},
_show
:
function
()
{
_onSuggestionMouseLeave
:
function
onSuggestionMouseLeave
(
$e
)
{
this
.
$menu
.
css
(
"display"
,
"block"
);
this
.
_removeCursor
();
},
_onRendered
:
function
onRendered
()
{
this
.
isEmpty
=
_
.
every
(
this
.
datasets
,
isDatasetEmpty
);
this
.
isEmpty
?
this
.
_hide
()
:
this
.
isOpen
&&
this
.
_show
();
this
.
trigger
(
"datasetRendered"
);
function
isDatasetEmpty
(
dataset
)
{
return
dataset
.
isEmpty
();
}
},
},
_hide
:
function
()
{
_hide
:
function
()
{
this
.
$menu
.
hide
();
this
.
$menu
.
hide
();
},
},
_moveCursor
:
function
(
increment
)
{
_show
:
function
()
{
var
$suggestions
,
$cur
,
nextIndex
,
$underCursor
;
this
.
$menu
.
css
(
"display"
,
"block"
);
if
(
!
this
.
isVisible
())
{
},
_getSuggestions
:
function
getSuggestions
()
{
return
this
.
$menu
.
find
(
".tt-suggestion"
);
},
_getCursor
:
function
getCursor
()
{
return
this
.
$menu
.
find
(
".tt-cursor"
).
first
();
},
_setCursor
:
function
setCursor
(
$el
,
silent
)
{
$el
.
first
().
addClass
(
"tt-cursor"
);
!
silent
&&
this
.
trigger
(
"cursorMoved"
);
},
_removeCursor
:
function
removeCursor
()
{
this
.
_getCursor
().
removeClass
(
"tt-cursor"
);
},
_moveCursor
:
function
moveCursor
(
increment
)
{
var
$suggestions
,
$oldCursor
,
newCursorIndex
,
$newCursor
;
if
(
!
this
.
isOpen
)
{
return
;
return
;
}
}
$oldCursor
=
this
.
_getCursor
();
$suggestions
=
this
.
_getSuggestions
();
$suggestions
=
this
.
_getSuggestions
();
$cur
=
$suggestions
.
filter
(
".tt-is-under-cursor"
);
this
.
_removeCursor
();
$cur
.
removeClass
(
"tt-is-under-cursor"
);
newCursorIndex
=
$suggestions
.
index
(
$oldCursor
)
+
increment
;
nextIndex
=
$suggestions
.
index
(
$cur
)
+
increment
;
newCursorIndex
=
(
newCursorIndex
+
1
)
%
(
$suggestions
.
length
+
1
)
-
1
;
nextIndex
=
(
nextIndex
+
1
)
%
(
$suggestions
.
length
+
1
)
-
1
;
if
(
newCursorIndex
===
-
1
)
{
if
(
nextIndex
===
-
1
)
{
this
.
trigger
(
"cursorRemoved"
);
this
.
trigger
(
"cursorRemoved"
);
return
;
return
;
}
else
if
(
ne
xt
Index
<
-
1
)
{
}
else
if
(
ne
wCursor
Index
<
-
1
)
{
ne
xt
Index
=
$suggestions
.
length
-
1
;
ne
wCursor
Index
=
$suggestions
.
length
-
1
;
}
}
$underCursor
=
$suggestions
.
eq
(
nextIndex
).
addClass
(
"tt-is-under-cursor"
);
this
.
_setCursor
(
$newCursor
=
$suggestions
.
eq
(
newCursorIndex
)
);
this
.
_ensureVisib
ility
(
$under
Cursor
);
this
.
_ensureVisib
le
(
$new
Cursor
);
this
.
trigger
(
"cursorMoved"
,
extractSuggestion
(
$underCursor
));
},
},
_ensureVisible
:
function
ensureVisible
(
$el
)
{
_getSuggestions
:
function
()
{
var
elTop
,
elBottom
,
menuScrollTop
,
menuHeight
;
return
this
.
$menu
.
find
(
".tt-suggestions > .tt-suggestion"
)
;
elTop
=
$el
.
position
().
top
;
},
elBottom
=
elTop
+
$el
.
outerHeight
(
true
);
_ensureVisibility
:
function
(
$el
)
{
menuScrollTop
=
this
.
$menu
.
scrollTop
();
var
menuHeight
=
this
.
$menu
.
height
()
+
parseInt
(
this
.
$menu
.
css
(
"paddingTop"
),
10
)
+
parseInt
(
this
.
$menu
.
css
(
"paddingBottom"
),
10
),
menuScrollTop
=
this
.
$menu
.
scrollTop
(),
elTop
=
$el
.
position
().
top
,
elBottom
=
elTop
+
$el
.
outerHeight
(
true
);
menuHeight
=
this
.
$menu
.
height
()
+
parseInt
(
this
.
$menu
.
css
(
"paddingTop"
),
10
)
+
parseInt
(
this
.
$menu
.
css
(
"paddingBottom"
),
10
);
if
(
elTop
<
0
)
{
if
(
elTop
<
0
)
{
this
.
$menu
.
scrollTop
(
menuScrollTop
+
elTop
);
this
.
$menu
.
scrollTop
(
menuScrollTop
+
elTop
);
}
else
if
(
menuHeight
<
elBottom
)
{
}
else
if
(
menuHeight
<
elBottom
)
{
this
.
$menu
.
scrollTop
(
menuScrollTop
+
(
elBottom
-
menuHeight
));
this
.
$menu
.
scrollTop
(
menuScrollTop
+
(
elBottom
-
menuHeight
));
}
}
},
},
destroy
:
function
()
{
close
:
function
close
()
{
this
.
$menu
.
off
(
".tt"
);
this
.
$menu
=
null
;
},
isVisible
:
function
()
{
return
this
.
isOpen
&&
!
this
.
isEmpty
;
},
closeUnlessMouseIsOverDropdown
:
function
()
{
if
(
!
this
.
isMouseOverDropdown
)
{
this
.
close
();
}
},
close
:
function
()
{
if
(
this
.
isOpen
)
{
if
(
this
.
isOpen
)
{
this
.
isOpen
=
false
;
this
.
isOpen
=
this
.
isMouseOverDropdown
=
false
;
this
.
isMouseOverDropdown
=
false
;
this
.
_removeCursor
()
;
this
.
_hide
();
this
.
_hide
();
this
.
$menu
.
find
(
".tt-suggestions > .tt-suggestion"
).
removeClass
(
"tt-is-under-cursor"
);
this
.
trigger
(
"closed"
);
this
.
trigger
(
"closed"
);
}
}
},
},
open
:
function
()
{
open
:
function
open
()
{
if
(
!
this
.
isOpen
)
{
if
(
!
this
.
isOpen
)
{
this
.
isOpen
=
true
;
this
.
isOpen
=
true
;
!
this
.
isEmpty
&&
this
.
_show
();
!
this
.
isEmpty
&&
this
.
_show
();
this
.
trigger
(
"opened"
);
this
.
trigger
(
"opened"
);
}
}
},
},
setLanguageDirection
:
function
(
dir
)
{
setLanguageDirection
:
function
setLanguageDirection
(
dir
)
{
var
ltrCss
=
{
this
.
$menu
.
css
(
dir
===
"ltr"
?
css
.
ltr
:
css
.
rtl
);
left
:
"0"
,
right
:
"auto"
},
rtlCss
=
{
left
:
"auto"
,
right
:
" 0"
};
dir
===
"ltr"
?
this
.
$menu
.
css
(
ltrCss
)
:
this
.
$menu
.
css
(
rtlCss
);
},
},
moveCursorUp
:
function
()
{
moveCursorUp
:
function
moveCursorUp
()
{
this
.
_moveCursor
(
-
1
);
this
.
_moveCursor
(
-
1
);
},
},
moveCursorDown
:
function
()
{
moveCursorDown
:
function
moveCursorDown
()
{
this
.
_moveCursor
(
+
1
);
this
.
_moveCursor
(
+
1
);
},
},
getSuggestionUnderCursor
:
function
()
{
getDatumForSuggestion
:
function
getDatumForSuggestion
(
$el
)
{
var
$suggestion
=
this
.
_getSuggestions
().
filter
(
".tt-is-under-cursor"
).
first
();
var
datum
=
null
;
return
$suggestion
.
length
>
0
?
extractSuggestion
(
$suggestion
)
:
null
;
if
(
$el
.
length
)
{
},
datum
=
{
getFirstSuggestion
:
function
()
{
raw
:
Dataset
.
extractDatum
(
$el
),
var
$suggestion
=
this
.
_getSuggestions
().
first
();
value
:
Dataset
.
extractValue
(
$el
),
return
$suggestion
.
length
>
0
?
extractSuggestion
(
$suggestion
)
:
null
;
datasetName
:
Dataset
.
extractDatasetName
(
$el
)
},
};
renderSuggestions
:
function
(
dataset
,
suggestions
)
{
var
datasetClassName
=
"tt-dataset-"
+
dataset
.
name
,
wrapper
=
'<div class="tt-suggestion">%body</div>'
,
compiledHtml
,
$suggestionsList
,
$dataset
=
this
.
$menu
.
find
(
"."
+
datasetClassName
),
elBuilder
,
fragment
,
$el
;
if
(
$dataset
.
length
===
0
)
{
$suggestionsList
=
$
(
html
.
suggestionsList
).
css
(
css
.
suggestionsList
);
$dataset
=
$
(
"<div></div>"
).
addClass
(
datasetClassName
).
append
(
dataset
.
header
).
append
(
$suggestionsList
).
append
(
dataset
.
footer
).
appendTo
(
this
.
$menu
);
}
if
(
suggestions
.
length
>
0
)
{
this
.
isEmpty
=
false
;
this
.
isOpen
&&
this
.
_show
();
elBuilder
=
document
.
createElement
(
"div"
);
fragment
=
document
.
createDocumentFragment
();
utils
.
each
(
suggestions
,
function
(
i
,
suggestion
)
{
suggestion
.
dataset
=
dataset
.
name
;
compiledHtml
=
dataset
.
template
(
suggestion
.
datum
);
elBuilder
.
innerHTML
=
wrapper
.
replace
(
"%body"
,
compiledHtml
);
$el
=
$
(
elBuilder
.
firstChild
).
css
(
css
.
suggestion
).
data
(
"suggestion"
,
suggestion
);
$el
.
children
().
each
(
function
()
{
$
(
this
).
css
(
css
.
suggestionChild
);
});
fragment
.
appendChild
(
$el
[
0
]);
});
$dataset
.
show
().
find
(
".tt-suggestions"
).
html
(
fragment
);
}
else
{
this
.
clearSuggestions
(
dataset
.
name
);
}
}
this
.
trigger
(
"suggestionsRendered"
)
;
return
datum
;
},
},
clearSuggestions
:
function
(
datasetName
)
{
getDatumForCursor
:
function
getDatumForCursor
()
{
var
$datasets
=
datasetName
?
this
.
$menu
.
find
(
".tt-dataset-"
+
datasetName
)
:
this
.
$menu
.
find
(
'[class^="tt-dataset-"]'
),
$suggestions
=
$datasets
.
find
(
".tt-suggestions"
);
return
this
.
getDatumForSuggestion
(
this
.
_getCursor
().
first
());
$datasets
.
hide
();
},
$suggestions
.
empty
();
getDatumForTopSuggestion
:
function
getDatumForTopSuggestion
()
{
if
(
this
.
_getSuggestions
().
length
===
0
)
{
return
this
.
getDatumForSuggestion
(
this
.
_getSuggestions
().
first
());
this
.
isEmpty
=
true
;
},
this
.
_hide
();
update
:
function
update
(
query
)
{
}
_
.
each
(
this
.
datasets
,
updateDataset
);
}
function
updateDataset
(
dataset
)
{
});
dataset
.
update
(
query
);
return
DropdownView
;
function
extractSuggestion
(
$el
)
{
return
$el
.
data
(
"suggestion"
);
}
}
}();
var
TypeaheadView
=
function
()
{
var
html
=
{
wrapper
:
'<span class="twitter-typeahead"></span>'
,
hint
:
'<input class="tt-hint" type="text" autocomplete="off" spellcheck="off" disabled>'
,
dropdown
:
'<span class="tt-dropdown-menu"></span>'
},
css
=
{
wrapper
:
{
position
:
"relative"
,
display
:
"inline-block"
},
},
hint
:
{
empty
:
function
empty
()
{
position
:
"absolute"
,
_
.
each
(
this
.
datasets
,
clearDataset
);
top
:
"0"
,
function
clearDataset
(
dataset
)
{
left
:
"0"
,
dataset
.
clear
();
borderColor
:
"transparent"
,
}
boxShadow
:
"none"
},
},
query
:
{
isVisible
:
function
isVisible
()
{
position
:
"relative"
,
return
this
.
isOpen
&&
!
this
.
isEmpty
;
verticalAlign
:
"top"
,
backgroundColor
:
"transparent"
},
},
d
ropdown
:
{
d
estroy
:
function
destroy
()
{
position
:
"absolute"
,
this
.
$menu
.
off
(
".tt"
);
t
op
:
"100%"
,
t
his
.
$menu
=
null
;
left
:
"0"
,
_
.
each
(
this
.
datasets
,
destroyDataset
);
zIndex
:
"100"
,
function
destroyDataset
(
dataset
)
{
display
:
"none"
dataset
.
destroy
();
}
}
};
if
(
utils
.
isMsie
())
{
utils
.
mixin
(
css
.
query
,
{
backgroundImage
:
"url()"
});
}
}
if
(
utils
.
isMsie
()
&&
utils
.
isMsie
()
<=
7
)
{
utils
.
mixin
(
css
.
wrapper
,
{
display
:
"inline"
,
zoom
:
"1"
});
utils
.
mixin
(
css
.
query
,
{
marginTop
:
"-1px"
});
});
return
Dropdown
;
function
initializeDataset
(
oDataset
)
{
return
new
Dataset
(
oDataset
);
}
}
function
TypeaheadView
(
o
)
{
}();
var
$menu
,
$input
,
$hint
;
var
Typeahead
=
function
()
{
utils
.
bindAll
(
this
);
var
attrsKey
=
"ttAttrs"
;
this
.
$node
=
buildDomStructure
(
o
.
input
);
function
Typeahead
(
o
)
{
this
.
datasets
=
o
.
datasets
;
var
$menu
,
$input
,
$hint
,
datasets
;
this
.
dir
=
null
;
o
=
o
||
{};
this
.
eventBus
=
o
.
eventBus
;
if
(
!
o
.
input
)
{
$
.
error
(
"missing input"
);
}
this
.
autoselect
=
!!
o
.
autoselect
;
this
.
minLength
=
_
.
isNumber
(
o
.
minLength
)
?
o
.
minLength
:
1
;
this
.
$node
=
buildDomStructure
(
o
.
input
,
o
.
withHint
);
$menu
=
this
.
$node
.
find
(
".tt-dropdown-menu"
);
$menu
=
this
.
$node
.
find
(
".tt-dropdown-menu"
);
$input
=
this
.
$node
.
find
(
".tt-
query
"
);
$input
=
this
.
$node
.
find
(
".tt-
input
"
);
$hint
=
this
.
$node
.
find
(
".tt-hint"
);
$hint
=
this
.
$node
.
find
(
".tt-hint"
);
this
.
dropdownView
=
new
DropdownView
({
this
.
eventBus
=
o
.
eventBus
||
new
EventBus
({
menu
:
$menu
el
:
$input
}).
on
(
"suggestionSelected"
,
this
.
_handleSelection
).
on
(
"cursorMoved"
,
this
.
_clearHint
).
on
(
"cursorMoved"
,
this
.
_setInputValueToSuggestionUnderCursor
).
on
(
"cursorRemoved"
,
this
.
_setInputValueToQuery
).
on
(
"cursorRemoved"
,
this
.
_updateHint
).
on
(
"suggestionsRendered"
,
this
.
_updateHint
).
on
(
"opened"
,
this
.
_updateHint
).
on
(
"closed"
,
this
.
_clearHint
).
on
(
"opened closed"
,
this
.
_propagateEvent
);
});
this
.
inputView
=
new
InputView
({
this
.
dropdown
=
new
Dropdown
({
menu
:
$menu
,
datasets
:
o
.
datasets
}).
onSync
(
"suggestionClicked"
,
this
.
_onSuggestionClicked
,
this
).
onSync
(
"cursorMoved"
,
this
.
_onCursorMoved
,
this
).
onSync
(
"cursorRemoved"
,
this
.
_onCursorRemoved
,
this
).
onSync
(
"opened"
,
this
.
_onOpened
,
this
).
onSync
(
"closed"
,
this
.
_onClosed
,
this
).
onAsync
(
"datasetRendered"
,
this
.
_onDatasetRendered
,
this
);
this
.
input
=
new
Input
({
input
:
$input
,
input
:
$input
,
hint
:
$hint
hint
:
$hint
}).
on
(
"focused"
,
this
.
_openDropdown
).
on
(
"blured"
,
this
.
_closeDropdown
).
on
(
"blured"
,
this
.
_setInputValueToQuery
).
on
(
"enterKeyed tabKeyed"
,
this
.
_handleSelection
).
on
(
"queryChanged"
,
this
.
_clearHint
).
on
(
"queryChanged"
,
this
.
_clearSuggestions
).
on
(
"queryChanged"
,
this
.
_getSuggestions
).
on
(
"whitespaceChanged"
,
this
.
_updateHint
).
on
(
"queryChanged whitespaceChanged"
,
this
.
_openDropdown
).
on
(
"queryChanged whitespaceChanged"
,
this
.
_setLanguageDirection
).
on
(
"escKeyed"
,
this
.
_closeDropdown
).
on
(
"escKeyed"
,
this
.
_setInputValueToQuery
).
on
(
"tabKeyed upKeyed downKeyed"
,
this
.
_managePreventDefault
).
on
(
"upKeyed downKeyed"
,
this
.
_moveDropdownCursor
).
on
(
"upKeyed downKeyed"
,
this
.
_openDropdown
).
on
(
"tabKeyed leftKeyed rightKeyed"
,
this
.
_autocomplete
);
}).
onSync
(
"focused"
,
this
.
_onFocused
,
this
).
onSync
(
"blurred"
,
this
.
_onBlurred
,
this
).
onSync
(
"enterKeyed"
,
this
.
_onEnterKeyed
,
this
).
onSync
(
"tabKeyed"
,
this
.
_onTabKeyed
,
this
).
onSync
(
"escKeyed"
,
this
.
_onEscKeyed
,
this
).
onSync
(
"upKeyed"
,
this
.
_onUpKeyed
,
this
).
onSync
(
"downKeyed"
,
this
.
_onDownKeyed
,
this
).
onSync
(
"leftKeyed"
,
this
.
_onLeftKeyed
,
this
).
onSync
(
"rightKeyed"
,
this
.
_onRightKeyed
,
this
).
onSync
(
"queryChanged"
,
this
.
_onQueryChanged
,
this
).
onSync
(
"whitespaceChanged"
,
this
.
_onWhitespaceChanged
,
this
);
}
$menu
.
on
(
"mousedown.tt"
,
function
(
$e
)
{
utils
.
mixin
(
TypeaheadView
.
prototype
,
EventTarget
,
{
if
(
_
.
isMsie
()
&&
_
.
isMsie
()
<
9
)
{
_managePreventDefault
:
function
(
e
)
{
$input
[
0
].
onbeforedeactivate
=
function
()
{
var
$e
=
e
.
data
,
hint
,
inputValue
,
preventDefault
=
false
;
window
.
event
.
returnValue
=
false
;
switch
(
e
.
type
)
{
$input
[
0
].
onbeforedeactivate
=
null
;
case
"tabKeyed"
:
};
hint
=
this
.
inputView
.
getHintValue
();
inputValue
=
this
.
inputView
.
getInputValue
();
preventDefault
=
hint
&&
hint
!==
inputValue
;
break
;
case
"upKeyed"
:
case
"downKeyed"
:
preventDefault
=
!
$e
.
shiftKey
&&
!
$e
.
ctrlKey
&&
!
$e
.
metaKey
;
break
;
}
}
preventDefault
&&
$e
.
preventDefault
();
$e
.
preventDefault
();
},
});
_setLanguageDirection
:
function
()
{
var
dir
=
this
.
inputView
.
getLanguageDirection
();
if
(
dir
!==
this
.
dir
)
{
this
.
dir
=
dir
;
this
.
$node
.
css
(
"direction"
,
dir
);
this
.
dropdownView
.
setLanguageDirection
(
dir
);
}
}
},
_
.
mixin
(
Typeahead
.
prototype
,
{
_updateHint
:
function
()
{
_onSuggestionClicked
:
function
onSuggestionClicked
(
type
,
$el
)
{
var
suggestion
=
this
.
dropdownView
.
getFirstSuggestion
(),
hint
=
suggestion
?
suggestion
.
value
:
null
,
dropdownIsVisible
=
this
.
dropdownView
.
isVisible
(),
inputHasOverflow
=
this
.
inputView
.
isOverflow
(),
inputValue
,
query
,
escapedQuery
,
beginsWithQuery
,
match
;
var
datum
;
if
(
hint
&&
dropdownIsVisible
&&
!
inputHasOverflow
)
{
if
(
datum
=
this
.
dropdown
.
getDatumForSuggestion
(
$el
))
{
inputValue
=
this
.
inputView
.
getInputValue
();
this
.
_select
(
datum
);
query
=
inputValue
.
replace
(
/
\s{2,}
/g
,
" "
).
replace
(
/^
\s
+/g
,
""
);
}
escapedQuery
=
utils
.
escapeRegExChars
(
query
);
},
beginsWithQuery
=
new
RegExp
(
"^(?:"
+
escapedQuery
+
")(.*$)"
,
"i"
);
_onCursorMoved
:
function
onCursorMoved
()
{
match
=
beginsWithQuery
.
exec
(
hint
);
var
datum
=
this
.
dropdown
.
getDatumForCursor
();
this
.
inputView
.
setHintValue
(
inputValue
+
(
match
?
match
[
1
]
:
""
));
this
.
input
.
clearHint
();
this
.
input
.
setInputValue
(
datum
.
value
,
true
);
this
.
eventBus
.
trigger
(
"cursorchanged"
,
datum
.
raw
,
datum
.
datasetName
);
},
_onCursorRemoved
:
function
onCursorRemoved
()
{
this
.
input
.
resetInputValue
();
this
.
_updateHint
();
},
_onDatasetRendered
:
function
onDatasetRendered
()
{
this
.
_updateHint
();
},
_onOpened
:
function
onOpened
()
{
this
.
_updateHint
();
this
.
eventBus
.
trigger
(
"opened"
);
},
_onClosed
:
function
onClosed
()
{
this
.
input
.
clearHint
();
this
.
eventBus
.
trigger
(
"closed"
);
},
_onFocused
:
function
onFocused
()
{
this
.
dropdown
.
open
();
},
_onBlurred
:
function
onBlurred
()
{
!
this
.
dropdown
.
isMouseOverDropdown
&&
this
.
dropdown
.
close
();
},
_onEnterKeyed
:
function
onEnterKeyed
(
type
,
$e
)
{
var
cursorDatum
,
topSuggestionDatum
;
cursorDatum
=
this
.
dropdown
.
getDatumForCursor
();
topSuggestionDatum
=
this
.
dropdown
.
getDatumForTopSuggestion
();
if
(
cursorDatum
)
{
this
.
_select
(
cursorDatum
);
$e
.
preventDefault
();
}
else
if
(
this
.
autoselect
&&
topSuggestionDatum
)
{
this
.
_select
(
topSuggestionDatum
);
$e
.
preventDefault
();
}
},
_onTabKeyed
:
function
onTabKeyed
(
type
,
$e
)
{
var
datum
;
if
(
datum
=
this
.
dropdown
.
getDatumForCursor
())
{
this
.
_select
(
datum
);
$e
.
preventDefault
();
}
else
{
this
.
_autocomplete
();
}
}
},
},
_clearHint
:
function
()
{
_onEscKeyed
:
function
onEscKeyed
()
{
this
.
inputView
.
setHintValue
(
""
);
this
.
dropdown
.
close
();
},
this
.
input
.
resetInputValue
();
_clearSuggestions
:
function
()
{
this
.
dropdownView
.
clearSuggestions
();
},
_setInputValueToQuery
:
function
()
{
this
.
inputView
.
setInputValue
(
this
.
inputView
.
getQuery
());
},
_setInputValueToSuggestionUnderCursor
:
function
(
e
)
{
var
suggestion
=
e
.
data
;
this
.
inputView
.
setInputValue
(
suggestion
.
value
,
true
);
},
_openDropdown
:
function
()
{
this
.
dropdownView
.
open
();
},
_closeDropdown
:
function
(
e
)
{
this
.
dropdownView
[
e
.
type
===
"blured"
?
"closeUnlessMouseIsOverDropdown"
:
"close"
]();
},
},
_
moveDropdownCursor
:
function
(
e
)
{
_
onUpKeyed
:
function
onUpKeyed
(
)
{
var
$e
=
e
.
data
;
var
query
=
this
.
input
.
getQuery
()
;
if
(
!
$e
.
shiftKey
&&
!
$e
.
ctrlKey
&&
!
$e
.
metaKey
)
{
if
(
!
this
.
dropdown
.
isOpen
&&
query
.
length
>=
this
.
minLength
)
{
this
.
dropdown
View
[
e
.
type
===
"upKeyed"
?
"moveCursorUp"
:
"moveCursorDown"
](
);
this
.
dropdown
.
update
(
query
);
}
}
this
.
dropdown
.
open
();
this
.
dropdown
.
moveCursorUp
();
},
},
_handleSelection
:
function
(
e
)
{
_onDownKeyed
:
function
onDownKeyed
()
{
var
byClick
=
e
.
type
===
"suggestionSelected"
,
suggestion
=
byClick
?
e
.
data
:
this
.
dropdownView
.
getSuggestionUnderCursor
();
var
query
=
this
.
input
.
getQuery
();
if
(
suggestion
)
{
if
(
!
this
.
dropdown
.
isOpen
&&
query
.
length
>=
this
.
minLength
)
{
this
.
inputView
.
setInputValue
(
suggestion
.
value
);
this
.
dropdown
.
update
(
query
);
byClick
?
this
.
inputView
.
focus
()
:
e
.
data
.
preventDefault
();
byClick
&&
utils
.
isMsie
()
?
utils
.
defer
(
this
.
dropdownView
.
close
)
:
this
.
dropdownView
.
close
();
this
.
eventBus
.
trigger
(
"selected"
,
suggestion
.
datum
,
suggestion
.
dataset
);
}
}
this
.
dropdown
.
open
();
this
.
dropdown
.
moveCursorDown
();
},
},
_getSuggestions
:
function
()
{
_onLeftKeyed
:
function
onLeftKeyed
()
{
var
that
=
this
,
query
=
this
.
inputView
.
getQuery
();
this
.
dir
===
"rtl"
&&
this
.
_autocomplete
();
if
(
utils
.
isBlankString
(
query
))
{
return
;
}
utils
.
each
(
this
.
datasets
,
function
(
i
,
dataset
)
{
dataset
.
getSuggestions
(
query
,
function
(
suggestions
)
{
if
(
query
===
that
.
inputView
.
getQuery
())
{
that
.
dropdownView
.
renderSuggestions
(
dataset
,
suggestions
);
}
});
});
},
},
_autocomplete
:
function
(
e
)
{
_onRightKeyed
:
function
onRightKeyed
()
{
var
isCursorAtEnd
,
ignoreEvent
,
query
,
hint
,
suggestion
;
this
.
dir
===
"ltr"
&&
this
.
_autocomplete
();
if
(
e
.
type
===
"rightKeyed"
||
e
.
type
===
"leftKeyed"
)
{
},
isCursorAtEnd
=
this
.
inputView
.
isCursorAtEnd
();
_onQueryChanged
:
function
onQueryChanged
(
e
,
query
)
{
ignoreEvent
=
this
.
inputView
.
getLanguageDirection
()
===
"ltr"
?
e
.
type
===
"leftKeyed"
:
e
.
type
===
"rightKeyed"
;
this
.
input
.
clearHint
();
if
(
!
isCursorAtEnd
||
ignoreEvent
)
{
this
.
dropdown
.
empty
();
return
;
query
.
length
>=
this
.
minLength
&&
this
.
dropdown
.
update
(
query
);
}
this
.
dropdown
.
open
();
}
this
.
_setLanguageDirection
();
query
=
this
.
inputView
.
getQuery
();
hint
=
this
.
inputView
.
getHintValue
();
if
(
hint
!==
""
&&
query
!==
hint
)
{
suggestion
=
this
.
dropdownView
.
getFirstSuggestion
();
this
.
inputView
.
setInputValue
(
suggestion
.
value
);
this
.
eventBus
.
trigger
(
"autocompleted"
,
suggestion
.
datum
,
suggestion
.
dataset
);
}
},
},
_propagateEvent
:
function
(
e
)
{
_onWhitespaceChanged
:
function
onWhitespaceChanged
()
{
this
.
eventBus
.
trigger
(
e
.
type
);
this
.
_updateHint
();
this
.
dropdown
.
open
();
},
},
destroy
:
function
()
{
_setLanguageDirection
:
function
setLanguageDirection
()
{
this
.
inputView
.
destroy
();
var
dir
;
this
.
dropdownView
.
destroy
();
if
(
this
.
dir
!==
(
dir
=
this
.
input
.
getLanguageDirection
()))
{
this
.
dir
=
dir
;
this
.
$node
.
css
(
"direction"
,
dir
);
this
.
dropdown
.
setLanguageDirection
(
dir
);
}
},
_updateHint
:
function
updateHint
()
{
var
datum
,
inputValue
,
query
,
escapedQuery
,
frontMatchRegEx
,
match
;
datum
=
this
.
dropdown
.
getDatumForTopSuggestion
();
if
(
datum
&&
this
.
dropdown
.
isVisible
()
&&
!
this
.
input
.
hasOverflow
())
{
inputValue
=
this
.
input
.
getInputValue
();
query
=
Input
.
normalizeQuery
(
inputValue
);
escapedQuery
=
_
.
escapeRegExChars
(
query
);
frontMatchRegEx
=
new
RegExp
(
"^(?:"
+
escapedQuery
+
")(.*$)"
,
"i"
);
match
=
frontMatchRegEx
.
exec
(
datum
.
value
);
this
.
input
.
setHintValue
(
inputValue
+
(
match
?
match
[
1
]
:
""
));
}
},
_autocomplete
:
function
autocomplete
()
{
var
hint
,
query
,
datum
;
hint
=
this
.
input
.
getHintValue
();
query
=
this
.
input
.
getQuery
();
if
(
hint
&&
query
!==
hint
&&
this
.
input
.
isCursorAtEnd
())
{
datum
=
this
.
dropdown
.
getDatumForTopSuggestion
();
datum
&&
this
.
input
.
setInputValue
(
datum
.
value
);
this
.
eventBus
.
trigger
(
"autocompleted"
,
datum
.
raw
,
datum
.
datasetName
);
}
},
_select
:
function
select
(
datum
)
{
this
.
input
.
clearHint
();
this
.
input
.
setQuery
(
datum
.
value
);
this
.
input
.
setInputValue
(
datum
.
value
,
true
);
this
.
dropdown
.
empty
();
this
.
_setLanguageDirection
();
_
.
defer
(
_
.
bind
(
this
.
dropdown
.
close
,
this
.
dropdown
));
this
.
eventBus
.
trigger
(
"selected"
,
datum
.
raw
,
datum
.
datasetName
);
},
open
:
function
open
()
{
this
.
dropdown
.
open
();
},
close
:
function
close
()
{
this
.
dropdown
.
close
();
},
getQuery
:
function
getQuery
()
{
return
this
.
input
.
getQuery
();
},
setQuery
:
function
setQuery
(
val
)
{
this
.
input
.
setInputValue
(
val
);
},
destroy
:
function
destroy
()
{
this
.
input
.
destroy
();
this
.
dropdown
.
destroy
();
destroyDomStructure
(
this
.
$node
);
destroyDomStructure
(
this
.
$node
);
this
.
$node
=
null
;
this
.
$node
=
null
;
},
setQuery
:
function
(
query
)
{
this
.
inputView
.
setQuery
(
query
);
this
.
inputView
.
setInputValue
(
query
);
this
.
_clearHint
();
this
.
_clearSuggestions
();
this
.
_getSuggestions
();
}
}
});
});
return
TypeaheadView
;
return
Typeahead
;
function
buildDomStructure
(
input
)
{
function
buildDomStructure
(
input
,
withHint
)
{
var
$wrapper
=
$
(
html
.
wrapper
),
$dropdown
=
$
(
html
.
dropdown
),
$input
=
$
(
input
),
$hint
=
$
(
html
.
hint
);
var
$input
,
$wrapper
,
$dropdown
,
$hint
;
$wrapper
=
$wrapper
.
css
(
css
.
wrapper
);
$input
=
$
(
input
);
$dropdown
=
$dropdown
.
css
(
css
.
dropdown
);
$wrapper
=
$
(
html
.
wrapper
).
css
(
css
.
wrapper
);
$hint
.
css
(
css
.
hint
).
css
({
$dropdown
=
$
(
html
.
dropdown
).
css
(
css
.
dropdown
);
backgroundAttachment
:
$input
.
css
(
"background-attachment"
),
$hint
=
$input
.
clone
().
css
(
css
.
hint
).
css
(
getBackgroundStyles
(
$input
));
backgroundClip
:
$input
.
css
(
"background-clip"
),
$hint
.
removeData
().
addClass
(
"tt-hint"
).
removeAttr
(
"id name placeholder"
).
prop
(
"disabled"
,
true
).
attr
({
backgroundColor
:
$input
.
css
(
"background-color"
),
autocomplete
:
"off"
,
backgroundImage
:
$input
.
css
(
"background-image"
),
spellcheck
:
"false"
backgroundOrigin
:
$input
.
css
(
"background-origin"
),
backgroundPosition
:
$input
.
css
(
"background-position"
),
backgroundRepeat
:
$input
.
css
(
"background-repeat"
),
backgroundSize
:
$input
.
css
(
"background-size"
)
});
});
$input
.
data
(
"ttAttrs"
,
{
$input
.
data
(
attrsKey
,
{
dir
:
$input
.
attr
(
"dir"
),
dir
:
$input
.
attr
(
"dir"
),
autocomplete
:
$input
.
attr
(
"autocomplete"
),
autocomplete
:
$input
.
attr
(
"autocomplete"
),
spellcheck
:
$input
.
attr
(
"spellcheck"
),
spellcheck
:
$input
.
attr
(
"spellcheck"
),
style
:
$input
.
attr
(
"style"
)
style
:
$input
.
attr
(
"style"
)
});
});
$input
.
addClass
(
"tt-
query
"
).
attr
({
$input
.
addClass
(
"tt-
input
"
).
attr
({
autocomplete
:
"off"
,
autocomplete
:
"off"
,
spellcheck
:
false
spellcheck
:
false
}).
css
(
css
.
query
);
}).
css
(
withHint
?
css
.
input
:
css
.
inputWithNoHint
);
try
{
try
{
!
$input
.
attr
(
"dir"
)
&&
$input
.
attr
(
"dir"
,
"auto"
);
!
$input
.
attr
(
"dir"
)
&&
$input
.
attr
(
"dir"
,
"auto"
);
}
catch
(
e
)
{}
}
catch
(
e
)
{}
return
$input
.
wrap
(
$wrapper
).
parent
().
prepend
(
$hint
).
append
(
$dropdown
);
return
$input
.
wrap
(
$wrapper
).
parent
().
prepend
(
withHint
?
$hint
:
null
).
append
(
$dropdown
);
}
function
getBackgroundStyles
(
$el
)
{
return
{
backgroundAttachment
:
$el
.
css
(
"background-attachment"
),
backgroundClip
:
$el
.
css
(
"background-clip"
),
backgroundColor
:
$el
.
css
(
"background-color"
),
backgroundImage
:
$el
.
css
(
"background-image"
),
backgroundOrigin
:
$el
.
css
(
"background-origin"
),
backgroundPosition
:
$el
.
css
(
"background-position"
),
backgroundRepeat
:
$el
.
css
(
"background-repeat"
),
backgroundSize
:
$el
.
css
(
"background-size"
)
};
}
}
function
destroyDomStructure
(
$node
)
{
function
destroyDomStructure
(
$node
)
{
var
$input
=
$node
.
find
(
".tt-
query
"
);
var
$input
=
$node
.
find
(
".tt-
input
"
);
utils
.
each
(
$input
.
data
(
"ttAttrs"
),
function
(
key
,
val
)
{
_
.
each
(
$input
.
data
(
attrsKey
),
function
(
val
,
key
)
{
utils
.
isUndefined
(
val
)
?
$input
.
removeAttr
(
key
)
:
$input
.
attr
(
key
,
val
);
_
.
isUndefined
(
val
)
?
$input
.
removeAttr
(
key
)
:
$input
.
attr
(
key
,
val
);
});
});
$input
.
detach
().
removeData
(
"ttAttrs"
).
removeClass
(
"tt-query
"
).
insertAfter
(
$node
);
$input
.
detach
().
removeData
(
attrsKey
).
removeClass
(
"tt-input
"
).
insertAfter
(
$node
);
$node
.
remove
();
$node
.
remove
();
}
}
}();
}();
(
function
()
{
(
function
()
{
var
cache
=
{},
viewKey
=
"ttView"
,
methods
;
var
typeaheadKey
,
methods
;
typeaheadKey
=
"ttTypeahead"
;
methods
=
{
methods
=
{
initialize
:
function
(
datasetDefs
)
{
initialize
:
function
initialize
(
o
)
{
var
datasets
;
var
datasets
=
[].
slice
.
call
(
arguments
,
1
);
datasetDefs
=
utils
.
isArray
(
datasetDefs
)
?
datasetDefs
:
[
datasetDefs
];
o
=
o
||
{};
if
(
datasetDefs
.
length
===
0
)
{
return
this
.
each
(
attach
);
$
.
error
(
"no datasets provided"
);
function
attach
()
{
}
var
$input
=
$
(
this
),
eventBus
,
typeahead
;
datasets
=
utils
.
map
(
datasetDefs
,
function
(
o
)
{
_
.
each
(
datasets
,
function
(
d
)
{
var
dataset
=
cache
[
o
.
name
]
?
cache
[
o
.
name
]
:
new
Dataset
(
o
);
d
.
highlight
=
!!
o
.
highlight
;
if
(
o
.
name
)
{
cache
[
o
.
name
]
=
dataset
;
}
return
dataset
;
});
return
this
.
each
(
initialize
);
function
initialize
()
{
var
$input
=
$
(
this
),
deferreds
,
eventBus
=
new
EventBus
({
el
:
$input
});
deferreds
=
utils
.
map
(
datasets
,
function
(
dataset
)
{
return
dataset
.
initialize
();
});
});
$input
.
data
(
viewKey
,
new
TypeaheadView
({
typeahead
=
new
Typeahead
({
input
:
$input
,
input
:
$input
,
eventBus
:
eventBus
=
new
EventBus
({
eventBus
:
eventBus
=
new
EventBus
({
el
:
$input
el
:
$input
}),
}),
withHint
:
_
.
isUndefined
(
o
.
hint
)
?
true
:
!!
o
.
hint
,
minLength
:
o
.
minLength
,
autoselect
:
o
.
autoselect
,
datasets
:
datasets
datasets
:
datasets
}));
$
.
when
.
apply
(
$
,
deferreds
).
always
(
function
()
{
utils
.
defer
(
function
()
{
eventBus
.
trigger
(
"initialized"
);
});
});
$input
.
data
(
typeaheadKey
,
typeahead
);
function
trigger
(
eventName
)
{
return
function
()
{
_
.
defer
(
function
()
{
eventBus
.
trigger
(
eventName
);
});
});
};
}
}
},
open
:
function
open
()
{
return
this
.
each
(
openTypeahead
);
function
openTypeahead
()
{
var
$input
=
$
(
this
),
typeahead
;
if
(
typeahead
=
$input
.
data
(
typeaheadKey
))
{
typeahead
.
open
();
}
}
}
},
},
destroy
:
function
()
{
close
:
function
close
()
{
return
this
.
each
(
destroy
);
return
this
.
each
(
closeTypeahead
);
function
destroy
()
{
function
closeTypeahead
()
{
var
$this
=
$
(
this
),
view
=
$this
.
data
(
viewKey
);
var
$input
=
$
(
this
),
typeahead
;
if
(
view
)
{
if
(
typeahead
=
$input
.
data
(
typeaheadKey
))
{
view
.
destroy
();
typeahead
.
close
();
$this
.
removeData
(
viewKey
);
}
}
}
}
},
},
setQuery
:
function
(
query
)
{
val
:
function
val
(
newVal
)
{
return
this
.
each
(
setQuery
);
return
_
.
isString
(
newVal
)
?
this
.
each
(
setQuery
)
:
this
.
map
(
getQuery
).
get
(
);
function
setQuery
()
{
function
setQuery
()
{
var
view
=
$
(
this
).
data
(
viewKey
);
var
$input
=
$
(
this
),
typeahead
;
view
&&
view
.
setQuery
(
query
);
if
(
typeahead
=
$input
.
data
(
typeaheadKey
))
{
typeahead
.
setQuery
(
newVal
);
}
}
function
getQuery
()
{
var
$input
=
$
(
this
),
typeahead
,
query
;
if
(
typeahead
=
$input
.
data
(
typeaheadKey
))
{
query
=
typeahead
.
getQuery
();
}
return
query
;
}
},
destroy
:
function
destroy
()
{
return
this
.
each
(
unattach
);
function
unattach
()
{
var
$input
=
$
(
this
),
typeahead
;
if
(
typeahead
=
$input
.
data
(
typeaheadKey
))
{
typeahead
.
destroy
();
$input
.
removeData
(
typeaheadKey
);
}
}
}
}
}
};
};
...
...
extensions/gii/components/ActiveField.php
View file @
69f0e5be
...
@@ -63,7 +63,10 @@ class ActiveField extends \yii\widgets\ActiveField
...
@@ -63,7 +63,10 @@ class ActiveField extends \yii\widgets\ActiveField
{
{
static
$counter
=
0
;
static
$counter
=
0
;
$this
->
inputOptions
[
'class'
]
.=
' typeahead-'
.
(
++
$counter
);
$this
->
inputOptions
[
'class'
]
.=
' typeahead-'
.
(
++
$counter
);
$this
->
form
->
getView
()
->
registerJs
(
"jQuery('.typeahead-
{
$counter
}
').typeahead({local: "
.
Json
::
encode
(
$data
)
.
"});"
);
foreach
(
$data
as
&
$item
)
{
$item
=
array
(
'word'
=>
$item
);
}
$this
->
form
->
getView
()
->
registerJs
(
"yii.gii.autocomplete(
$counter
, "
.
Json
::
encode
(
$data
)
.
");"
);
return
$this
;
return
$this
;
}
}
}
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment