Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
c3254358
Commit
c3254358
authored
Dec 10, 2015
by
Valery Sizov
Browse files
Options
Browse Files
Download
Plain Diff
Merge remote-tracking branch 'ee_com/master' into ce_upstream
parents
ada1ea2a
4b8154af
Changes
12
Show whitespace changes
Inline
Side-by-side
Showing
12 changed files
with
1278 additions
and
12 deletions
+1278
-12
CHANGELOG
CHANGELOG
+7
-11
CHANGELOG-EE
CHANGELOG-EE
+16
-0
app/assets/javascripts/application.js.coffee
app/assets/javascripts/application.js.coffee
+1
-0
app/assets/stylesheets/framework/tables.scss
app/assets/stylesheets/framework/tables.scss
+5
-1
app/controllers/groups/stats_controller.rb
app/controllers/groups/stats_controller.rb
+27
-0
app/models/event.rb
app/models/event.rb
+6
-0
app/views/groups/stats/show.html.haml
app/views/groups/stats/show.html.haml
+141
-0
app/views/layouts/nav/_group.html.haml
app/views/layouts/nav/_group.html.haml
+5
-0
config/routes.rb
config/routes.rb
+1
-0
features/group/statistics.feature
features/group/statistics.feature
+9
-0
features/steps/group/statistics.rb
features/steps/group/statistics.rb
+14
-0
vendor/assets/javascripts/jquery.tablesorter.js
vendor/assets/javascripts/jquery.tablesorter.js
+1046
-0
No files found.
CHANGELOG
View file @
c3254358
Please view this file on the master branch, on stable branches it's out of date.
Please view this file on the master branch, on stable branches it's out of date.
v 8.3.0 (unreleased)
v 8.3.0 (unreleased)
- Merge when build succeeds (Zeger-Jan van de Weg)
- Bump gollum-lib to 4.1.0 (Stan Hu)
- Bump gollum-lib to 4.1.0 (Stan Hu)
- Fix broken group avatar upload under "New group" (Stan Hu)
- Fix broken group avatar upload under "New group" (Stan Hu)
- Update project repositorize size and commit count during import:repos task (Stan Hu)
- Update project repositorize size and commit count during import:repos task (Stan Hu)
...
@@ -25,6 +26,7 @@ v 8.3.0 (unreleased)
...
@@ -25,6 +26,7 @@ v 8.3.0 (unreleased)
- Add languages page to graphs
- Add languages page to graphs
- Block LDAP user when they are no longer found in the LDAP server
- Block LDAP user when they are no longer found in the LDAP server
- Improve wording on project visibility levels (Zeger-Jan van de Weg)
- Improve wording on project visibility levels (Zeger-Jan van de Weg)
- Automatically select default clone protocol based on user preferences (Eirik Lygre)
v 8.2.3
v 8.2.3
- Fix application settings cache not expiring after changes (Stan Hu)
- Fix application settings cache not expiring after changes (Stan Hu)
...
@@ -32,6 +34,10 @@ v 8.2.3
...
@@ -32,6 +34,10 @@ v 8.2.3
v 8.2.3
v 8.2.3
- Webhook payload has an added, modified and removed properties for each commit
- Webhook payload has an added, modified and removed properties for each commit
- Update documentation for "Guest" permissions
- Properly convert Emoji-only comments into Award Emojis
- Webhook payload has an added, modified and removed properties for each commit
- Fix 500 error when creating a merge request that removes a submodule
v 8.2.2
v 8.2.2
- Fix 404 in redirection after removing a project (Stan Hu)
- Fix 404 in redirection after removing a project (Stan Hu)
...
@@ -39,6 +45,7 @@ v 8.2.2
...
@@ -39,6 +45,7 @@ v 8.2.2
- Fix Error 500 when viewing user's personal projects from admin page (Stan Hu)
- Fix Error 500 when viewing user's personal projects from admin page (Stan Hu)
- Fix: Raw private snippets access workflow
- Fix: Raw private snippets access workflow
- Prevent "413 Request entity too large" errors when pushing large files with LFS
- Prevent "413 Request entity too large" errors when pushing large files with LFS
- Fix: As an admin, cannot add oneself as a member to a group/project
- Fix invalid links within projects dashboard header
- Fix invalid links within projects dashboard header
- Make current user the first user in assignee dropdown in issues detail page (Stan Hu)
- Make current user the first user in assignee dropdown in issues detail page (Stan Hu)
- Fix: duplicate email notifications on issue comments
- Fix: duplicate email notifications on issue comments
...
@@ -47,16 +54,7 @@ v 8.2.1
...
@@ -47,16 +54,7 @@ v 8.2.1
- Forcefully update builds that didn't want to update with state machine
- Forcefully update builds that didn't want to update with state machine
- Fix: saving GitLabCiService as Admin Template
- Fix: saving GitLabCiService as Admin Template
v 8.2.0
v 8.0.1
v 8.1.0 (unreleased)
v 8.2.0 (unreleased)
v 8.3.0 (unreleased)
v 8.2.0
v 8.2.0
- Improved performance of finding projects and groups in various places
- Improved performance of rendering user profile pages and Atom feeds
- Fix grouping of contributors by email in graph.
- Improved performance of finding projects and groups in various places
- Improved performance of finding projects and groups in various places
- Improved performance of rendering user profile pages and Atom feeds
- Improved performance of rendering user profile pages and Atom feeds
- Expose build artifacts path as config option
- Expose build artifacts path as config option
...
@@ -100,7 +98,6 @@ v 8.2.0
...
@@ -100,7 +98,6 @@ v 8.2.0
- Add email notification to former assignee upon unassignment (Adam Lieskovský)
- Add email notification to former assignee upon unassignment (Adam Lieskovský)
- New design for project graphs page
- New design for project graphs page
- Remove deprecated dumped yaml file generated from previous job definitions
- Remove deprecated dumped yaml file generated from previous job definitions
- Fix incoming email config defaults
- Show specific runners from projects where user is master or owner
- Show specific runners from projects where user is master or owner
- MR target branch is now visible on a list view when it is different from project's default one
- MR target branch is now visible on a list view when it is different from project's default one
- Improve Continuous Integration graphs page
- Improve Continuous Integration graphs page
...
@@ -258,7 +255,6 @@ v 8.0.2
...
@@ -258,7 +255,6 @@ v 8.0.2
- Allow AWS S3 Server-Side Encryption with Amazon S3-Managed Keys for backups (Paul Beattie)
- Allow AWS S3 Server-Side Encryption with Amazon S3-Managed Keys for backups (Paul Beattie)
v 8.0.1
v 8.0.1
- Remove git refs used internally by GitLab from network graph (Stan Hu)
- Improve CI migration procedure and documentation
- Improve CI migration procedure and documentation
v 8.0.0
v 8.0.0
...
...
CHANGELOG-EE
View file @
c3254358
v 8.3.0 (unreleased)
v 8.3.0 (unreleased)
- License information can now be retrieved via the API
- License information can now be retrieved via the API
- Fix bug with negative approvals required
- Fix bug with negative approvals required
- Add group contribution statistics page
v 8.2.3
- No EE-specific changes
v 8.2.2
- Fix 404 in redirection after removing a project (Stan Hu)
- Ensure cached application settings are refreshed at startup (Stan Hu)
- Fix Error 500 when viewing user's personal projects from admin page (Stan Hu)
- Fix: Raw private snippets access workflow
- Prevent "413 Request entity too large" errors when pushing large files with LFS
- Ensure GitLab fires custom update hooks after commit via UI
v 8.2.1
- Forcefully update builds that didn't want to update with state machine
- Fix: saving GitLabCiService as Admin Template
v 8.2.0
v 8.2.0
- Invalidate stored jira password if the endpoint URL is changed
- Invalidate stored jira password if the endpoint URL is changed
...
...
app/assets/javascripts/application.js.coffee
View file @
c3254358
...
@@ -16,6 +16,7 @@
...
@@ -16,6 +16,7 @@
#= require jquery.scrollTo
#= require jquery.scrollTo
#= require jquery.blockUI
#= require jquery.blockUI
#= require jquery.turbolinks
#= require jquery.turbolinks
#= require jquery.tablesorter
#= require turbolinks
#= require turbolinks
#= require autosave
#= require autosave
#= require bootstrap
#= require bootstrap
...
...
app/assets/stylesheets/framework/tables.scss
View file @
c3254358
...
@@ -35,6 +35,10 @@ table {
...
@@ -35,6 +35,10 @@ table {
font-weight
:
normal
;
font-weight
:
normal
;
font-size
:
15px
;
font-size
:
15px
;
border-bottom
:
1px
solid
$border-color
!
important
;
border-bottom
:
1px
solid
$border-color
!
important
;
&
.sortable
{
cursor
:
pointer
;
}
}
}
td
{
td
{
...
...
app/controllers/groups/stats_controller.rb
0 → 100644
View file @
c3254358
class
Groups::StatsController
<
Groups
::
ApplicationController
before_action
:group
layout
'group'
def
show
@users
=
@group
.
users
@start_date
=
params
[
:start_date
]
||
Date
.
today
-
1
.
week
@events
=
Event
.
contributions
.
where
(
"created_at > ?"
,
@start_date
).
where
(
project_id:
@group
.
projects
)
@stats
=
{}
@stats
[
:merge_requests
]
=
@users
.
map
do
|
user
|
@events
.
merge_requests
.
created
.
where
(
author_id:
user
).
count
end
@stats
[
:issues
]
=
@users
.
map
do
|
user
|
@events
.
issues
.
closed
.
where
(
author_id:
user
).
count
end
@stats
[
:push
]
=
@users
.
map
do
|
user
|
@events
.
code_push
.
where
(
author_id:
user
).
count
end
end
end
app/models/event.rb
View file @
c3254358
...
@@ -50,6 +50,12 @@ class Event < ActiveRecord::Base
...
@@ -50,6 +50,12 @@ class Event < ActiveRecord::Base
scope
:in_projects
,
->
(
project_ids
)
{
where
(
project_id:
project_ids
).
recent
}
scope
:in_projects
,
->
(
project_ids
)
{
where
(
project_id:
project_ids
).
recent
}
scope
:with_associations
,
->
{
includes
(
project: :namespace
)
}
scope
:with_associations
,
->
{
includes
(
project: :namespace
)
}
scope
:for_milestone_id
,
->
(
milestone_id
)
{
where
(
target_type:
"Milestone"
,
target_id:
milestone_id
)
}
scope
:for_milestone_id
,
->
(
milestone_id
)
{
where
(
target_type:
"Milestone"
,
target_id:
milestone_id
)
}
scope
:issues
,
->
{
where
(
target_type:
'Issue'
)
}
scope
:merge_requests
,
->
{
where
(
target_type:
'MergeRequest'
)
}
scope
:created
,
->
{
where
(
action:
CREATED
)
}
scope
:closed
,
->
{
where
(
action:
CLOSED
)
}
scope
:merged
,
->
{
where
(
action:
MERGED
)
}
class
<<
self
class
<<
self
def
reset_event_cache_for
(
target
)
def
reset_event_cache_for
(
target
)
...
...
app/views/groups/stats/show.html.haml
0 → 100644
View file @
c3254358
-
page_title
"Statistics"
-
header_title
group_title
(
@group
,
"Statistics"
,
group_stats_path
(
@group
))
.gray-content-block
.pull-right
.dropdown.inline
%button
.dropdown-toggle.btn
{
type:
'button'
,
'data-toggle'
=>
'dropdown'
}
=
icon
(
'calendar-o'
)
%b
.caret
%ul
.dropdown-menu
%li
=
link_to
group_stats_path
(
@group
,
start_date:
Date
.
today
-
1
.
week
)
do
Last week
%li
=
link_to
group_stats_path
(
@group
,
start_date:
Date
.
today
-
1
.
month
)
do
Last month
%li
=
link_to
group_stats_path
(
@group
,
start_date:
Date
.
today
-
3
.
months
)
do
Last 3 months
.oneline
Contribution statistics for issues, merge requests and push events since
#{
@start_date
}
%h3
Push
.row
.col-md-4
%ul
%li
=
@events
.
code_push
.
count
times
%li
more than
=
@events
.
code_push
.
map
(
&
:commits_count
).
sum
commits
%li
by
=
pluralize
@events
.
code_push
.
pluck
(
:author_id
).
uniq
.
count
,
'person'
.col-md-8
%div
%p
.light
Push events per group member
%canvas
#push
{
height:
250
}
%h3
Merge Requests
.row
.col-md-4
%ul
%li
=
@events
.
merge_requests
.
created
.
count
created
%li
=
@events
.
merge_requests
.
merged
.
count
accepted
.col-md-8
%div
%p
.light
Merge requests created per group member
%canvas
#merge_requests
{
height:
250
}
%h3
Issues
.row
.col-md-4
%ul
%li
=
@events
.
issues
.
created
.
count
created
%li
=
@events
.
issues
.
closed
.
pluck
(
:target_id
).
uniq
.
count
closed
.col-md-8
%div
%p
.light
Issues closed per group member
%canvas
#issues
{
height:
250
}
.gray-content-block
.oneline
Contributions per group member
.table-holder
%table
.table.sortable-table
#event-stats
%thead
%tr
%th
.sortable
Name
=
icon
(
'sort'
)
%th
.sortable
Pushed
=
icon
(
'sort'
)
%th
.sortable
Opened issues
=
icon
(
'sort'
)
%th
.sortable
Closed issues
=
icon
(
'sort'
)
%th
.sortable
Opened MR
=
icon
(
'sort'
)
%th
.sortable
Accepted MR
=
icon
(
'sort'
)
%th
.sortable
Total Contributions
=
icon
(
'sort'
)
%tbody
-
@users
.
each
do
|
user
|
%tr
%td
%strong
=
link_to
user
.
name
,
user
%td
=
@events
.
code_push
.
where
(
author_id:
user
).
count
%td
=
@events
.
issues
.
created
.
where
(
author_id:
user
).
count
%td
=
@events
.
issues
.
closed
.
where
(
author_id:
user
).
count
%td
=
@events
.
merge_requests
.
created
.
where
(
author_id:
user
).
count
%td
=
@events
.
merge_requests
.
merged
.
where
(
author_id:
user
).
count
%td
=
@events
.
where
(
author_id:
user
).
count
-
[
:push
,
:issues
,
:merge_requests
].
each
do
|
scope
|
:javascript
var
data
=
{
labels
:
#{
@users
.
map
(
&
:name
).
to_json
}
,
datasets
:
[
{
fillColor
:
"
rgba(220,220,220,0.5)
"
,
strokeColor
:
"
rgba(220,220,220,1)
"
,
barStrokeWidth
:
1
,
barValueSpacing
:
1
,
barDatasetSpacing
:
1
,
data
:
#{
@stats
[
scope
].
to_json
}
}
]
}
var
ctx
=
$
(
"
##{scope}
"
).
get
(
0
).
getContext
(
"
2d
"
);
new
Chart
(
ctx
).
Bar
(
data
,{
"
scaleOverlay
"
:
true
,
responsive
:
true
,
maintainAspectRatio
:
false
});
:javascript
$
(
"
#event-stats
"
).
tablesorter
();
app/views/layouts/nav/_group.html.haml
View file @
c3254358
...
@@ -38,6 +38,11 @@
...
@@ -38,6 +38,11 @@
=
icon
(
'users fw'
)
=
icon
(
'users fw'
)
%span
%span
Members
Members
=
nav_link
(
controller:
[
:stats
])
do
=
link_to
group_stats_path
(
@group
),
title:
'Stats'
,
data:
{
placement:
'right'
}
do
=
icon
(
'table fw'
)
%span
Statistics
-
if
can?
(
current_user
,
:admin_group
,
@group
)
-
if
can?
(
current_user
,
:admin_group
,
@group
)
=
nav_link
(
html_options:
{
class:
"separate-item"
})
do
=
nav_link
(
html_options:
{
class:
"separate-item"
})
do
=
link_to
edit_group_path
(
@group
),
title:
'Settings'
do
=
link_to
edit_group_path
(
@group
),
title:
'Settings'
do
...
...
config/routes.rb
View file @
c3254358
...
@@ -386,6 +386,7 @@ Rails.application.routes.draw do
...
@@ -386,6 +386,7 @@ Rails.application.routes.draw do
end
end
scope
module: :groups
do
scope
module: :groups
do
resource
:stats
,
only:
[
:show
]
resource
:ldap
,
only:
[]
do
resource
:ldap
,
only:
[]
do
member
do
member
do
put
:reset_access
put
:reset_access
...
...
features/group/statistics.feature
0 → 100644
View file @
c3254358
Feature
:
Group Statistics
Background
:
Given
I sign in as
"John Doe"
And
"John Doe"
is owner of group
"Owned"
Scenario
:
I
should see group
"Owned"
statistics page
When
I visit group
"Owned"
page
And
I click on group statistics
Then
I should see group statistics page
features/steps/group/statistics.rb
0 → 100644
View file @
c3254358
class
Spinach::Features::GroupStatistics
<
Spinach
::
FeatureSteps
include
SharedAuthentication
include
SharedPaths
include
SharedGroup
include
SharedUser
step
'I click on group statistics'
do
click_link
'Statistics'
end
step
'I should see group statistics page'
do
expect
(
page
).
to
have_content
"Contribution statistics for issues, merge requests and push"
end
end
vendor/assets/javascripts/jquery.tablesorter.js
0 → 100644
View file @
c3254358
/*
*
* TableSorter 2.0 - Client-side table sorting with ease!
* Version 2.0.5b
* @requires jQuery v1.2.3
*
* Copyright (c) 2007 Christian Bach
* Examples and docs at: http://tablesorter.com
* Dual licensed under the MIT and GPL licenses:
* http://www.opensource.org/licenses/mit-license.php
* http://www.gnu.org/licenses/gpl.html
*
*/
/**
*
* @description Create a sortable table with multi-column sorting capabilitys
*
* @example $('table').tablesorter();
* @desc Create a simple tablesorter interface.
*
* @example $('table').tablesorter({ sortList:[[0,0],[1,0]] });
* @desc Create a tablesorter interface and sort on the first and secound column column headers.
*
* @example $('table').tablesorter({ headers: { 0: { sorter: false}, 1: {sorter: false} } });
*
* @desc Create a tablesorter interface and disableing the first and second column headers.
*
*
* @example $('table').tablesorter({ headers: { 0: {sorter:"integer"}, 1: {sorter:"currency"} } });
*
* @desc Create a tablesorter interface and set a column parser for the first
* and second column.
*
*
* @param Object
* settings An object literal containing key/value pairs to provide
* optional settings.
*
*
* @option String cssHeader (optional) A string of the class name to be appended
* to sortable tr elements in the thead of the table. Default value:
* "header"
*
* @option String cssAsc (optional) A string of the class name to be appended to
* sortable tr elements in the thead on a ascending sort. Default value:
* "headerSortUp"
*
* @option String cssDesc (optional) A string of the class name to be appended
* to sortable tr elements in the thead on a descending sort. Default
* value: "headerSortDown"
*
* @option String sortInitialOrder (optional) A string of the inital sorting
* order can be asc or desc. Default value: "asc"
*
* @option String sortMultisortKey (optional) A string of the multi-column sort
* key. Default value: "shiftKey"
*
* @option String textExtraction (optional) A string of the text-extraction
* method to use. For complex html structures inside td cell set this
* option to "complex", on large tables the complex option can be slow.
* Default value: "simple"
*
* @option Object headers (optional) An object of instructions for per-column
* controls in the format: headers: { 0: { option: setting }, ... }. For
* example, to disable sorting on the first two columns of a table:
* headers: { 0: { sorter: false}, 1: {sorter: false} }.
* Default value: null.
*
* @option Array sortList (optional) An array of instructions for per-column sorting
* and direction in the format: [[columnIndex, sortDirection], ... ] where
* columnIndex is a zero-based index for your columns left-to-right and
* sortDirection is 0 for Ascending and 1 for Descending. A valid argument
* that sorts ascending first by column 1 and then column 2 looks like:
* [[0,0],[1,0]]. Default value: null.
*
* @option Array sortForce (optional) An array containing forced sorting rules.
* Use to add an additional forced sort that will be appended to the dynamic
* selections by the user. For example, can be used to sort people alphabetically
* after some other user-selected sort that results in rows with the same value
* like dates or money due. It can help prevent data from appearing as though it
* has a random secondary sort. Default value: null.
*
* @option Boolean sortLocaleCompare (optional) Boolean flag indicating whatever
* to use String.localeCampare method or not. Default set to true.
*
*
* @option Array sortAppend (optional) An array containing forced sorting rules.
* This option let's you specify a default sorting rule, which is
* appended to user-selected rules. Default value: null
*
* @option Boolean widthFixed (optional) Boolean flag indicating if tablesorter
* should apply fixed widths to the table columns. This is usefull when
* using the pager companion plugin. This options requires the dimension
* jquery plugin. Default value: false
*
* @option Boolean cancelSelection (optional) Boolean flag indicating if
* tablesorter should cancel selection of the table headers text.
* Default value: true
*
* @option Boolean debug (optional) Boolean flag indicating if tablesorter
* should display debuging information usefull for development.
*
* @type jQuery
*
* @name tablesorter
*
* @cat Plugins/Tablesorter
*
* @author Christian Bach/christian.bach@polyester.se
*/
(
function
(
$
)
{
$
.
extend
({
tablesorter
:
new
function
()
{
var
parsers
=
[],
widgets
=
[];
this
.
defaults
=
{
cssHeader
:
"
header
"
,
cssAsc
:
"
headerSortUp
"
,
cssDesc
:
"
headerSortDown
"
,
cssChildRow
:
"
expand-child
"
,
sortInitialOrder
:
"
asc
"
,
sortMultiSortKey
:
"
shiftKey
"
,
sortForce
:
null
,
sortAppend
:
null
,
sortLocaleCompare
:
true
,
textExtraction
:
"
simple
"
,
parsers
:
{},
widgets
:
[],
widgetZebra
:
{
css
:
[
"
even
"
,
"
odd
"
]
},
headers
:
{},
widthFixed
:
false
,
cancelSelection
:
true
,
sortList
:
[],
headerList
:
[],
dateFormat
:
"
us
"
,
decimal
:
'
/
\
.|
\
,/g
'
,
onRenderHeader
:
null
,
selectorHeaders
:
'
thead th
'
,
debug
:
false
};
/* debuging utils */
function
benchmark
(
s
,
d
)
{
log
(
s
+
"
,
"
+
(
new
Date
().
getTime
()
-
d
.
getTime
())
+
"
ms
"
);
}
this
.
benchmark
=
benchmark
;
function
log
(
s
)
{
if
(
typeof
console
!=
"
undefined
"
&&
typeof
console
.
debug
!=
"
undefined
"
)
{
console
.
log
(
s
);
}
else
{
alert
(
s
);
}
}
/* parsers utils */
function
buildParserCache
(
table
,
$headers
)
{
if
(
table
.
config
.
debug
)
{
var
parsersDebug
=
""
;
}
if
(
table
.
tBodies
.
length
==
0
)
return
;
// In the case of empty tables
var
rows
=
table
.
tBodies
[
0
].
rows
;
if
(
rows
[
0
])
{
var
list
=
[],
cells
=
rows
[
0
].
cells
,
l
=
cells
.
length
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
var
p
=
false
;
if
(
$
.
metadata
&&
(
$
(
$headers
[
i
]).
metadata
()
&&
$
(
$headers
[
i
]).
metadata
().
sorter
))
{
p
=
getParserById
(
$
(
$headers
[
i
]).
metadata
().
sorter
);
}
else
if
((
table
.
config
.
headers
[
i
]
&&
table
.
config
.
headers
[
i
].
sorter
))
{
p
=
getParserById
(
table
.
config
.
headers
[
i
].
sorter
);
}
if
(
!
p
)
{
p
=
detectParserForColumn
(
table
,
rows
,
-
1
,
i
);
}
if
(
table
.
config
.
debug
)
{
parsersDebug
+=
"
column:
"
+
i
+
"
parser:
"
+
p
.
id
+
"
\n
"
;
}
list
.
push
(
p
);
}
}
if
(
table
.
config
.
debug
)
{
log
(
parsersDebug
);
}
return
list
;
};
function
detectParserForColumn
(
table
,
rows
,
rowIndex
,
cellIndex
)
{
var
l
=
parsers
.
length
,
node
=
false
,
nodeValue
=
false
,
keepLooking
=
true
;
while
(
nodeValue
==
''
&&
keepLooking
)
{
rowIndex
++
;
if
(
rows
[
rowIndex
])
{
node
=
getNodeFromRowAndCellIndex
(
rows
,
rowIndex
,
cellIndex
);
nodeValue
=
trimAndGetNodeText
(
table
.
config
,
node
);
if
(
table
.
config
.
debug
)
{
log
(
'
Checking if value was empty on row:
'
+
rowIndex
);
}
}
else
{
keepLooking
=
false
;
}
}
for
(
var
i
=
1
;
i
<
l
;
i
++
)
{
if
(
parsers
[
i
].
is
(
nodeValue
,
table
,
node
))
{
return
parsers
[
i
];
}
}
// 0 is always the generic parser (text)
return
parsers
[
0
];
}
function
getNodeFromRowAndCellIndex
(
rows
,
rowIndex
,
cellIndex
)
{
return
rows
[
rowIndex
].
cells
[
cellIndex
];
}
function
trimAndGetNodeText
(
config
,
node
)
{
return
$
.
trim
(
getElementText
(
config
,
node
));
}
function
getParserById
(
name
)
{
var
l
=
parsers
.
length
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
if
(
parsers
[
i
].
id
.
toLowerCase
()
==
name
.
toLowerCase
())
{
return
parsers
[
i
];
}
}
return
false
;
}
/* utils */
function
buildCache
(
table
)
{
if
(
table
.
config
.
debug
)
{
var
cacheTime
=
new
Date
();
}
var
totalRows
=
(
table
.
tBodies
[
0
]
&&
table
.
tBodies
[
0
].
rows
.
length
)
||
0
,
totalCells
=
(
table
.
tBodies
[
0
].
rows
[
0
]
&&
table
.
tBodies
[
0
].
rows
[
0
].
cells
.
length
)
||
0
,
parsers
=
table
.
config
.
parsers
,
cache
=
{
row
:
[],
normalized
:
[]
};
for
(
var
i
=
0
;
i
<
totalRows
;
++
i
)
{
/** Add the table data to main data array */
var
c
=
$
(
table
.
tBodies
[
0
].
rows
[
i
]),
cols
=
[];
// if this is a child row, add it to the last row's children and
// continue to the next row
if
(
c
.
hasClass
(
table
.
config
.
cssChildRow
))
{
cache
.
row
[
cache
.
row
.
length
-
1
]
=
cache
.
row
[
cache
.
row
.
length
-
1
].
add
(
c
);
// go to the next for loop
continue
;
}
cache
.
row
.
push
(
c
);
for
(
var
j
=
0
;
j
<
totalCells
;
++
j
)
{
cols
.
push
(
parsers
[
j
].
format
(
getElementText
(
table
.
config
,
c
[
0
].
cells
[
j
]),
table
,
c
[
0
].
cells
[
j
]));
}
cols
.
push
(
cache
.
normalized
.
length
);
// add position for rowCache
cache
.
normalized
.
push
(
cols
);
cols
=
null
;
};
if
(
table
.
config
.
debug
)
{
benchmark
(
"
Building cache for
"
+
totalRows
+
"
rows:
"
,
cacheTime
);
}
return
cache
;
};
function
getElementText
(
config
,
node
)
{
if
(
!
node
)
return
""
;
var
$node
=
$
(
node
),
data
=
$node
.
attr
(
'
data-sort-value
'
);
if
(
data
!==
undefined
)
return
data
;
var
text
=
""
;
if
(
!
config
.
supportsTextContent
)
config
.
supportsTextContent
=
node
.
textContent
||
false
;
if
(
config
.
textExtraction
==
"
simple
"
)
{
if
(
config
.
supportsTextContent
)
{
text
=
node
.
textContent
;
}
else
{
if
(
node
.
childNodes
[
0
]
&&
node
.
childNodes
[
0
].
hasChildNodes
())
{
text
=
node
.
childNodes
[
0
].
innerHTML
;
}
else
{
text
=
node
.
innerHTML
;
}
}
}
else
{
if
(
typeof
(
config
.
textExtraction
)
==
"
function
"
)
{
text
=
config
.
textExtraction
(
node
);
}
else
{
text
=
$
(
node
).
text
();
}
}
return
text
;
}
function
appendToTable
(
table
,
cache
)
{
if
(
table
.
config
.
debug
)
{
var
appendTime
=
new
Date
()
}
var
c
=
cache
,
r
=
c
.
row
,
n
=
c
.
normalized
,
totalRows
=
n
.
length
,
checkCell
=
(
n
[
0
].
length
-
1
),
tableBody
=
$
(
table
.
tBodies
[
0
]),
rows
=
[];
for
(
var
i
=
0
;
i
<
totalRows
;
i
++
)
{
var
pos
=
n
[
i
][
checkCell
];
rows
.
push
(
r
[
pos
]);
if
(
!
table
.
config
.
appender
)
{
//var o = ;
var
l
=
r
[
pos
].
length
;
for
(
var
j
=
0
;
j
<
l
;
j
++
)
{
tableBody
[
0
].
appendChild
(
r
[
pos
][
j
]);
}
//
}
}
if
(
table
.
config
.
appender
)
{
table
.
config
.
appender
(
table
,
rows
);
}
rows
=
null
;
if
(
table
.
config
.
debug
)
{
benchmark
(
"
Rebuilt table:
"
,
appendTime
);
}
// apply table widgets
applyWidget
(
table
);
// trigger sortend
setTimeout
(
function
()
{
$
(
table
).
trigger
(
"
sortEnd
"
);
},
0
);
};
function
buildHeaders
(
table
)
{
if
(
table
.
config
.
debug
)
{
var
time
=
new
Date
();
}
var
meta
=
(
$
.
metadata
)
?
true
:
false
;
var
header_index
=
computeTableHeaderCellIndexes
(
table
);
var
$tableHeaders
=
$
(
table
.
config
.
selectorHeaders
,
table
).
each
(
function
(
index
)
{
this
.
column
=
header_index
[
this
.
parentNode
.
rowIndex
+
"
-
"
+
this
.
cellIndex
];
// this.column = index;
this
.
order
=
formatSortingOrder
(
table
.
config
.
sortInitialOrder
);
this
.
count
=
this
.
order
;
if
(
checkHeaderMetadata
(
this
)
||
checkHeaderOptions
(
table
,
index
))
this
.
sortDisabled
=
true
;
if
(
checkHeaderOptionsSortingLocked
(
table
,
index
))
this
.
order
=
this
.
lockedOrder
=
checkHeaderOptionsSortingLocked
(
table
,
index
);
if
(
!
this
.
sortDisabled
)
{
var
$th
=
$
(
this
).
addClass
(
table
.
config
.
cssHeader
);
if
(
table
.
config
.
onRenderHeader
)
table
.
config
.
onRenderHeader
.
apply
(
$th
);
}
// add cell to headerList
table
.
config
.
headerList
[
index
]
=
this
;
});
if
(
table
.
config
.
debug
)
{
benchmark
(
"
Built headers:
"
,
time
);
log
(
$tableHeaders
);
}
return
$tableHeaders
;
};
// from:
// http://www.javascripttoolbox.com/lib/table/examples.php
// http://www.javascripttoolbox.com/temp/table_cellindex.html
function
computeTableHeaderCellIndexes
(
t
)
{
var
matrix
=
[];
var
lookup
=
{};
var
thead
=
t
.
getElementsByTagName
(
'
THEAD
'
)[
0
];
var
trs
=
thead
.
getElementsByTagName
(
'
TR
'
);
for
(
var
i
=
0
;
i
<
trs
.
length
;
i
++
)
{
var
cells
=
trs
[
i
].
cells
;
for
(
var
j
=
0
;
j
<
cells
.
length
;
j
++
)
{
var
c
=
cells
[
j
];
var
rowIndex
=
c
.
parentNode
.
rowIndex
;
var
cellId
=
rowIndex
+
"
-
"
+
c
.
cellIndex
;
var
rowSpan
=
c
.
rowSpan
||
1
;
var
colSpan
=
c
.
colSpan
||
1
var
firstAvailCol
;
if
(
typeof
(
matrix
[
rowIndex
])
==
"
undefined
"
)
{
matrix
[
rowIndex
]
=
[];
}
// Find first available column in the first row
for
(
var
k
=
0
;
k
<
matrix
[
rowIndex
].
length
+
1
;
k
++
)
{
if
(
typeof
(
matrix
[
rowIndex
][
k
])
==
"
undefined
"
)
{
firstAvailCol
=
k
;
break
;
}
}
lookup
[
cellId
]
=
firstAvailCol
;
for
(
var
k
=
rowIndex
;
k
<
rowIndex
+
rowSpan
;
k
++
)
{
if
(
typeof
(
matrix
[
k
])
==
"
undefined
"
)
{
matrix
[
k
]
=
[];
}
var
matrixrow
=
matrix
[
k
];
for
(
var
l
=
firstAvailCol
;
l
<
firstAvailCol
+
colSpan
;
l
++
)
{
matrixrow
[
l
]
=
"
x
"
;
}
}
}
}
return
lookup
;
}
function
checkCellColSpan
(
table
,
rows
,
row
)
{
var
arr
=
[],
r
=
table
.
tHead
.
rows
,
c
=
r
[
row
].
cells
;
for
(
var
i
=
0
;
i
<
c
.
length
;
i
++
)
{
var
cell
=
c
[
i
];
if
(
cell
.
colSpan
>
1
)
{
arr
=
arr
.
concat
(
checkCellColSpan
(
table
,
headerArr
,
row
++
));
}
else
{
if
(
table
.
tHead
.
length
==
1
||
(
cell
.
rowSpan
>
1
||
!
r
[
row
+
1
]))
{
arr
.
push
(
cell
);
}
// headerArr[row] = (i+row);
}
}
return
arr
;
};
function
checkHeaderMetadata
(
cell
)
{
if
((
$
.
metadata
)
&&
(
$
(
cell
).
metadata
().
sorter
===
false
))
{
return
true
;
};
return
false
;
}
function
checkHeaderOptions
(
table
,
i
)
{
if
((
table
.
config
.
headers
[
i
])
&&
(
table
.
config
.
headers
[
i
].
sorter
===
false
))
{
return
true
;
};
return
false
;
}
function
checkHeaderOptionsSortingLocked
(
table
,
i
)
{
if
((
table
.
config
.
headers
[
i
])
&&
(
table
.
config
.
headers
[
i
].
lockedOrder
))
return
table
.
config
.
headers
[
i
].
lockedOrder
;
return
false
;
}
function
applyWidget
(
table
)
{
var
c
=
table
.
config
.
widgets
;
var
l
=
c
.
length
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
getWidgetById
(
c
[
i
]).
format
(
table
);
}
}
function
getWidgetById
(
name
)
{
var
l
=
widgets
.
length
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
if
(
widgets
[
i
].
id
.
toLowerCase
()
==
name
.
toLowerCase
())
{
return
widgets
[
i
];
}
}
};
function
formatSortingOrder
(
v
)
{
if
(
typeof
(
v
)
!=
"
Number
"
)
{
return
(
v
.
toLowerCase
()
==
"
desc
"
)
?
1
:
0
;
}
else
{
return
(
v
==
1
)
?
1
:
0
;
}
}
function
isValueInArray
(
v
,
a
)
{
var
l
=
a
.
length
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
if
(
a
[
i
][
0
]
==
v
)
{
return
true
;
}
}
return
false
;
}
function
setHeadersCss
(
table
,
$headers
,
list
,
css
)
{
// remove all header information
$headers
.
removeClass
(
css
[
0
]).
removeClass
(
css
[
1
]);
var
h
=
[];
$headers
.
each
(
function
(
offset
)
{
if
(
!
this
.
sortDisabled
)
{
h
[
this
.
column
]
=
$
(
this
);
}
});
var
l
=
list
.
length
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
h
[
list
[
i
][
0
]].
addClass
(
css
[
list
[
i
][
1
]]);
}
}
function
fixColumnWidth
(
table
,
$headers
)
{
var
c
=
table
.
config
;
if
(
c
.
widthFixed
)
{
var
colgroup
=
$
(
'
<colgroup>
'
);
$
(
"
tr:first td
"
,
table
.
tBodies
[
0
]).
each
(
function
()
{
colgroup
.
append
(
$
(
'
<col>
'
).
css
(
'
width
'
,
$
(
this
).
width
()));
});
$
(
table
).
prepend
(
colgroup
);
};
}
function
updateHeaderSortCount
(
table
,
sortList
)
{
var
c
=
table
.
config
,
l
=
sortList
.
length
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
var
s
=
sortList
[
i
],
o
=
c
.
headerList
[
s
[
0
]];
o
.
count
=
s
[
1
];
o
.
count
++
;
}
}
/* sorting methods */
var
sortWrapper
;
function
multisort
(
table
,
sortList
,
cache
)
{
if
(
table
.
config
.
debug
)
{
var
sortTime
=
new
Date
();
}
var
dynamicExp
=
"
sortWrapper = function(a,b) {
"
,
l
=
sortList
.
length
;
// TODO: inline functions.
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
var
c
=
sortList
[
i
][
0
];
var
order
=
sortList
[
i
][
1
];
// var s = (getCachedSortType(table.config.parsers,c) == "text") ?
// ((order == 0) ? "sortText" : "sortTextDesc") : ((order == 0) ?
// "sortNumeric" : "sortNumericDesc");
// var s = (table.config.parsers[c].type == "text") ? ((order == 0)
// ? makeSortText(c) : makeSortTextDesc(c)) : ((order == 0) ?
// makeSortNumeric(c) : makeSortNumericDesc(c));
var
s
=
(
table
.
config
.
parsers
[
c
].
type
==
"
text
"
)
?
((
order
==
0
)
?
makeSortFunction
(
"
text
"
,
"
asc
"
,
c
)
:
makeSortFunction
(
"
text
"
,
"
desc
"
,
c
))
:
((
order
==
0
)
?
makeSortFunction
(
"
numeric
"
,
"
asc
"
,
c
)
:
makeSortFunction
(
"
numeric
"
,
"
desc
"
,
c
));
var
e
=
"
e
"
+
i
;
dynamicExp
+=
"
var
"
+
e
+
"
=
"
+
s
;
// + "(a[" + c + "],b[" + c
// + "]); ";
dynamicExp
+=
"
if(
"
+
e
+
"
) { return
"
+
e
+
"
; }
"
;
dynamicExp
+=
"
else {
"
;
}
// if value is the same keep orignal order
var
orgOrderCol
=
cache
.
normalized
[
0
].
length
-
1
;
dynamicExp
+=
"
return a[
"
+
orgOrderCol
+
"
]-b[
"
+
orgOrderCol
+
"
];
"
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
dynamicExp
+=
"
};
"
;
}
dynamicExp
+=
"
return 0;
"
;
dynamicExp
+=
"
};
"
;
if
(
table
.
config
.
debug
)
{
benchmark
(
"
Evaling expression:
"
+
dynamicExp
,
new
Date
());
}
eval
(
dynamicExp
);
cache
.
normalized
.
sort
(
sortWrapper
);
if
(
table
.
config
.
debug
)
{
benchmark
(
"
Sorting on
"
+
sortList
.
toString
()
+
"
and dir
"
+
order
+
"
time:
"
,
sortTime
);
}
return
cache
;
};
function
makeSortFunction
(
type
,
direction
,
index
)
{
var
a
=
"
a[
"
+
index
+
"
]
"
,
b
=
"
b[
"
+
index
+
"
]
"
;
if
(
type
==
'
text
'
&&
direction
==
'
asc
'
)
{
return
"
(
"
+
a
+
"
==
"
+
b
+
"
? 0 : (
"
+
a
+
"
=== null ? Number.POSITIVE_INFINITY : (
"
+
b
+
"
=== null ? Number.NEGATIVE_INFINITY : (
"
+
a
+
"
<
"
+
b
+
"
) ? -1 : 1 )));
"
;
}
else
if
(
type
==
'
text
'
&&
direction
==
'
desc
'
)
{
return
"
(
"
+
a
+
"
==
"
+
b
+
"
? 0 : (
"
+
a
+
"
=== null ? Number.POSITIVE_INFINITY : (
"
+
b
+
"
=== null ? Number.NEGATIVE_INFINITY : (
"
+
b
+
"
<
"
+
a
+
"
) ? -1 : 1 )));
"
;
}
else
if
(
type
==
'
numeric
'
&&
direction
==
'
asc
'
)
{
return
"
(
"
+
a
+
"
=== null &&
"
+
b
+
"
=== null) ? 0 :(
"
+
a
+
"
=== null ? Number.POSITIVE_INFINITY : (
"
+
b
+
"
=== null ? Number.NEGATIVE_INFINITY :
"
+
a
+
"
-
"
+
b
+
"
));
"
;
}
else
if
(
type
==
'
numeric
'
&&
direction
==
'
desc
'
)
{
return
"
(
"
+
a
+
"
=== null &&
"
+
b
+
"
=== null) ? 0 :(
"
+
a
+
"
=== null ? Number.POSITIVE_INFINITY : (
"
+
b
+
"
=== null ? Number.NEGATIVE_INFINITY :
"
+
b
+
"
-
"
+
a
+
"
));
"
;
}
};
function
makeSortText
(
i
)
{
return
"
((a[
"
+
i
+
"
] < b[
"
+
i
+
"
]) ? -1 : ((a[
"
+
i
+
"
] > b[
"
+
i
+
"
]) ? 1 : 0));
"
;
};
function
makeSortTextDesc
(
i
)
{
return
"
((b[
"
+
i
+
"
] < a[
"
+
i
+
"
]) ? -1 : ((b[
"
+
i
+
"
] > a[
"
+
i
+
"
]) ? 1 : 0));
"
;
};
function
makeSortNumeric
(
i
)
{
return
"
a[
"
+
i
+
"
]-b[
"
+
i
+
"
];
"
;
};
function
makeSortNumericDesc
(
i
)
{
return
"
b[
"
+
i
+
"
]-a[
"
+
i
+
"
];
"
;
};
function
sortText
(
a
,
b
)
{
if
(
table
.
config
.
sortLocaleCompare
)
return
a
.
localeCompare
(
b
);
return
((
a
<
b
)
?
-
1
:
((
a
>
b
)
?
1
:
0
));
};
function
sortTextDesc
(
a
,
b
)
{
if
(
table
.
config
.
sortLocaleCompare
)
return
b
.
localeCompare
(
a
);
return
((
b
<
a
)
?
-
1
:
((
b
>
a
)
?
1
:
0
));
};
function
sortNumeric
(
a
,
b
)
{
return
a
-
b
;
};
function
sortNumericDesc
(
a
,
b
)
{
return
b
-
a
;
};
function
getCachedSortType
(
parsers
,
i
)
{
return
parsers
[
i
].
type
;
};
/* public methods */
this
.
construct
=
function
(
settings
)
{
return
this
.
each
(
function
()
{
// if no thead or tbody quit.
if
(
!
this
.
tHead
||
!
this
.
tBodies
)
return
;
// declare
var
$this
,
$document
,
$headers
,
cache
,
config
,
shiftDown
=
0
,
sortOrder
;
// new blank config object
this
.
config
=
{};
// merge and extend.
config
=
$
.
extend
(
this
.
config
,
$
.
tablesorter
.
defaults
,
settings
);
// store common expression for speed
$this
=
$
(
this
);
// save the settings where they read
$
.
data
(
this
,
"
tablesorter
"
,
config
);
// build headers
$headers
=
buildHeaders
(
this
);
// try to auto detect column type, and store in tables config
this
.
config
.
parsers
=
buildParserCache
(
this
,
$headers
);
// build the cache for the tbody cells
cache
=
buildCache
(
this
);
// get the css class names, could be done else where.
var
sortCSS
=
[
config
.
cssDesc
,
config
.
cssAsc
];
// fixate columns if the users supplies the fixedWidth option
fixColumnWidth
(
this
);
// apply event handling to headers
// this is to big, perhaps break it out?
$headers
.
click
(
function
(
e
)
{
var
totalRows
=
(
$this
[
0
].
tBodies
[
0
]
&&
$this
[
0
].
tBodies
[
0
].
rows
.
length
)
||
0
;
if
(
!
this
.
sortDisabled
&&
totalRows
>
0
)
{
// Only call sortStart if sorting is
// enabled.
$this
.
trigger
(
"
sortStart
"
);
// store exp, for speed
var
$cell
=
$
(
this
);
// get current column index
var
i
=
this
.
column
;
// get current column sort order
this
.
order
=
this
.
count
++
%
2
;
// always sort on the locked order.
if
(
this
.
lockedOrder
)
this
.
order
=
this
.
lockedOrder
;
// user only whants to sort on one
// column
if
(
!
e
[
config
.
sortMultiSortKey
])
{
// flush the sort list
config
.
sortList
=
[];
if
(
config
.
sortForce
!=
null
)
{
var
a
=
config
.
sortForce
;
for
(
var
j
=
0
;
j
<
a
.
length
;
j
++
)
{
if
(
a
[
j
][
0
]
!=
i
)
{
config
.
sortList
.
push
(
a
[
j
]);
}
}
}
// add column to sort list
config
.
sortList
.
push
([
i
,
this
.
order
]);
// multi column sorting
}
else
{
// the user has clicked on an all
// ready sortet column.
if
(
isValueInArray
(
i
,
config
.
sortList
))
{
// revers the sorting direction
// for all tables.
for
(
var
j
=
0
;
j
<
config
.
sortList
.
length
;
j
++
)
{
var
s
=
config
.
sortList
[
j
],
o
=
config
.
headerList
[
s
[
0
]];
if
(
s
[
0
]
==
i
)
{
o
.
count
=
s
[
1
];
o
.
count
++
;
s
[
1
]
=
o
.
count
%
2
;
}
}
}
else
{
// add column to sort list array
config
.
sortList
.
push
([
i
,
this
.
order
]);
}
};
setTimeout
(
function
()
{
// set css for headers
setHeadersCss
(
$this
[
0
],
$headers
,
config
.
sortList
,
sortCSS
);
appendToTable
(
$this
[
0
],
multisort
(
$this
[
0
],
config
.
sortList
,
cache
)
);
},
1
);
// stop normal event by returning false
return
false
;
}
// cancel selection
}).
mousedown
(
function
()
{
if
(
config
.
cancelSelection
)
{
this
.
onselectstart
=
function
()
{
return
false
};
return
false
;
}
});
// apply easy methods that trigger binded events
$this
.
bind
(
"
update
"
,
function
()
{
var
me
=
this
;
setTimeout
(
function
()
{
// rebuild parsers.
me
.
config
.
parsers
=
buildParserCache
(
me
,
$headers
);
// rebuild the cache map
cache
=
buildCache
(
me
);
},
1
);
}).
bind
(
"
updateCell
"
,
function
(
e
,
cell
)
{
var
config
=
this
.
config
;
// get position from the dom.
var
pos
=
[(
cell
.
parentNode
.
rowIndex
-
1
),
cell
.
cellIndex
];
// update cache
cache
.
normalized
[
pos
[
0
]][
pos
[
1
]]
=
config
.
parsers
[
pos
[
1
]].
format
(
getElementText
(
config
,
cell
),
cell
);
}).
bind
(
"
sorton
"
,
function
(
e
,
list
)
{
$
(
this
).
trigger
(
"
sortStart
"
);
config
.
sortList
=
list
;
// update and store the sortlist
var
sortList
=
config
.
sortList
;
// update header count index
updateHeaderSortCount
(
this
,
sortList
);
// set css for headers
setHeadersCss
(
this
,
$headers
,
sortList
,
sortCSS
);
// sort the table and append it to the dom
appendToTable
(
this
,
multisort
(
this
,
sortList
,
cache
));
}).
bind
(
"
appendCache
"
,
function
()
{
appendToTable
(
this
,
cache
);
}).
bind
(
"
applyWidgetId
"
,
function
(
e
,
id
)
{
getWidgetById
(
id
).
format
(
this
);
}).
bind
(
"
applyWidgets
"
,
function
()
{
// apply widgets
applyWidget
(
this
);
});
if
(
$
.
metadata
&&
(
$
(
this
).
metadata
()
&&
$
(
this
).
metadata
().
sortlist
))
{
config
.
sortList
=
$
(
this
).
metadata
().
sortlist
;
}
// if user has supplied a sort list to constructor.
if
(
config
.
sortList
.
length
>
0
)
{
$this
.
trigger
(
"
sorton
"
,
[
config
.
sortList
]);
}
// apply widgets
applyWidget
(
this
);
});
};
this
.
addParser
=
function
(
parser
)
{
var
l
=
parsers
.
length
,
a
=
true
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
if
(
parsers
[
i
].
id
.
toLowerCase
()
==
parser
.
id
.
toLowerCase
())
{
a
=
false
;
}
}
if
(
a
)
{
parsers
.
push
(
parser
);
};
};
this
.
addWidget
=
function
(
widget
)
{
widgets
.
push
(
widget
);
};
this
.
formatFloat
=
function
(
s
)
{
var
i
=
parseFloat
(
s
);
return
(
isNaN
(
i
))
?
0
:
i
;
};
this
.
formatInt
=
function
(
s
)
{
var
i
=
parseInt
(
s
);
return
(
isNaN
(
i
))
?
0
:
i
;
};
this
.
isDigit
=
function
(
s
,
config
)
{
// replace all an wanted chars and match.
return
/^
[
-+
]?\d
*$/
.
test
(
$
.
trim
(
s
.
replace
(
/
[
,.'
]
/g
,
''
)));
};
this
.
clearTableBody
=
function
(
table
)
{
if
(
$
.
browser
.
msie
)
{
while
(
table
.
tBodies
[
0
].
firstChild
)
{
table
.
tBodies
[
0
].
removeChild
(
table
.
tBodies
[
0
].
firstChild
);
}
}
else
{
table
.
tBodies
[
0
].
innerHTML
=
""
;
}
};
}
});
// extend plugin scope
$
.
fn
.
extend
({
tablesorter
:
$
.
tablesorter
.
construct
});
// make shortcut
var
ts
=
$
.
tablesorter
;
// add default parsers
ts
.
addParser
({
id
:
"
text
"
,
is
:
function
(
s
)
{
return
true
;
},
format
:
function
(
s
)
{
return
$
.
trim
(
s
.
toLocaleLowerCase
());
},
type
:
"
text
"
});
ts
.
addParser
({
id
:
"
digit
"
,
is
:
function
(
s
,
table
)
{
var
c
=
table
.
config
;
return
$
.
tablesorter
.
isDigit
(
s
,
c
);
},
format
:
function
(
s
)
{
return
$
.
tablesorter
.
formatFloat
(
s
);
},
type
:
"
numeric
"
});
ts
.
addParser
({
id
:
"
currency
"
,
is
:
function
(
s
)
{
return
/^
[
£$€?.
]
/
.
test
(
s
);
},
format
:
function
(
s
)
{
return
$
.
tablesorter
.
formatFloat
(
s
.
replace
(
new
RegExp
(
/
[
£$€
]
/g
),
""
));
},
type
:
"
numeric
"
});
ts
.
addParser
({
id
:
"
ipAddress
"
,
is
:
function
(
s
)
{
return
/^
\d{2,3}[\.]\d{2,3}[\.]\d{2,3}[\.]\d{2,3}
$/
.
test
(
s
);
},
format
:
function
(
s
)
{
var
a
=
s
.
split
(
"
.
"
),
r
=
""
,
l
=
a
.
length
;
for
(
var
i
=
0
;
i
<
l
;
i
++
)
{
var
item
=
a
[
i
];
if
(
item
.
length
==
2
)
{
r
+=
"
0
"
+
item
;
}
else
{
r
+=
item
;
}
}
return
$
.
tablesorter
.
formatFloat
(
r
);
},
type
:
"
numeric
"
});
ts
.
addParser
({
id
:
"
url
"
,
is
:
function
(
s
)
{
return
/^
(
https
?
|ftp|file
)
:
\/\/
$/
.
test
(
s
);
},
format
:
function
(
s
)
{
return
jQuery
.
trim
(
s
.
replace
(
new
RegExp
(
/
(
https
?
|ftp|file
)
:
\/\/
/
),
''
));
},
type
:
"
text
"
});
ts
.
addParser
({
id
:
"
isoDate
"
,
is
:
function
(
s
)
{
return
/^
\d{4}[\/
-
]\d{1,2}[\/
-
]\d{1,2}
$/
.
test
(
s
);
},
format
:
function
(
s
)
{
return
$
.
tablesorter
.
formatFloat
((
s
!=
""
)
?
new
Date
(
s
.
replace
(
new
RegExp
(
/-/g
),
"
/
"
)).
getTime
()
:
"
0
"
);
},
type
:
"
numeric
"
});
ts
.
addParser
({
id
:
"
percent
"
,
is
:
function
(
s
)
{
return
/
\%
$/
.
test
(
$
.
trim
(
s
));
},
format
:
function
(
s
)
{
return
$
.
tablesorter
.
formatFloat
(
s
.
replace
(
new
RegExp
(
/%/g
),
""
));
},
type
:
"
numeric
"
});
ts
.
addParser
({
id
:
"
usLongDate
"
,
is
:
function
(
s
)
{
return
s
.
match
(
new
RegExp
(
/^
[
A-Za-z
]{3,10}\.?
[
0-9
]{1,2}
,
([
0-9
]{4}
|'
?[
0-9
]{2})
(([
0-2
]?[
0-9
]
:
[
0-5
][
0-9
])
|
([
0-1
]?[
0-9
]
:
[
0-5
][
0-9
]\s(
AM|PM
)))
$/
));
},
format
:
function
(
s
)
{
return
$
.
tablesorter
.
formatFloat
(
new
Date
(
s
).
getTime
());
},
type
:
"
numeric
"
});
ts
.
addParser
({
id
:
"
shortDate
"
,
is
:
function
(
s
)
{
return
/
\d{1,2}[\/\-]\d{1,2}[\/\-]\d{2,4}
/
.
test
(
s
);
},
format
:
function
(
s
,
table
)
{
var
c
=
table
.
config
;
s
=
s
.
replace
(
/
\-
/g
,
"
/
"
);
if
(
c
.
dateFormat
==
"
us
"
)
{
// reformat the string in ISO format
s
=
s
.
replace
(
/
(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{4})
/
,
"
$3/$1/$2
"
);
}
if
(
c
.
dateFormat
==
"
pt
"
)
{
s
=
s
.
replace
(
/
(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{4})
/
,
"
$3/$2/$1
"
);
}
else
if
(
c
.
dateFormat
==
"
uk
"
)
{
// reformat the string in ISO format
s
=
s
.
replace
(
/
(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{4})
/
,
"
$3/$2/$1
"
);
}
else
if
(
c
.
dateFormat
==
"
dd/mm/yy
"
||
c
.
dateFormat
==
"
dd-mm-yy
"
)
{
s
=
s
.
replace
(
/
(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{2})
/
,
"
$1/$2/$3
"
);
}
return
$
.
tablesorter
.
formatFloat
(
new
Date
(
s
).
getTime
());
},
type
:
"
numeric
"
});
ts
.
addParser
({
id
:
"
time
"
,
is
:
function
(
s
)
{
return
/^
(([
0-2
]?[
0-9
]
:
[
0-5
][
0-9
])
|
([
0-1
]?[
0-9
]
:
[
0-5
][
0-9
]\s(
am|pm
)))
$/
.
test
(
s
);
},
format
:
function
(
s
)
{
return
$
.
tablesorter
.
formatFloat
(
new
Date
(
"
2000/01/01
"
+
s
).
getTime
());
},
type
:
"
numeric
"
});
ts
.
addParser
({
id
:
"
metadata
"
,
is
:
function
(
s
)
{
return
false
;
},
format
:
function
(
s
,
table
,
cell
)
{
var
c
=
table
.
config
,
p
=
(
!
c
.
parserMetadataName
)
?
'
sortValue
'
:
c
.
parserMetadataName
;
return
$
(
cell
).
metadata
()[
p
];
},
type
:
"
numeric
"
});
// add default widgets
ts
.
addWidget
({
id
:
"
zebra
"
,
format
:
function
(
table
)
{
if
(
table
.
config
.
debug
)
{
var
time
=
new
Date
();
}
var
$tr
,
row
=
-
1
,
odd
;
// loop through the visible rows
$
(
"
tr:visible
"
,
table
.
tBodies
[
0
]).
each
(
function
(
i
)
{
$tr
=
$
(
this
);
// style children rows the same way the parent
// row was styled
if
(
!
$tr
.
hasClass
(
table
.
config
.
cssChildRow
))
row
++
;
odd
=
(
row
%
2
==
0
);
$tr
.
removeClass
(
table
.
config
.
widgetZebra
.
css
[
odd
?
0
:
1
]).
addClass
(
table
.
config
.
widgetZebra
.
css
[
odd
?
1
:
0
])
});
if
(
table
.
config
.
debug
)
{
$
.
tablesorter
.
benchmark
(
"
Applying Zebra widget
"
,
time
);
}
}
});
})(
jQuery
);
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