keithsalisbury-subtrac 0.1.4 → 0.1.5
Sign up to get free protection for your applications and to get access to all the features.
- data/VERSION.yml +1 -1
- data/bin/subtrac +2 -0
- data/lib/subtrac.rb +85 -50
- data/lib/subtrac/config/config.yml +22 -19
- data/lib/subtrac/templates/location.erb +2 -2
- data/lib/subtrac/templates/projects/blank/trac/wiki/WikiStart +1 -1
- data/lib/subtrac/templates/trac.erb +1 -1
- data/lib/subtrac/templates/vhost.erb +6 -6
- metadata +1 -105
- data/lib/subtrac/trac-plugins/advancedticketworkflowplugin/advancedworkflow/__init__.py +0 -0
- data/lib/subtrac/trac-plugins/advancedticketworkflowplugin/advancedworkflow/controller.py +0 -419
- data/lib/subtrac/trac-plugins/advancedticketworkflowplugin/setup.cfg +0 -3
- data/lib/subtrac/trac-plugins/advancedticketworkflowplugin/setup.py +0 -20
- data/lib/subtrac/trac-plugins/clientsplugin/clients/__init__.py +0 -0
- data/lib/subtrac/trac-plugins/clientsplugin/clients/action.py +0 -28
- data/lib/subtrac/trac-plugins/clientsplugin/clients/action_email.py +0 -168
- data/lib/subtrac/trac-plugins/clientsplugin/clients/action_zendesk_forum.py +0 -137
- data/lib/subtrac/trac-plugins/clientsplugin/clients/admin.py +0 -91
- data/lib/subtrac/trac-plugins/clientsplugin/clients/api.py +0 -199
- data/lib/subtrac/trac-plugins/clientsplugin/clients/client.py +0 -105
- data/lib/subtrac/trac-plugins/clientsplugin/clients/events.py +0 -287
- data/lib/subtrac/trac-plugins/clientsplugin/clients/eventsadmin.py +0 -71
- data/lib/subtrac/trac-plugins/clientsplugin/clients/htdocs/clients.css +0 -4
- data/lib/subtrac/trac-plugins/clientsplugin/clients/model.py +0 -135
- data/lib/subtrac/trac-plugins/clientsplugin/clients/processor.py +0 -70
- data/lib/subtrac/trac-plugins/clientsplugin/clients/reportmanager.py +0 -142
- data/lib/subtrac/trac-plugins/clientsplugin/clients/reports.py +0 -231
- data/lib/subtrac/trac-plugins/clientsplugin/clients/summary.py +0 -27
- data/lib/subtrac/trac-plugins/clientsplugin/clients/summary_milestone.py +0 -152
- data/lib/subtrac/trac-plugins/clientsplugin/clients/summary_ticketchanges.py +0 -160
- data/lib/subtrac/trac-plugins/clientsplugin/clients/templates/admin_client_events.html +0 -124
- data/lib/subtrac/trac-plugins/clientsplugin/clients/templates/admin_clients.html +0 -134
- data/lib/subtrac/trac-plugins/clientsplugin/cron/changes.xslt +0 -132
- data/lib/subtrac/trac-plugins/clientsplugin/cron/run-client-event +0 -97
- data/lib/subtrac/trac-plugins/clientsplugin/cron/summary.xslt +0 -161
- data/lib/subtrac/trac-plugins/clientsplugin/setup.py +0 -43
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/__init__.py +0 -4
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/burndownchart.py +0 -273
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/hoursinplaceeditor.py +0 -44
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/hoursremaining.py +0 -36
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/htdocs/jquery-1.2.3.min.js +0 -32
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/htdocs/jquery.jeditable.js +0 -409
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/htdocs/jquery.jeditable.mini.js +0 -30
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/templates/edithours.html +0 -53
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/tests/burndownchart.py +0 -181
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/tests/hoursremaining.py +0 -66
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/tests/workloadchart.py +0 -47
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/utils.py +0 -93
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/workloadchart.py +0 -86
- data/lib/subtrac/trac-plugins/estimationtoolsplugin/setup.py +0 -20
- data/lib/subtrac/trac-plugins/timingandestimationplugin/scripts/SumRollups.js +0 -23
- data/lib/subtrac/trac-plugins/timingandestimationplugin/scripts/adw_tracdb.py +0 -128
- data/lib/subtrac/trac-plugins/timingandestimationplugin/scripts/git-post-receive +0 -40
- data/lib/subtrac/trac-plugins/timingandestimationplugin/scripts/trac-post-commit.py +0 -285
- data/lib/subtrac/trac-plugins/timingandestimationplugin/scripts/trac_billing.py +0 -173
- data/lib/subtrac/trac-plugins/timingandestimationplugin/scripts/utils/__init__.py +0 -0
- data/lib/subtrac/trac-plugins/timingandestimationplugin/scripts/utils/mail.py +0 -164
- data/lib/subtrac/trac-plugins/timingandestimationplugin/setup.py +0 -69
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/__init__.py +0 -1
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/api.py +0 -292
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/blackmagic.py +0 -172
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/dbhelper.py +0 -178
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/htdocs/billingplugin.css +0 -25
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/htdocs/field_disabler.js +0 -6
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/htdocs/formatDate.js +0 -356
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/htdocs/js/tip_centerwindow.js +0 -100
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/htdocs/js/tip_followscroll.js +0 -84
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/htdocs/js/wz_tooltip.js +0 -1149
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/htdocs/linkifyer.js +0 -119
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/htdocs/query.js +0 -73
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/htdocs/ticket.js +0 -165
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/query_webui.py +0 -28
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/reportmanager.py +0 -221
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/reports.py +0 -675
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/reports_filter.py +0 -150
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/statuses.py +0 -25
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/tande_filters.py +0 -131
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/templates/billing.cs +0 -84
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/templates/billing.html +0 -104
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/ticket_daemon.py +0 -194
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/ticket_policy.py +0 -62
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/ticket_webui.py +0 -28
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/usermanual.py +0 -127
- data/lib/subtrac/trac-plugins/timingandestimationplugin/timingandestimationplugin/webui.py +0 -129
- data/lib/subtrac/trac-plugins/worklogplugin/setup.py +0 -29
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/__init__.py +0 -1
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/api.py +0 -187
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/jqModal.css +0 -40
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/jqModal.js +0 -67
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/jquery.mousewheel.pack.js +0 -12
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/jquery.timeentry.pack.js +0 -7
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/tracWorklog.js +0 -40
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/ui.datepicker.css +0 -208
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/ui.datepicker.js +0 -1439
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/work.png +0 -0
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/work.xcf +0 -0
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/worklogplugin.css +0 -80
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/workstart.png +0 -0
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/htdocs/workstop.png +0 -0
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/manager.py +0 -336
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/reports.py +0 -598
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/templates/worklog.html +0 -45
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/templates/worklog_stop.html +0 -70
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/templates/worklog_user.html +0 -40
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/templates/worklog_webadminui.html +0 -59
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/ticket_daemon.py +0 -33
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/ticket_filter.py +0 -153
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/timeline_hook.py +0 -96
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/usermanual.py +0 -29
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/util.py +0 -31
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/webadminui.py +0 -47
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/webui.py +0 -174
- data/lib/subtrac/trac-plugins/worklogplugin/worklog/xmlrpc.py +0 -73
@@ -1,43 +0,0 @@
|
|
1
|
-
#!/usr/bin/env python
|
2
|
-
|
3
|
-
from setuptools import setup
|
4
|
-
|
5
|
-
PACKAGE = 'clients'
|
6
|
-
|
7
|
-
setup(name=PACKAGE,
|
8
|
-
description='Plugin to allow management of which ticket belong to which client',
|
9
|
-
keywords='trac plugin ticket client',
|
10
|
-
version='0.3',
|
11
|
-
url='http://www.trac-hacks.org/wiki/ClientsPlugin',
|
12
|
-
license='http://www.opensource.org/licenses/mit-license.php',
|
13
|
-
author='Colin Guthrie',
|
14
|
-
author_email='trac@colin.guthr.ie',
|
15
|
-
long_description="""
|
16
|
-
I'll write this later!
|
17
|
-
""",
|
18
|
-
packages=[PACKAGE],
|
19
|
-
package_data={PACKAGE : ['templates/*.html', 'htdocs/*.css']},
|
20
|
-
entry_points = {
|
21
|
-
'trac.plugins': [
|
22
|
-
'clients.api = clients.api',
|
23
|
-
'clients.client = clients.client',
|
24
|
-
'clients.model = clients.model',
|
25
|
-
'clients.admin = clients.admin',
|
26
|
-
'clients.events = clients.events',
|
27
|
-
'clients.eventsadmin = clients.eventsadmin',
|
28
|
-
'clients.processor = clients.processor',
|
29
|
-
'clients.summary = clients.summary',
|
30
|
-
'clients.summary_milestone = clients.summary_milestone',
|
31
|
-
'clients.summary_ticketchanges = clients.summary_ticketchanges',
|
32
|
-
'clients.action = clients.action',
|
33
|
-
'clients.action_email = clients.action_email',
|
34
|
-
'clients.action_zendesk_forum = clients.action_zendesk_forum',
|
35
|
-
]
|
36
|
-
})
|
37
|
-
|
38
|
-
#### AUTHORS ####
|
39
|
-
## Primary Author:
|
40
|
-
## Colin Guthrie
|
41
|
-
## http://colin.guthr.ie/
|
42
|
-
## trac@colin.guthr.ie
|
43
|
-
## trac-hacks user: coling
|
@@ -1,273 +0,0 @@
|
|
1
|
-
from decimal import Decimal, InvalidOperation, ROUND_HALF_UP
|
2
|
-
from datetime import datetime
|
3
|
-
from datetime import timedelta
|
4
|
-
from estimationtools.utils import parse_options, execute_query, get_estimation_field, get_closed_states
|
5
|
-
from trac.core import TracError
|
6
|
-
from trac.util.html import Markup
|
7
|
-
from trac.util.datefmt import utc
|
8
|
-
from trac.wiki.macros import WikiMacroBase
|
9
|
-
import copy
|
10
|
-
|
11
|
-
DEFAULT_OPTIONS = {'width': '800', 'height': '200', 'color': 'ff9900', 'expected': '0',
|
12
|
-
'bgcolor': 'ffffff00', 'wecolor':'ccccccaa', 'colorexpected': 'ffddaa', 'weekends':'true', 'gridlines' : '0'}
|
13
|
-
|
14
|
-
class BurndownChart(WikiMacroBase):
|
15
|
-
"""Creates burn down chart for selected tickets.
|
16
|
-
|
17
|
-
This macro creates a chart that can be used to visualize the progress in a milestone (e.g., sprint or
|
18
|
-
product backlog).
|
19
|
-
For a given set of tickets and a time frame, the remaining estimated effort is calculated.
|
20
|
-
|
21
|
-
The macro has the following parameters:
|
22
|
-
* a comma-separated list of query parameters for the ticket selection, in the form "key=value" as specified in TracQuery#QueryLanguage.
|
23
|
-
* `startdate`: '''mandatory''' parameter that specifies the start date of the period (ISO8601 format)
|
24
|
-
* `enddate`: end date of the period. If omitted, it defaults to either the milestones (if given) `completed' date,
|
25
|
-
or `due` date, or today (in that order) (ISO8601 format)
|
26
|
-
* `weekends`: include weekends in chart. Defaults to `true`
|
27
|
-
* `expected`: show expected progress in chart, 0 or any number to define initial expected hours (defaults to 0).
|
28
|
-
* `gridlines`: show gridlines in chart, 0 or any number to define hour steps (defaults to 0)
|
29
|
-
* `width`: width of resulting diagram (defaults to 800)
|
30
|
-
* `height`: height of resulting diagram (defaults to 200)
|
31
|
-
* `color`: color specified as 6-letter string of hexadecimal values in the format `RRGGBB`.
|
32
|
-
Defaults to `ff9900`, a nice orange.
|
33
|
-
* `colorexpected`: color for expected hours graph specified as 6-letter string of hexadecimal values in the format `RRGGBB`.
|
34
|
-
Defaults to ffddaa, a nice yellow.
|
35
|
-
* `bgcolor`: chart drawing area background color specified as 6-letter string of hexadecimal values in the format `RRGGBB`.
|
36
|
-
Defaults to `ffffff`.
|
37
|
-
* `wecolor`: chart drawing area background color for weekends specified as 6-letter string of hexadecimal values in the format `RRGGBB`.
|
38
|
-
Defaults to `ccccccaa`.
|
39
|
-
|
40
|
-
Examples:
|
41
|
-
{{{
|
42
|
-
[[BurndownChart(milestone=Sprint 1, startdate=2008-01-01)]]
|
43
|
-
[[BurndownChart(milestone=Release 3.0|Sprint 1, startdate=2008-01-01, enddate=2008-01-15,
|
44
|
-
weekends=false, expected=100, gridlines=20, width=600, height=100, color=0000ff)]]
|
45
|
-
}}}
|
46
|
-
"""
|
47
|
-
|
48
|
-
estimation_field = get_estimation_field()
|
49
|
-
closed_states = get_closed_states()
|
50
|
-
|
51
|
-
def render_macro(self, req, name, content):
|
52
|
-
|
53
|
-
# prepare options
|
54
|
-
options, query_args = parse_options(self.env.get_db_cnx(), content, copy.copy(DEFAULT_OPTIONS))
|
55
|
-
|
56
|
-
if not options['startdate']:
|
57
|
-
raise TracError("No start date specified!")
|
58
|
-
|
59
|
-
# minimum time frame is one day
|
60
|
-
if (options['startdate'] >= options['enddate']):
|
61
|
-
options['enddate'] = options['startdate'] + timedelta(days=1)
|
62
|
-
|
63
|
-
# calculate data
|
64
|
-
timetable = self._calculate_timetable(options, query_args, req)
|
65
|
-
|
66
|
-
# remove weekends
|
67
|
-
if not options['weekends']:
|
68
|
-
for date in timetable.keys():
|
69
|
-
if date.weekday() >= 5:
|
70
|
-
del timetable[date]
|
71
|
-
|
72
|
-
# scale data
|
73
|
-
xdata, ydata, maxhours = self._scale_data(timetable, options)
|
74
|
-
|
75
|
-
# build html for google chart api
|
76
|
-
dates = sorted(timetable.keys())
|
77
|
-
bottomaxis = "0:|" + ("|").join([str(date.day) for date in dates]) + \
|
78
|
-
"|1:|%s/%s|%s/%s" % (dates[0].month, dates[0].year, dates[ - 1].month, dates[ - 1].year)
|
79
|
-
leftaxis = "2,0,%s" % maxhours
|
80
|
-
|
81
|
-
# add line for expected progress
|
82
|
-
if options['expected'] == '0':
|
83
|
-
expecteddata = ""
|
84
|
-
else:
|
85
|
-
expecteddata = "|0,100|%s,0" % (round(Decimal(options['expected']) * 100 / maxhours, 2))
|
86
|
-
|
87
|
-
# prepare gridlines
|
88
|
-
if options['gridlines'] == '0':
|
89
|
-
gridlinesdata = "100.0,100.0,1,0" # create top and right bounding line by using grid
|
90
|
-
else:
|
91
|
-
gridlinesdata = "%s,%s" % (xdata[1], (round(Decimal(options['gridlines']) * 100 / maxhours, 4)))
|
92
|
-
|
93
|
-
# mark weekends
|
94
|
-
weekends = []
|
95
|
-
saturday = None
|
96
|
-
index = 0
|
97
|
-
halfday = self._round(Decimal("0.5") / (len(dates) - 1))
|
98
|
-
for date in dates:
|
99
|
-
if date.weekday() == 5:
|
100
|
-
saturday = index
|
101
|
-
if saturday and date.weekday() == 6:
|
102
|
-
weekends.append("R,%s,0,%s,%s" %
|
103
|
-
(options['wecolor'],
|
104
|
-
self._round((Decimal(xdata[saturday]) / 100) - halfday),
|
105
|
-
self._round((Decimal(xdata[index]) / 100) + halfday)))
|
106
|
-
saturday = None
|
107
|
-
index += 1
|
108
|
-
# special handling if time period starts with Sundays...
|
109
|
-
if len(dates) > 0 and dates[0].weekday() == 6:
|
110
|
-
weekends.append("R,%s,0,0.0,%s" % (options['wecolor'], halfday))
|
111
|
-
# or ends with Saturday
|
112
|
-
if len(dates) > 0 and dates[ - 1].weekday() == 5:
|
113
|
-
weekends.append("R,%s,0,%s,1.0" % (options['wecolor'], Decimal(1) - halfday))
|
114
|
-
|
115
|
-
title = ''
|
116
|
-
if options.get('milestone'):
|
117
|
-
title = options['milestone'].split('|')[0]
|
118
|
-
|
119
|
-
return Markup("<img src=\"http://chart.apis.google.com/chart?"
|
120
|
-
"chs=%sx%s"
|
121
|
-
"&chf=c,s,%s|bg,s,00000000"
|
122
|
-
"&chd=t:%s|%s%s"
|
123
|
-
"&cht=lxy"
|
124
|
-
"&chxt=x,x,y"
|
125
|
-
"&chxl=%s"
|
126
|
-
"&chxr=%s"
|
127
|
-
"&chm=%s"
|
128
|
-
"&chg=%s"
|
129
|
-
"&chco=%s,%s"
|
130
|
-
"&chtt=%s\" "
|
131
|
-
"alt=\'Burndown Chart\' />"
|
132
|
-
% (options['width'], options['height'],
|
133
|
-
options['bgcolor'],
|
134
|
-
",".join(xdata), ",".join(ydata), expecteddata, bottomaxis, leftaxis,
|
135
|
-
"|".join(weekends), gridlinesdata,
|
136
|
-
options['color'], options['colorexpected'], title))
|
137
|
-
|
138
|
-
def _calculate_timetable(self, options, query_args, req):
|
139
|
-
db = self.env.get_db_cnx()
|
140
|
-
|
141
|
-
# create dictionary with entry for each day of the required time period
|
142
|
-
timetable = {}
|
143
|
-
|
144
|
-
current_date = options['startdate']
|
145
|
-
while current_date <= options['enddate']:
|
146
|
-
timetable[current_date] = Decimal(0)
|
147
|
-
current_date += timedelta(days=1)
|
148
|
-
|
149
|
-
# get current values for all tickets within milestone and sprints
|
150
|
-
|
151
|
-
query_args[self.estimation_field + "!"] = None
|
152
|
-
tickets = execute_query(self.env, req, query_args)
|
153
|
-
|
154
|
-
# add the open effort for each ticket for each day to the timetable
|
155
|
-
|
156
|
-
for t in tickets:
|
157
|
-
|
158
|
-
# Record the current (latest) status and estimate, and ticket
|
159
|
-
# creation date
|
160
|
-
|
161
|
-
creation_date = t['time'].date()
|
162
|
-
latest_status = t['status']
|
163
|
-
latest_estimate = self._cast_estimate(t[self.estimation_field])
|
164
|
-
if latest_estimate is None:
|
165
|
-
latest_estimate = Decimal(0)
|
166
|
-
|
167
|
-
# Fetch change history for status and effort fields for this ticket
|
168
|
-
history_cursor = db.cursor()
|
169
|
-
history_cursor.execute("SELECT "
|
170
|
-
"DISTINCT c.field as field, c.time AS time, c.oldvalue as oldvalue, c.newvalue as newvalue "
|
171
|
-
"FROM ticket t, ticket_change c "
|
172
|
-
"WHERE t.id = %s and c.ticket = t.id and (c.field=%s or c.field='status')"
|
173
|
-
"ORDER BY c.time ASC", [t['id'], self.estimation_field])
|
174
|
-
|
175
|
-
# Build up two dictionaries, mapping dates when effort/status
|
176
|
-
# changed, to the latest effort/status on that day (in case of
|
177
|
-
# several changes on the same day). Also record the oldest known
|
178
|
-
# effort/status, i.e. that at the time of ticket creation
|
179
|
-
|
180
|
-
estimate_history = {}
|
181
|
-
status_history = {}
|
182
|
-
|
183
|
-
earliest_estimate = None
|
184
|
-
earliest_status = None
|
185
|
-
|
186
|
-
for row in history_cursor:
|
187
|
-
row_field, row_time, row_old, row_new = row
|
188
|
-
event_date = datetime.fromtimestamp(row_time, utc).date()
|
189
|
-
if row_field == self.estimation_field:
|
190
|
-
new_value = self._cast_estimate(row_new)
|
191
|
-
if new_value is not None:
|
192
|
-
estimate_history[event_date] = new_value
|
193
|
-
if earliest_estimate is None:
|
194
|
-
earliest_estimate = self._cast_estimate(row_old)
|
195
|
-
elif row_field == 'status':
|
196
|
-
status_history[event_date] = row_new
|
197
|
-
if earliest_status is None:
|
198
|
-
earliest_status = row_old
|
199
|
-
|
200
|
-
# If we don't know already (i.e. the ticket effort/status was
|
201
|
-
# not changed on the creation date), set the effort on the
|
202
|
-
# creation date. It may be that we don't have an "earliest"
|
203
|
-
# estimate/status, because it was never changed. In this case,
|
204
|
-
# use the current (latest) value.
|
205
|
-
|
206
|
-
if not creation_date in estimate_history:
|
207
|
-
if earliest_estimate is not None:
|
208
|
-
estimate_history[creation_date] = earliest_estimate
|
209
|
-
else:
|
210
|
-
estimate_history[creation_date] = latest_estimate
|
211
|
-
if not creation_date in status_history:
|
212
|
-
if earliest_status is not None:
|
213
|
-
status_history[creation_date] = earliest_status
|
214
|
-
else:
|
215
|
-
status_history[creation_date] = latest_status
|
216
|
-
|
217
|
-
# Finally estimates to the timetable. Treat any period where the
|
218
|
-
# ticket was closed as estimate 0. We need to loop from ticket
|
219
|
-
# creation date, not just from the timetable start date, since
|
220
|
-
# it's possible that the ticket was changed between these two
|
221
|
-
# dates.
|
222
|
-
|
223
|
-
current_date = creation_date
|
224
|
-
current_estimate = None
|
225
|
-
is_open = None
|
226
|
-
|
227
|
-
while current_date <= options['enddate']:
|
228
|
-
if current_date in status_history:
|
229
|
-
is_open = (status_history[current_date] not in self.closed_states)
|
230
|
-
|
231
|
-
if current_date in estimate_history:
|
232
|
-
current_estimate = estimate_history[current_date]
|
233
|
-
|
234
|
-
if current_date >= options['startdate'] and is_open:
|
235
|
-
timetable[current_date] += current_estimate
|
236
|
-
|
237
|
-
current_date += timedelta(days=1)
|
238
|
-
|
239
|
-
return timetable
|
240
|
-
|
241
|
-
def _scale_data(self, timetable, options):
|
242
|
-
# create sorted list of dates
|
243
|
-
dates = timetable.keys()
|
244
|
-
dates.sort()
|
245
|
-
|
246
|
-
maxhours = max(timetable.values())
|
247
|
-
|
248
|
-
if maxhours <= Decimal(0):
|
249
|
-
maxhours = Decimal(100)
|
250
|
-
ydata = [str(self._round(timetable[d] * Decimal(100) / maxhours))
|
251
|
-
for d in dates]
|
252
|
-
xdata = [str(self._round(x * Decimal(100) / (len(dates) - 1)))
|
253
|
-
for x in range((options['enddate'] - options['startdate']).days + 1)]
|
254
|
-
|
255
|
-
# mark ydata invalid that is after today
|
256
|
-
if options['enddate'] > options['today']:
|
257
|
-
remaining_days = (options['enddate'] - options['today']).days;
|
258
|
-
ydata = ydata[: - remaining_days] + ['-1' for x in xrange(0, remaining_days)]
|
259
|
-
|
260
|
-
return xdata, ydata, maxhours
|
261
|
-
|
262
|
-
def _round(self, decimal_):
|
263
|
-
return decimal_.quantize(Decimal("0.01"), ROUND_HALF_UP)
|
264
|
-
|
265
|
-
def _cast_estimate(self, estimate):
|
266
|
-
# Treat 0, empty string or None as 0.0
|
267
|
-
if not estimate:
|
268
|
-
return Decimal(0)
|
269
|
-
try:
|
270
|
-
return Decimal(estimate)
|
271
|
-
except (TypeError, ValueError, InvalidOperation):
|
272
|
-
# Treat other incorrect values as None
|
273
|
-
return None
|
@@ -1,44 +0,0 @@
|
|
1
|
-
from estimationtools.utils import get_estimation_field
|
2
|
-
from pkg_resources import resource_filename
|
3
|
-
from trac.core import implements, Component
|
4
|
-
from trac.web.api import IRequestFilter, IRequestHandler
|
5
|
-
from trac.web.chrome import ITemplateProvider, add_script
|
6
|
-
|
7
|
-
class HoursInPlaceEditor(Component):
|
8
|
-
"""A filter to implement in-place editing for estimated hours field in query page.
|
9
|
-
|
10
|
-
Requires Trac XML-RPC Plug-in.
|
11
|
-
"""
|
12
|
-
|
13
|
-
implements(IRequestFilter, IRequestHandler, ITemplateProvider)
|
14
|
-
|
15
|
-
estimation_field = get_estimation_field()
|
16
|
-
|
17
|
-
# IRequestHandler methods
|
18
|
-
def match_request(self, req):
|
19
|
-
return req.path_info.startswith('/estimationtools')
|
20
|
-
|
21
|
-
def process_request(self, req):
|
22
|
-
data = {}
|
23
|
-
data['field'] = self.estimation_field
|
24
|
-
return 'edithours.html', {'data': data}, 'text/javascript'
|
25
|
-
|
26
|
-
# IRequestFilter methods
|
27
|
-
def pre_process_request(self, req, handler):
|
28
|
-
return handler
|
29
|
-
|
30
|
-
def post_process_request(self, req, template, data, content_type):
|
31
|
-
if (req.path_info.startswith('/query') or req.path_info.startswith('/report')
|
32
|
-
and req.perm.has_permission('TICKET_MODIFY')
|
33
|
-
and req.perm.has_permission('XML_RPC')):
|
34
|
-
# add_script(req, 'estimationtools/jquery-1.2.3.min.js')
|
35
|
-
add_script(req, 'estimationtools/jquery.jeditable.mini.js')
|
36
|
-
add_script(req, '/estimationtools/edithours.js')
|
37
|
-
return template, data, content_type
|
38
|
-
|
39
|
-
# ITemplateProvider methods
|
40
|
-
def get_htdocs_dirs(self):
|
41
|
-
return [('estimationtools', resource_filename(__name__, 'htdocs'))]
|
42
|
-
|
43
|
-
def get_templates_dirs(self):
|
44
|
-
return [resource_filename(__name__, 'templates')]
|
@@ -1,36 +0,0 @@
|
|
1
|
-
from estimationtools.utils import get_estimation_field, execute_query
|
2
|
-
from trac.wiki.macros import WikiMacroBase
|
3
|
-
from trac.wiki.api import parse_args
|
4
|
-
|
5
|
-
class HoursRemaining(WikiMacroBase):
|
6
|
-
"""Calculates remaining estimated hours for the queried tickets.
|
7
|
-
|
8
|
-
The macro accepts a comma-separated list of query parameters for the ticket selection,
|
9
|
-
in the form "key=value" as specified in TracQuery#QueryLanguage.
|
10
|
-
|
11
|
-
Example:
|
12
|
-
{{{
|
13
|
-
[[HoursRemaining(milestone=Sprint 1)]]
|
14
|
-
}}}
|
15
|
-
"""
|
16
|
-
|
17
|
-
estimation_field = get_estimation_field()
|
18
|
-
|
19
|
-
def render_macro(self, req, name, content):
|
20
|
-
_, options = parse_args(content, strict=False)
|
21
|
-
|
22
|
-
# we have to add custom estimation field to query so that field is added to
|
23
|
-
# resulting ticket list
|
24
|
-
options[self.estimation_field + "!"] = None
|
25
|
-
|
26
|
-
tickets = execute_query(self.env, req, options)
|
27
|
-
|
28
|
-
sum = 0.0
|
29
|
-
for t in tickets:
|
30
|
-
try:
|
31
|
-
sum += float(t[self.estimation_field])
|
32
|
-
except:
|
33
|
-
pass
|
34
|
-
|
35
|
-
return "%s" % int(sum)
|
36
|
-
|
data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/htdocs/jquery-1.2.3.min.js
DELETED
@@ -1,32 +0,0 @@
|
|
1
|
-
/*
|
2
|
-
* jQuery 1.2.3 - New Wave Javascript
|
3
|
-
*
|
4
|
-
* Copyright (c) 2008 John Resig (jquery.com)
|
5
|
-
* Dual licensed under the MIT (MIT-LICENSE.txt)
|
6
|
-
* and GPL (GPL-LICENSE.txt) licenses.
|
7
|
-
*
|
8
|
-
* $Date: 2008-02-06 00:21:25 -0500 (Wed, 06 Feb 2008) $
|
9
|
-
* $Rev: 4663 $
|
10
|
-
*/
|
11
|
-
(function(){if(window.jQuery)var _jQuery=window.jQuery;var jQuery=window.jQuery=function(selector,context){return new jQuery.prototype.init(selector,context);};if(window.$)var _$=window.$;window.$=jQuery;var quickExpr=/^[^<]*(<(.|\s)+>)[^>]*$|^#(\w+)$/;var isSimple=/^.[^:#\[\.]*$/;jQuery.fn=jQuery.prototype={init:function(selector,context){selector=selector||document;if(selector.nodeType){this[0]=selector;this.length=1;return this;}else if(typeof selector=="string"){var match=quickExpr.exec(selector);if(match&&(match[1]||!context)){if(match[1])selector=jQuery.clean([match[1]],context);else{var elem=document.getElementById(match[3]);if(elem)if(elem.id!=match[3])return jQuery().find(selector);else{this[0]=elem;this.length=1;return this;}else
|
12
|
-
selector=[];}}else
|
13
|
-
return new jQuery(context).find(selector);}else if(jQuery.isFunction(selector))return new jQuery(document)[jQuery.fn.ready?"ready":"load"](selector);return this.setArray(selector.constructor==Array&&selector||(selector.jquery||selector.length&&selector!=window&&!selector.nodeType&&selector[0]!=undefined&&selector[0].nodeType)&&jQuery.makeArray(selector)||[selector]);},jquery:"1.2.3",size:function(){return this.length;},length:0,get:function(num){return num==undefined?jQuery.makeArray(this):this[num];},pushStack:function(elems){var ret=jQuery(elems);ret.prevObject=this;return ret;},setArray:function(elems){this.length=0;Array.prototype.push.apply(this,elems);return this;},each:function(callback,args){return jQuery.each(this,callback,args);},index:function(elem){var ret=-1;this.each(function(i){if(this==elem)ret=i;});return ret;},attr:function(name,value,type){var options=name;if(name.constructor==String)if(value==undefined)return this.length&&jQuery[type||"attr"](this[0],name)||undefined;else{options={};options[name]=value;}return this.each(function(i){for(name in options)jQuery.attr(type?this.style:this,name,jQuery.prop(this,options[name],type,i,name));});},css:function(key,value){if((key=='width'||key=='height')&&parseFloat(value)<0)value=undefined;return this.attr(key,value,"curCSS");},text:function(text){if(typeof text!="object"&&text!=null)return this.empty().append((this[0]&&this[0].ownerDocument||document).createTextNode(text));var ret="";jQuery.each(text||this,function(){jQuery.each(this.childNodes,function(){if(this.nodeType!=8)ret+=this.nodeType!=1?this.nodeValue:jQuery.fn.text([this]);});});return ret;},wrapAll:function(html){if(this[0])jQuery(html,this[0].ownerDocument).clone().insertBefore(this[0]).map(function(){var elem=this;while(elem.firstChild)elem=elem.firstChild;return elem;}).append(this);return this;},wrapInner:function(html){return this.each(function(){jQuery(this).contents().wrapAll(html);});},wrap:function(html){return this.each(function(){jQuery(this).wrapAll(html);});},append:function(){return this.domManip(arguments,true,false,function(elem){if(this.nodeType==1)this.appendChild(elem);});},prepend:function(){return this.domManip(arguments,true,true,function(elem){if(this.nodeType==1)this.insertBefore(elem,this.firstChild);});},before:function(){return this.domManip(arguments,false,false,function(elem){this.parentNode.insertBefore(elem,this);});},after:function(){return this.domManip(arguments,false,true,function(elem){this.parentNode.insertBefore(elem,this.nextSibling);});},end:function(){return this.prevObject||jQuery([]);},find:function(selector){var elems=jQuery.map(this,function(elem){return jQuery.find(selector,elem);});return this.pushStack(/[^+>] [^+>]/.test(selector)||selector.indexOf("..")>-1?jQuery.unique(elems):elems);},clone:function(events){var ret=this.map(function(){if(jQuery.browser.msie&&!jQuery.isXMLDoc(this)){var clone=this.cloneNode(true),container=document.createElement("div");container.appendChild(clone);return jQuery.clean([container.innerHTML])[0];}else
|
14
|
-
return this.cloneNode(true);});var clone=ret.find("*").andSelf().each(function(){if(this[expando]!=undefined)this[expando]=null;});if(events===true)this.find("*").andSelf().each(function(i){if(this.nodeType==3)return;var events=jQuery.data(this,"events");for(var type in events)for(var handler in events[type])jQuery.event.add(clone[i],type,events[type][handler],events[type][handler].data);});return ret;},filter:function(selector){return this.pushStack(jQuery.isFunction(selector)&&jQuery.grep(this,function(elem,i){return selector.call(elem,i);})||jQuery.multiFilter(selector,this));},not:function(selector){if(selector.constructor==String)if(isSimple.test(selector))return this.pushStack(jQuery.multiFilter(selector,this,true));else
|
15
|
-
selector=jQuery.multiFilter(selector,this);var isArrayLike=selector.length&&selector[selector.length-1]!==undefined&&!selector.nodeType;return this.filter(function(){return isArrayLike?jQuery.inArray(this,selector)<0:this!=selector;});},add:function(selector){return!selector?this:this.pushStack(jQuery.merge(this.get(),selector.constructor==String?jQuery(selector).get():selector.length!=undefined&&(!selector.nodeName||jQuery.nodeName(selector,"form"))?selector:[selector]));},is:function(selector){return selector?jQuery.multiFilter(selector,this).length>0:false;},hasClass:function(selector){return this.is("."+selector);},val:function(value){if(value==undefined){if(this.length){var elem=this[0];if(jQuery.nodeName(elem,"select")){var index=elem.selectedIndex,values=[],options=elem.options,one=elem.type=="select-one";if(index<0)return null;for(var i=one?index:0,max=one?index+1:options.length;i<max;i++){var option=options[i];if(option.selected){value=jQuery.browser.msie&&!option.attributes.value.specified?option.text:option.value;if(one)return value;values.push(value);}}return values;}else
|
16
|
-
return(this[0].value||"").replace(/\r/g,"");}return undefined;}return this.each(function(){if(this.nodeType!=1)return;if(value.constructor==Array&&/radio|checkbox/.test(this.type))this.checked=(jQuery.inArray(this.value,value)>=0||jQuery.inArray(this.name,value)>=0);else if(jQuery.nodeName(this,"select")){var values=value.constructor==Array?value:[value];jQuery("option",this).each(function(){this.selected=(jQuery.inArray(this.value,values)>=0||jQuery.inArray(this.text,values)>=0);});if(!values.length)this.selectedIndex=-1;}else
|
17
|
-
this.value=value;});},html:function(value){return value==undefined?(this.length?this[0].innerHTML:null):this.empty().append(value);},replaceWith:function(value){return this.after(value).remove();},eq:function(i){return this.slice(i,i+1);},slice:function(){return this.pushStack(Array.prototype.slice.apply(this,arguments));},map:function(callback){return this.pushStack(jQuery.map(this,function(elem,i){return callback.call(elem,i,elem);}));},andSelf:function(){return this.add(this.prevObject);},data:function(key,value){var parts=key.split(".");parts[1]=parts[1]?"."+parts[1]:"";if(value==null){var data=this.triggerHandler("getData"+parts[1]+"!",[parts[0]]);if(data==undefined&&this.length)data=jQuery.data(this[0],key);return data==null&&parts[1]?this.data(parts[0]):data;}else
|
18
|
-
return this.trigger("setData"+parts[1]+"!",[parts[0],value]).each(function(){jQuery.data(this,key,value);});},removeData:function(key){return this.each(function(){jQuery.removeData(this,key);});},domManip:function(args,table,reverse,callback){var clone=this.length>1,elems;return this.each(function(){if(!elems){elems=jQuery.clean(args,this.ownerDocument);if(reverse)elems.reverse();}var obj=this;if(table&&jQuery.nodeName(this,"table")&&jQuery.nodeName(elems[0],"tr"))obj=this.getElementsByTagName("tbody")[0]||this.appendChild(this.ownerDocument.createElement("tbody"));var scripts=jQuery([]);jQuery.each(elems,function(){var elem=clone?jQuery(this).clone(true)[0]:this;if(jQuery.nodeName(elem,"script")){scripts=scripts.add(elem);}else{if(elem.nodeType==1)scripts=scripts.add(jQuery("script",elem).remove());callback.call(obj,elem);}});scripts.each(evalScript);});}};jQuery.prototype.init.prototype=jQuery.prototype;function evalScript(i,elem){if(elem.src)jQuery.ajax({url:elem.src,async:false,dataType:"script"});else
|
19
|
-
jQuery.globalEval(elem.text||elem.textContent||elem.innerHTML||"");if(elem.parentNode)elem.parentNode.removeChild(elem);}jQuery.extend=jQuery.fn.extend=function(){var target=arguments[0]||{},i=1,length=arguments.length,deep=false,options;if(target.constructor==Boolean){deep=target;target=arguments[1]||{};i=2;}if(typeof target!="object"&&typeof target!="function")target={};if(length==1){target=this;i=0;}for(;i<length;i++)if((options=arguments[i])!=null)for(var name in options){if(target===options[name])continue;if(deep&&options[name]&&typeof options[name]=="object"&&target[name]&&!options[name].nodeType)target[name]=jQuery.extend(target[name],options[name]);else if(options[name]!=undefined)target[name]=options[name];}return target;};var expando="jQuery"+(new Date()).getTime(),uuid=0,windowData={};var exclude=/z-?index|font-?weight|opacity|zoom|line-?height/i;jQuery.extend({noConflict:function(deep){window.$=_$;if(deep)window.jQuery=_jQuery;return jQuery;},isFunction:function(fn){return!!fn&&typeof fn!="string"&&!fn.nodeName&&fn.constructor!=Array&&/function/i.test(fn+"");},isXMLDoc:function(elem){return elem.documentElement&&!elem.body||elem.tagName&&elem.ownerDocument&&!elem.ownerDocument.body;},globalEval:function(data){data=jQuery.trim(data);if(data){var head=document.getElementsByTagName("head")[0]||document.documentElement,script=document.createElement("script");script.type="text/javascript";if(jQuery.browser.msie)script.text=data;else
|
20
|
-
script.appendChild(document.createTextNode(data));head.appendChild(script);head.removeChild(script);}},nodeName:function(elem,name){return elem.nodeName&&elem.nodeName.toUpperCase()==name.toUpperCase();},cache:{},data:function(elem,name,data){elem=elem==window?windowData:elem;var id=elem[expando];if(!id)id=elem[expando]=++uuid;if(name&&!jQuery.cache[id])jQuery.cache[id]={};if(data!=undefined)jQuery.cache[id][name]=data;return name?jQuery.cache[id][name]:id;},removeData:function(elem,name){elem=elem==window?windowData:elem;var id=elem[expando];if(name){if(jQuery.cache[id]){delete jQuery.cache[id][name];name="";for(name in jQuery.cache[id])break;if(!name)jQuery.removeData(elem);}}else{try{delete elem[expando];}catch(e){if(elem.removeAttribute)elem.removeAttribute(expando);}delete jQuery.cache[id];}},each:function(object,callback,args){if(args){if(object.length==undefined){for(var name in object)if(callback.apply(object[name],args)===false)break;}else
|
21
|
-
for(var i=0,length=object.length;i<length;i++)if(callback.apply(object[i],args)===false)break;}else{if(object.length==undefined){for(var name in object)if(callback.call(object[name],name,object[name])===false)break;}else
|
22
|
-
for(var i=0,length=object.length,value=object[0];i<length&&callback.call(value,i,value)!==false;value=object[++i]){}}return object;},prop:function(elem,value,type,i,name){if(jQuery.isFunction(value))value=value.call(elem,i);return value&&value.constructor==Number&&type=="curCSS"&&!exclude.test(name)?value+"px":value;},className:{add:function(elem,classNames){jQuery.each((classNames||"").split(/\s+/),function(i,className){if(elem.nodeType==1&&!jQuery.className.has(elem.className,className))elem.className+=(elem.className?" ":"")+className;});},remove:function(elem,classNames){if(elem.nodeType==1)elem.className=classNames!=undefined?jQuery.grep(elem.className.split(/\s+/),function(className){return!jQuery.className.has(classNames,className);}).join(" "):"";},has:function(elem,className){return jQuery.inArray(className,(elem.className||elem).toString().split(/\s+/))>-1;}},swap:function(elem,options,callback){var old={};for(var name in options){old[name]=elem.style[name];elem.style[name]=options[name];}callback.call(elem);for(var name in options)elem.style[name]=old[name];},css:function(elem,name,force){if(name=="width"||name=="height"){var val,props={position:"absolute",visibility:"hidden",display:"block"},which=name=="width"?["Left","Right"]:["Top","Bottom"];function getWH(){val=name=="width"?elem.offsetWidth:elem.offsetHeight;var padding=0,border=0;jQuery.each(which,function(){padding+=parseFloat(jQuery.curCSS(elem,"padding"+this,true))||0;border+=parseFloat(jQuery.curCSS(elem,"border"+this+"Width",true))||0;});val-=Math.round(padding+border);}if(jQuery(elem).is(":visible"))getWH();else
|
23
|
-
jQuery.swap(elem,props,getWH);return Math.max(0,val);}return jQuery.curCSS(elem,name,force);},curCSS:function(elem,name,force){var ret;function color(elem){if(!jQuery.browser.safari)return false;var ret=document.defaultView.getComputedStyle(elem,null);return!ret||ret.getPropertyValue("color")=="";}if(name=="opacity"&&jQuery.browser.msie){ret=jQuery.attr(elem.style,"opacity");return ret==""?"1":ret;}if(jQuery.browser.opera&&name=="display"){var save=elem.style.outline;elem.style.outline="0 solid black";elem.style.outline=save;}if(name.match(/float/i))name=styleFloat;if(!force&&elem.style&&elem.style[name])ret=elem.style[name];else if(document.defaultView&&document.defaultView.getComputedStyle){if(name.match(/float/i))name="float";name=name.replace(/([A-Z])/g,"-$1").toLowerCase();var getComputedStyle=document.defaultView.getComputedStyle(elem,null);if(getComputedStyle&&!color(elem))ret=getComputedStyle.getPropertyValue(name);else{var swap=[],stack=[];for(var a=elem;a&&color(a);a=a.parentNode)stack.unshift(a);for(var i=0;i<stack.length;i++)if(color(stack[i])){swap[i]=stack[i].style.display;stack[i].style.display="block";}ret=name=="display"&&swap[stack.length-1]!=null?"none":(getComputedStyle&&getComputedStyle.getPropertyValue(name))||"";for(var i=0;i<swap.length;i++)if(swap[i]!=null)stack[i].style.display=swap[i];}if(name=="opacity"&&ret=="")ret="1";}else if(elem.currentStyle){var camelCase=name.replace(/\-(\w)/g,function(all,letter){return letter.toUpperCase();});ret=elem.currentStyle[name]||elem.currentStyle[camelCase];if(!/^\d+(px)?$/i.test(ret)&&/^\d/.test(ret)){var style=elem.style.left,runtimeStyle=elem.runtimeStyle.left;elem.runtimeStyle.left=elem.currentStyle.left;elem.style.left=ret||0;ret=elem.style.pixelLeft+"px";elem.style.left=style;elem.runtimeStyle.left=runtimeStyle;}}return ret;},clean:function(elems,context){var ret=[];context=context||document;if(typeof context.createElement=='undefined')context=context.ownerDocument||context[0]&&context[0].ownerDocument||document;jQuery.each(elems,function(i,elem){if(!elem)return;if(elem.constructor==Number)elem=elem.toString();if(typeof elem=="string"){elem=elem.replace(/(<(\w+)[^>]*?)\/>/g,function(all,front,tag){return tag.match(/^(abbr|br|col|img|input|link|meta|param|hr|area|embed)$/i)?all:front+"></"+tag+">";});var tags=jQuery.trim(elem).toLowerCase(),div=context.createElement("div");var wrap=!tags.indexOf("<opt")&&[1,"<select multiple='multiple'>","</select>"]||!tags.indexOf("<leg")&&[1,"<fieldset>","</fieldset>"]||tags.match(/^<(thead|tbody|tfoot|colg|cap)/)&&[1,"<table>","</table>"]||!tags.indexOf("<tr")&&[2,"<table><tbody>","</tbody></table>"]||(!tags.indexOf("<td")||!tags.indexOf("<th"))&&[3,"<table><tbody><tr>","</tr></tbody></table>"]||!tags.indexOf("<col")&&[2,"<table><tbody></tbody><colgroup>","</colgroup></table>"]||jQuery.browser.msie&&[1,"div<div>","</div>"]||[0,"",""];div.innerHTML=wrap[1]+elem+wrap[2];while(wrap[0]--)div=div.lastChild;if(jQuery.browser.msie){var tbody=!tags.indexOf("<table")&&tags.indexOf("<tbody")<0?div.firstChild&&div.firstChild.childNodes:wrap[1]=="<table>"&&tags.indexOf("<tbody")<0?div.childNodes:[];for(var j=tbody.length-1;j>=0;--j)if(jQuery.nodeName(tbody[j],"tbody")&&!tbody[j].childNodes.length)tbody[j].parentNode.removeChild(tbody[j]);if(/^\s/.test(elem))div.insertBefore(context.createTextNode(elem.match(/^\s*/)[0]),div.firstChild);}elem=jQuery.makeArray(div.childNodes);}if(elem.length===0&&(!jQuery.nodeName(elem,"form")&&!jQuery.nodeName(elem,"select")))return;if(elem[0]==undefined||jQuery.nodeName(elem,"form")||elem.options)ret.push(elem);else
|
24
|
-
ret=jQuery.merge(ret,elem);});return ret;},attr:function(elem,name,value){if(!elem||elem.nodeType==3||elem.nodeType==8)return undefined;var fix=jQuery.isXMLDoc(elem)?{}:jQuery.props;if(name=="selected"&&jQuery.browser.safari)elem.parentNode.selectedIndex;if(fix[name]){if(value!=undefined)elem[fix[name]]=value;return elem[fix[name]];}else if(jQuery.browser.msie&&name=="style")return jQuery.attr(elem.style,"cssText",value);else if(value==undefined&&jQuery.browser.msie&&jQuery.nodeName(elem,"form")&&(name=="action"||name=="method"))return elem.getAttributeNode(name).nodeValue;else if(elem.tagName){if(value!=undefined){if(name=="type"&&jQuery.nodeName(elem,"input")&&elem.parentNode)throw"type property can't be changed";elem.setAttribute(name,""+value);}if(jQuery.browser.msie&&/href|src/.test(name)&&!jQuery.isXMLDoc(elem))return elem.getAttribute(name,2);return elem.getAttribute(name);}else{if(name=="opacity"&&jQuery.browser.msie){if(value!=undefined){elem.zoom=1;elem.filter=(elem.filter||"").replace(/alpha\([^)]*\)/,"")+(parseFloat(value).toString()=="NaN"?"":"alpha(opacity="+value*100+")");}return elem.filter&&elem.filter.indexOf("opacity=")>=0?(parseFloat(elem.filter.match(/opacity=([^)]*)/)[1])/100).toString():"";}name=name.replace(/-([a-z])/ig,function(all,letter){return letter.toUpperCase();});if(value!=undefined)elem[name]=value;return elem[name];}},trim:function(text){return(text||"").replace(/^\s+|\s+$/g,"");},makeArray:function(array){var ret=[];if(typeof array!="array")for(var i=0,length=array.length;i<length;i++)ret.push(array[i]);else
|
25
|
-
ret=array.slice(0);return ret;},inArray:function(elem,array){for(var i=0,length=array.length;i<length;i++)if(array[i]==elem)return i;return-1;},merge:function(first,second){if(jQuery.browser.msie){for(var i=0;second[i];i++)if(second[i].nodeType!=8)first.push(second[i]);}else
|
26
|
-
for(var i=0;second[i];i++)first.push(second[i]);return first;},unique:function(array){var ret=[],done={};try{for(var i=0,length=array.length;i<length;i++){var id=jQuery.data(array[i]);if(!done[id]){done[id]=true;ret.push(array[i]);}}}catch(e){ret=array;}return ret;},grep:function(elems,callback,inv){var ret=[];for(var i=0,length=elems.length;i<length;i++)if(!inv&&callback(elems[i],i)||inv&&!callback(elems[i],i))ret.push(elems[i]);return ret;},map:function(elems,callback){var ret=[];for(var i=0,length=elems.length;i<length;i++){var value=callback(elems[i],i);if(value!==null&&value!=undefined){if(value.constructor!=Array)value=[value];ret=ret.concat(value);}}return ret;}});var userAgent=navigator.userAgent.toLowerCase();jQuery.browser={version:(userAgent.match(/.+(?:rv|it|ra|ie)[\/: ]([\d.]+)/)||[])[1],safari:/webkit/.test(userAgent),opera:/opera/.test(userAgent),msie:/msie/.test(userAgent)&&!/opera/.test(userAgent),mozilla:/mozilla/.test(userAgent)&&!/(compatible|webkit)/.test(userAgent)};var styleFloat=jQuery.browser.msie?"styleFloat":"cssFloat";jQuery.extend({boxModel:!jQuery.browser.msie||document.compatMode=="CSS1Compat",props:{"for":"htmlFor","class":"className","float":styleFloat,cssFloat:styleFloat,styleFloat:styleFloat,innerHTML:"innerHTML",className:"className",value:"value",disabled:"disabled",checked:"checked",readonly:"readOnly",selected:"selected",maxlength:"maxLength",selectedIndex:"selectedIndex",defaultValue:"defaultValue",tagName:"tagName",nodeName:"nodeName"}});jQuery.each({parent:function(elem){return elem.parentNode;},parents:function(elem){return jQuery.dir(elem,"parentNode");},next:function(elem){return jQuery.nth(elem,2,"nextSibling");},prev:function(elem){return jQuery.nth(elem,2,"previousSibling");},nextAll:function(elem){return jQuery.dir(elem,"nextSibling");},prevAll:function(elem){return jQuery.dir(elem,"previousSibling");},siblings:function(elem){return jQuery.sibling(elem.parentNode.firstChild,elem);},children:function(elem){return jQuery.sibling(elem.firstChild);},contents:function(elem){return jQuery.nodeName(elem,"iframe")?elem.contentDocument||elem.contentWindow.document:jQuery.makeArray(elem.childNodes);}},function(name,fn){jQuery.fn[name]=function(selector){var ret=jQuery.map(this,fn);if(selector&&typeof selector=="string")ret=jQuery.multiFilter(selector,ret);return this.pushStack(jQuery.unique(ret));};});jQuery.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(name,original){jQuery.fn[name]=function(){var args=arguments;return this.each(function(){for(var i=0,length=args.length;i<length;i++)jQuery(args[i])[original](this);});};});jQuery.each({removeAttr:function(name){jQuery.attr(this,name,"");if(this.nodeType==1)this.removeAttribute(name);},addClass:function(classNames){jQuery.className.add(this,classNames);},removeClass:function(classNames){jQuery.className.remove(this,classNames);},toggleClass:function(classNames){jQuery.className[jQuery.className.has(this,classNames)?"remove":"add"](this,classNames);},remove:function(selector){if(!selector||jQuery.filter(selector,[this]).r.length){jQuery("*",this).add(this).each(function(){jQuery.event.remove(this);jQuery.removeData(this);});if(this.parentNode)this.parentNode.removeChild(this);}},empty:function(){jQuery(">*",this).remove();while(this.firstChild)this.removeChild(this.firstChild);}},function(name,fn){jQuery.fn[name]=function(){return this.each(fn,arguments);};});jQuery.each(["Height","Width"],function(i,name){var type=name.toLowerCase();jQuery.fn[type]=function(size){return this[0]==window?jQuery.browser.opera&&document.body["client"+name]||jQuery.browser.safari&&window["inner"+name]||document.compatMode=="CSS1Compat"&&document.documentElement["client"+name]||document.body["client"+name]:this[0]==document?Math.max(Math.max(document.body["scroll"+name],document.documentElement["scroll"+name]),Math.max(document.body["offset"+name],document.documentElement["offset"+name])):size==undefined?(this.length?jQuery.css(this[0],type):null):this.css(type,size.constructor==String?size:size+"px");};});var chars=jQuery.browser.safari&&parseInt(jQuery.browser.version)<417?"(?:[\\w*_-]|\\\\.)":"(?:[\\w\u0128-\uFFFF*_-]|\\\\.)",quickChild=new RegExp("^>\\s*("+chars+"+)"),quickID=new RegExp("^("+chars+"+)(#)("+chars+"+)"),quickClass=new RegExp("^([#.]?)("+chars+"*)");jQuery.extend({expr:{"":function(a,i,m){return m[2]=="*"||jQuery.nodeName(a,m[2]);},"#":function(a,i,m){return a.getAttribute("id")==m[2];},":":{lt:function(a,i,m){return i<m[3]-0;},gt:function(a,i,m){return i>m[3]-0;},nth:function(a,i,m){return m[3]-0==i;},eq:function(a,i,m){return m[3]-0==i;},first:function(a,i){return i==0;},last:function(a,i,m,r){return i==r.length-1;},even:function(a,i){return i%2==0;},odd:function(a,i){return i%2;},"first-child":function(a){return a.parentNode.getElementsByTagName("*")[0]==a;},"last-child":function(a){return jQuery.nth(a.parentNode.lastChild,1,"previousSibling")==a;},"only-child":function(a){return!jQuery.nth(a.parentNode.lastChild,2,"previousSibling");},parent:function(a){return a.firstChild;},empty:function(a){return!a.firstChild;},contains:function(a,i,m){return(a.textContent||a.innerText||jQuery(a).text()||"").indexOf(m[3])>=0;},visible:function(a){return"hidden"!=a.type&&jQuery.css(a,"display")!="none"&&jQuery.css(a,"visibility")!="hidden";},hidden:function(a){return"hidden"==a.type||jQuery.css(a,"display")=="none"||jQuery.css(a,"visibility")=="hidden";},enabled:function(a){return!a.disabled;},disabled:function(a){return a.disabled;},checked:function(a){return a.checked;},selected:function(a){return a.selected||jQuery.attr(a,"selected");},text:function(a){return"text"==a.type;},radio:function(a){return"radio"==a.type;},checkbox:function(a){return"checkbox"==a.type;},file:function(a){return"file"==a.type;},password:function(a){return"password"==a.type;},submit:function(a){return"submit"==a.type;},image:function(a){return"image"==a.type;},reset:function(a){return"reset"==a.type;},button:function(a){return"button"==a.type||jQuery.nodeName(a,"button");},input:function(a){return/input|select|textarea|button/i.test(a.nodeName);},has:function(a,i,m){return jQuery.find(m[3],a).length;},header:function(a){return/h\d/i.test(a.nodeName);},animated:function(a){return jQuery.grep(jQuery.timers,function(fn){return a==fn.elem;}).length;}}},parse:[/^(\[) *@?([\w-]+) *([!*$^~=]*) *('?"?)(.*?)\4 *\]/,/^(:)([\w-]+)\("?'?(.*?(\(.*?\))?[^(]*?)"?'?\)/,new RegExp("^([:.#]*)("+chars+"+)")],multiFilter:function(expr,elems,not){var old,cur=[];while(expr&&expr!=old){old=expr;var f=jQuery.filter(expr,elems,not);expr=f.t.replace(/^\s*,\s*/,"");cur=not?elems=f.r:jQuery.merge(cur,f.r);}return cur;},find:function(t,context){if(typeof t!="string")return[t];if(context&&context.nodeType!=1&&context.nodeType!=9)return[];context=context||document;var ret=[context],done=[],last,nodeName;while(t&&last!=t){var r=[];last=t;t=jQuery.trim(t);var foundToken=false;var re=quickChild;var m=re.exec(t);if(m){nodeName=m[1].toUpperCase();for(var i=0;ret[i];i++)for(var c=ret[i].firstChild;c;c=c.nextSibling)if(c.nodeType==1&&(nodeName=="*"||c.nodeName.toUpperCase()==nodeName))r.push(c);ret=r;t=t.replace(re,"");if(t.indexOf(" ")==0)continue;foundToken=true;}else{re=/^([>+~])\s*(\w*)/i;if((m=re.exec(t))!=null){r=[];var merge={};nodeName=m[2].toUpperCase();m=m[1];for(var j=0,rl=ret.length;j<rl;j++){var n=m=="~"||m=="+"?ret[j].nextSibling:ret[j].firstChild;for(;n;n=n.nextSibling)if(n.nodeType==1){var id=jQuery.data(n);if(m=="~"&&merge[id])break;if(!nodeName||n.nodeName.toUpperCase()==nodeName){if(m=="~")merge[id]=true;r.push(n);}if(m=="+")break;}}ret=r;t=jQuery.trim(t.replace(re,""));foundToken=true;}}if(t&&!foundToken){if(!t.indexOf(",")){if(context==ret[0])ret.shift();done=jQuery.merge(done,ret);r=ret=[context];t=" "+t.substr(1,t.length);}else{var re2=quickID;var m=re2.exec(t);if(m){m=[0,m[2],m[3],m[1]];}else{re2=quickClass;m=re2.exec(t);}m[2]=m[2].replace(/\\/g,"");var elem=ret[ret.length-1];if(m[1]=="#"&&elem&&elem.getElementById&&!jQuery.isXMLDoc(elem)){var oid=elem.getElementById(m[2]);if((jQuery.browser.msie||jQuery.browser.opera)&&oid&&typeof oid.id=="string"&&oid.id!=m[2])oid=jQuery('[@id="'+m[2]+'"]',elem)[0];ret=r=oid&&(!m[3]||jQuery.nodeName(oid,m[3]))?[oid]:[];}else{for(var i=0;ret[i];i++){var tag=m[1]=="#"&&m[3]?m[3]:m[1]!=""||m[0]==""?"*":m[2];if(tag=="*"&&ret[i].nodeName.toLowerCase()=="object")tag="param";r=jQuery.merge(r,ret[i].getElementsByTagName(tag));}if(m[1]==".")r=jQuery.classFilter(r,m[2]);if(m[1]=="#"){var tmp=[];for(var i=0;r[i];i++)if(r[i].getAttribute("id")==m[2]){tmp=[r[i]];break;}r=tmp;}ret=r;}t=t.replace(re2,"");}}if(t){var val=jQuery.filter(t,r);ret=r=val.r;t=jQuery.trim(val.t);}}if(t)ret=[];if(ret&&context==ret[0])ret.shift();done=jQuery.merge(done,ret);return done;},classFilter:function(r,m,not){m=" "+m+" ";var tmp=[];for(var i=0;r[i];i++){var pass=(" "+r[i].className+" ").indexOf(m)>=0;if(!not&&pass||not&&!pass)tmp.push(r[i]);}return tmp;},filter:function(t,r,not){var last;while(t&&t!=last){last=t;var p=jQuery.parse,m;for(var i=0;p[i];i++){m=p[i].exec(t);if(m){t=t.substring(m[0].length);m[2]=m[2].replace(/\\/g,"");break;}}if(!m)break;if(m[1]==":"&&m[2]=="not")r=isSimple.test(m[3])?jQuery.filter(m[3],r,true).r:jQuery(r).not(m[3]);else if(m[1]==".")r=jQuery.classFilter(r,m[2],not);else if(m[1]=="["){var tmp=[],type=m[3];for(var i=0,rl=r.length;i<rl;i++){var a=r[i],z=a[jQuery.props[m[2]]||m[2]];if(z==null||/href|src|selected/.test(m[2]))z=jQuery.attr(a,m[2])||'';if((type==""&&!!z||type=="="&&z==m[5]||type=="!="&&z!=m[5]||type=="^="&&z&&!z.indexOf(m[5])||type=="$="&&z.substr(z.length-m[5].length)==m[5]||(type=="*="||type=="~=")&&z.indexOf(m[5])>=0)^not)tmp.push(a);}r=tmp;}else if(m[1]==":"&&m[2]=="nth-child"){var merge={},tmp=[],test=/(-?)(\d*)n((?:\+|-)?\d*)/.exec(m[3]=="even"&&"2n"||m[3]=="odd"&&"2n+1"||!/\D/.test(m[3])&&"0n+"+m[3]||m[3]),first=(test[1]+(test[2]||1))-0,last=test[3]-0;for(var i=0,rl=r.length;i<rl;i++){var node=r[i],parentNode=node.parentNode,id=jQuery.data(parentNode);if(!merge[id]){var c=1;for(var n=parentNode.firstChild;n;n=n.nextSibling)if(n.nodeType==1)n.nodeIndex=c++;merge[id]=true;}var add=false;if(first==0){if(node.nodeIndex==last)add=true;}else if((node.nodeIndex-last)%first==0&&(node.nodeIndex-last)/first>=0)add=true;if(add^not)tmp.push(node);}r=tmp;}else{var fn=jQuery.expr[m[1]];if(typeof fn=="object")fn=fn[m[2]];if(typeof fn=="string")fn=eval("false||function(a,i){return "+fn+";}");r=jQuery.grep(r,function(elem,i){return fn(elem,i,m,r);},not);}}return{r:r,t:t};},dir:function(elem,dir){var matched=[];var cur=elem[dir];while(cur&&cur!=document){if(cur.nodeType==1)matched.push(cur);cur=cur[dir];}return matched;},nth:function(cur,result,dir,elem){result=result||1;var num=0;for(;cur;cur=cur[dir])if(cur.nodeType==1&&++num==result)break;return cur;},sibling:function(n,elem){var r=[];for(;n;n=n.nextSibling){if(n.nodeType==1&&(!elem||n!=elem))r.push(n);}return r;}});jQuery.event={add:function(elem,types,handler,data){if(elem.nodeType==3||elem.nodeType==8)return;if(jQuery.browser.msie&&elem.setInterval!=undefined)elem=window;if(!handler.guid)handler.guid=this.guid++;if(data!=undefined){var fn=handler;handler=function(){return fn.apply(this,arguments);};handler.data=data;handler.guid=fn.guid;}var events=jQuery.data(elem,"events")||jQuery.data(elem,"events",{}),handle=jQuery.data(elem,"handle")||jQuery.data(elem,"handle",function(){var val;if(typeof jQuery=="undefined"||jQuery.event.triggered)return val;val=jQuery.event.handle.apply(arguments.callee.elem,arguments);return val;});handle.elem=elem;jQuery.each(types.split(/\s+/),function(index,type){var parts=type.split(".");type=parts[0];handler.type=parts[1];var handlers=events[type];if(!handlers){handlers=events[type]={};if(!jQuery.event.special[type]||jQuery.event.special[type].setup.call(elem)===false){if(elem.addEventListener)elem.addEventListener(type,handle,false);else if(elem.attachEvent)elem.attachEvent("on"+type,handle);}}handlers[handler.guid]=handler;jQuery.event.global[type]=true;});elem=null;},guid:1,global:{},remove:function(elem,types,handler){if(elem.nodeType==3||elem.nodeType==8)return;var events=jQuery.data(elem,"events"),ret,index;if(events){if(types==undefined||(typeof types=="string"&&types.charAt(0)=="."))for(var type in events)this.remove(elem,type+(types||""));else{if(types.type){handler=types.handler;types=types.type;}jQuery.each(types.split(/\s+/),function(index,type){var parts=type.split(".");type=parts[0];if(events[type]){if(handler)delete events[type][handler.guid];else
|
27
|
-
for(handler in events[type])if(!parts[1]||events[type][handler].type==parts[1])delete events[type][handler];for(ret in events[type])break;if(!ret){if(!jQuery.event.special[type]||jQuery.event.special[type].teardown.call(elem)===false){if(elem.removeEventListener)elem.removeEventListener(type,jQuery.data(elem,"handle"),false);else if(elem.detachEvent)elem.detachEvent("on"+type,jQuery.data(elem,"handle"));}ret=null;delete events[type];}}});}for(ret in events)break;if(!ret){var handle=jQuery.data(elem,"handle");if(handle)handle.elem=null;jQuery.removeData(elem,"events");jQuery.removeData(elem,"handle");}}},trigger:function(type,data,elem,donative,extra){data=jQuery.makeArray(data||[]);if(type.indexOf("!")>=0){type=type.slice(0,-1);var exclusive=true;}if(!elem){if(this.global[type])jQuery("*").add([window,document]).trigger(type,data);}else{if(elem.nodeType==3||elem.nodeType==8)return undefined;var val,ret,fn=jQuery.isFunction(elem[type]||null),event=!data[0]||!data[0].preventDefault;if(event)data.unshift(this.fix({type:type,target:elem}));data[0].type=type;if(exclusive)data[0].exclusive=true;if(jQuery.isFunction(jQuery.data(elem,"handle")))val=jQuery.data(elem,"handle").apply(elem,data);if(!fn&&elem["on"+type]&&elem["on"+type].apply(elem,data)===false)val=false;if(event)data.shift();if(extra&&jQuery.isFunction(extra)){ret=extra.apply(elem,val==null?data:data.concat(val));if(ret!==undefined)val=ret;}if(fn&&donative!==false&&val!==false&&!(jQuery.nodeName(elem,'a')&&type=="click")){this.triggered=true;try{elem[type]();}catch(e){}}this.triggered=false;}return val;},handle:function(event){var val;event=jQuery.event.fix(event||window.event||{});var parts=event.type.split(".");event.type=parts[0];var handlers=jQuery.data(this,"events")&&jQuery.data(this,"events")[event.type],args=Array.prototype.slice.call(arguments,1);args.unshift(event);for(var j in handlers){var handler=handlers[j];args[0].handler=handler;args[0].data=handler.data;if(!parts[1]&&!event.exclusive||handler.type==parts[1]){var ret=handler.apply(this,args);if(val!==false)val=ret;if(ret===false){event.preventDefault();event.stopPropagation();}}}if(jQuery.browser.msie)event.target=event.preventDefault=event.stopPropagation=event.handler=event.data=null;return val;},fix:function(event){var originalEvent=event;event=jQuery.extend({},originalEvent);event.preventDefault=function(){if(originalEvent.preventDefault)originalEvent.preventDefault();originalEvent.returnValue=false;};event.stopPropagation=function(){if(originalEvent.stopPropagation)originalEvent.stopPropagation();originalEvent.cancelBubble=true;};if(!event.target)event.target=event.srcElement||document;if(event.target.nodeType==3)event.target=originalEvent.target.parentNode;if(!event.relatedTarget&&event.fromElement)event.relatedTarget=event.fromElement==event.target?event.toElement:event.fromElement;if(event.pageX==null&&event.clientX!=null){var doc=document.documentElement,body=document.body;event.pageX=event.clientX+(doc&&doc.scrollLeft||body&&body.scrollLeft||0)-(doc.clientLeft||0);event.pageY=event.clientY+(doc&&doc.scrollTop||body&&body.scrollTop||0)-(doc.clientTop||0);}if(!event.which&&((event.charCode||event.charCode===0)?event.charCode:event.keyCode))event.which=event.charCode||event.keyCode;if(!event.metaKey&&event.ctrlKey)event.metaKey=event.ctrlKey;if(!event.which&&event.button)event.which=(event.button&1?1:(event.button&2?3:(event.button&4?2:0)));return event;},special:{ready:{setup:function(){bindReady();return;},teardown:function(){return;}},mouseenter:{setup:function(){if(jQuery.browser.msie)return false;jQuery(this).bind("mouseover",jQuery.event.special.mouseenter.handler);return true;},teardown:function(){if(jQuery.browser.msie)return false;jQuery(this).unbind("mouseover",jQuery.event.special.mouseenter.handler);return true;},handler:function(event){if(withinElement(event,this))return true;arguments[0].type="mouseenter";return jQuery.event.handle.apply(this,arguments);}},mouseleave:{setup:function(){if(jQuery.browser.msie)return false;jQuery(this).bind("mouseout",jQuery.event.special.mouseleave.handler);return true;},teardown:function(){if(jQuery.browser.msie)return false;jQuery(this).unbind("mouseout",jQuery.event.special.mouseleave.handler);return true;},handler:function(event){if(withinElement(event,this))return true;arguments[0].type="mouseleave";return jQuery.event.handle.apply(this,arguments);}}}};jQuery.fn.extend({bind:function(type,data,fn){return type=="unload"?this.one(type,data,fn):this.each(function(){jQuery.event.add(this,type,fn||data,fn&&data);});},one:function(type,data,fn){return this.each(function(){jQuery.event.add(this,type,function(event){jQuery(this).unbind(event);return(fn||data).apply(this,arguments);},fn&&data);});},unbind:function(type,fn){return this.each(function(){jQuery.event.remove(this,type,fn);});},trigger:function(type,data,fn){return this.each(function(){jQuery.event.trigger(type,data,this,true,fn);});},triggerHandler:function(type,data,fn){if(this[0])return jQuery.event.trigger(type,data,this[0],false,fn);return undefined;},toggle:function(){var args=arguments;return this.click(function(event){this.lastToggle=0==this.lastToggle?1:0;event.preventDefault();return args[this.lastToggle].apply(this,arguments)||false;});},hover:function(fnOver,fnOut){return this.bind('mouseenter',fnOver).bind('mouseleave',fnOut);},ready:function(fn){bindReady();if(jQuery.isReady)fn.call(document,jQuery);else
|
28
|
-
jQuery.readyList.push(function(){return fn.call(this,jQuery);});return this;}});jQuery.extend({isReady:false,readyList:[],ready:function(){if(!jQuery.isReady){jQuery.isReady=true;if(jQuery.readyList){jQuery.each(jQuery.readyList,function(){this.apply(document);});jQuery.readyList=null;}jQuery(document).triggerHandler("ready");}}});var readyBound=false;function bindReady(){if(readyBound)return;readyBound=true;if(document.addEventListener&&!jQuery.browser.opera)document.addEventListener("DOMContentLoaded",jQuery.ready,false);if(jQuery.browser.msie&&window==top)(function(){if(jQuery.isReady)return;try{document.documentElement.doScroll("left");}catch(error){setTimeout(arguments.callee,0);return;}jQuery.ready();})();if(jQuery.browser.opera)document.addEventListener("DOMContentLoaded",function(){if(jQuery.isReady)return;for(var i=0;i<document.styleSheets.length;i++)if(document.styleSheets[i].disabled){setTimeout(arguments.callee,0);return;}jQuery.ready();},false);if(jQuery.browser.safari){var numStyles;(function(){if(jQuery.isReady)return;if(document.readyState!="loaded"&&document.readyState!="complete"){setTimeout(arguments.callee,0);return;}if(numStyles===undefined)numStyles=jQuery("style, link[rel=stylesheet]").length;if(document.styleSheets.length!=numStyles){setTimeout(arguments.callee,0);return;}jQuery.ready();})();}jQuery.event.add(window,"load",jQuery.ready);}jQuery.each(("blur,focus,load,resize,scroll,unload,click,dblclick,"+"mousedown,mouseup,mousemove,mouseover,mouseout,change,select,"+"submit,keydown,keypress,keyup,error").split(","),function(i,name){jQuery.fn[name]=function(fn){return fn?this.bind(name,fn):this.trigger(name);};});var withinElement=function(event,elem){var parent=event.relatedTarget;while(parent&&parent!=elem)try{parent=parent.parentNode;}catch(error){parent=elem;}return parent==elem;};jQuery(window).bind("unload",function(){jQuery("*").add(document).unbind();});jQuery.fn.extend({load:function(url,params,callback){if(jQuery.isFunction(url))return this.bind("load",url);var off=url.indexOf(" ");if(off>=0){var selector=url.slice(off,url.length);url=url.slice(0,off);}callback=callback||function(){};var type="GET";if(params)if(jQuery.isFunction(params)){callback=params;params=null;}else{params=jQuery.param(params);type="POST";}var self=this;jQuery.ajax({url:url,type:type,dataType:"html",data:params,complete:function(res,status){if(status=="success"||status=="notmodified")self.html(selector?jQuery("<div/>").append(res.responseText.replace(/<script(.|\s)*?\/script>/g,"")).find(selector):res.responseText);self.each(callback,[res.responseText,status,res]);}});return this;},serialize:function(){return jQuery.param(this.serializeArray());},serializeArray:function(){return this.map(function(){return jQuery.nodeName(this,"form")?jQuery.makeArray(this.elements):this;}).filter(function(){return this.name&&!this.disabled&&(this.checked||/select|textarea/i.test(this.nodeName)||/text|hidden|password/i.test(this.type));}).map(function(i,elem){var val=jQuery(this).val();return val==null?null:val.constructor==Array?jQuery.map(val,function(val,i){return{name:elem.name,value:val};}):{name:elem.name,value:val};}).get();}});jQuery.each("ajaxStart,ajaxStop,ajaxComplete,ajaxError,ajaxSuccess,ajaxSend".split(","),function(i,o){jQuery.fn[o]=function(f){return this.bind(o,f);};});var jsc=(new Date).getTime();jQuery.extend({get:function(url,data,callback,type){if(jQuery.isFunction(data)){callback=data;data=null;}return jQuery.ajax({type:"GET",url:url,data:data,success:callback,dataType:type});},getScript:function(url,callback){return jQuery.get(url,null,callback,"script");},getJSON:function(url,data,callback){return jQuery.get(url,data,callback,"json");},post:function(url,data,callback,type){if(jQuery.isFunction(data)){callback=data;data={};}return jQuery.ajax({type:"POST",url:url,data:data,success:callback,dataType:type});},ajaxSetup:function(settings){jQuery.extend(jQuery.ajaxSettings,settings);},ajaxSettings:{global:true,type:"GET",timeout:0,contentType:"application/x-www-form-urlencoded",processData:true,async:true,data:null,username:null,password:null,accepts:{xml:"application/xml, text/xml",html:"text/html",script:"text/javascript, application/javascript",json:"application/json, text/javascript",text:"text/plain",_default:"*/*"}},lastModified:{},ajax:function(s){var jsonp,jsre=/=\?(&|$)/g,status,data;s=jQuery.extend(true,s,jQuery.extend(true,{},jQuery.ajaxSettings,s));if(s.data&&s.processData&&typeof s.data!="string")s.data=jQuery.param(s.data);if(s.dataType=="jsonp"){if(s.type.toLowerCase()=="get"){if(!s.url.match(jsre))s.url+=(s.url.match(/\?/)?"&":"?")+(s.jsonp||"callback")+"=?";}else if(!s.data||!s.data.match(jsre))s.data=(s.data?s.data+"&":"")+(s.jsonp||"callback")+"=?";s.dataType="json";}if(s.dataType=="json"&&(s.data&&s.data.match(jsre)||s.url.match(jsre))){jsonp="jsonp"+jsc++;if(s.data)s.data=(s.data+"").replace(jsre,"="+jsonp+"$1");s.url=s.url.replace(jsre,"="+jsonp+"$1");s.dataType="script";window[jsonp]=function(tmp){data=tmp;success();complete();window[jsonp]=undefined;try{delete window[jsonp];}catch(e){}if(head)head.removeChild(script);};}if(s.dataType=="script"&&s.cache==null)s.cache=false;if(s.cache===false&&s.type.toLowerCase()=="get"){var ts=(new Date()).getTime();var ret=s.url.replace(/(\?|&)_=.*?(&|$)/,"$1_="+ts+"$2");s.url=ret+((ret==s.url)?(s.url.match(/\?/)?"&":"?")+"_="+ts:"");}if(s.data&&s.type.toLowerCase()=="get"){s.url+=(s.url.match(/\?/)?"&":"?")+s.data;s.data=null;}if(s.global&&!jQuery.active++)jQuery.event.trigger("ajaxStart");if((!s.url.indexOf("http")||!s.url.indexOf("//"))&&s.dataType=="script"&&s.type.toLowerCase()=="get"){var head=document.getElementsByTagName("head")[0];var script=document.createElement("script");script.src=s.url;if(s.scriptCharset)script.charset=s.scriptCharset;if(!jsonp){var done=false;script.onload=script.onreadystatechange=function(){if(!done&&(!this.readyState||this.readyState=="loaded"||this.readyState=="complete")){done=true;success();complete();head.removeChild(script);}};}head.appendChild(script);return undefined;}var requestDone=false;var xml=window.ActiveXObject?new ActiveXObject("Microsoft.XMLHTTP"):new XMLHttpRequest();xml.open(s.type,s.url,s.async,s.username,s.password);try{if(s.data)xml.setRequestHeader("Content-Type",s.contentType);if(s.ifModified)xml.setRequestHeader("If-Modified-Since",jQuery.lastModified[s.url]||"Thu, 01 Jan 1970 00:00:00 GMT");xml.setRequestHeader("X-Requested-With","XMLHttpRequest");xml.setRequestHeader("Accept",s.dataType&&s.accepts[s.dataType]?s.accepts[s.dataType]+", */*":s.accepts._default);}catch(e){}if(s.beforeSend)s.beforeSend(xml);if(s.global)jQuery.event.trigger("ajaxSend",[xml,s]);var onreadystatechange=function(isTimeout){if(!requestDone&&xml&&(xml.readyState==4||isTimeout=="timeout")){requestDone=true;if(ival){clearInterval(ival);ival=null;}status=isTimeout=="timeout"&&"timeout"||!jQuery.httpSuccess(xml)&&"error"||s.ifModified&&jQuery.httpNotModified(xml,s.url)&&"notmodified"||"success";if(status=="success"){try{data=jQuery.httpData(xml,s.dataType);}catch(e){status="parsererror";}}if(status=="success"){var modRes;try{modRes=xml.getResponseHeader("Last-Modified");}catch(e){}if(s.ifModified&&modRes)jQuery.lastModified[s.url]=modRes;if(!jsonp)success();}else
|
29
|
-
jQuery.handleError(s,xml,status);complete();if(s.async)xml=null;}};if(s.async){var ival=setInterval(onreadystatechange,13);if(s.timeout>0)setTimeout(function(){if(xml){xml.abort();if(!requestDone)onreadystatechange("timeout");}},s.timeout);}try{xml.send(s.data);}catch(e){jQuery.handleError(s,xml,null,e);}if(!s.async)onreadystatechange();function success(){if(s.success)s.success(data,status);if(s.global)jQuery.event.trigger("ajaxSuccess",[xml,s]);}function complete(){if(s.complete)s.complete(xml,status);if(s.global)jQuery.event.trigger("ajaxComplete",[xml,s]);if(s.global&&!--jQuery.active)jQuery.event.trigger("ajaxStop");}return xml;},handleError:function(s,xml,status,e){if(s.error)s.error(xml,status,e);if(s.global)jQuery.event.trigger("ajaxError",[xml,s,e]);},active:0,httpSuccess:function(r){try{return!r.status&&location.protocol=="file:"||(r.status>=200&&r.status<300)||r.status==304||r.status==1223||jQuery.browser.safari&&r.status==undefined;}catch(e){}return false;},httpNotModified:function(xml,url){try{var xmlRes=xml.getResponseHeader("Last-Modified");return xml.status==304||xmlRes==jQuery.lastModified[url]||jQuery.browser.safari&&xml.status==undefined;}catch(e){}return false;},httpData:function(r,type){var ct=r.getResponseHeader("content-type");var xml=type=="xml"||!type&&ct&&ct.indexOf("xml")>=0;var data=xml?r.responseXML:r.responseText;if(xml&&data.documentElement.tagName=="parsererror")throw"parsererror";if(type=="script")jQuery.globalEval(data);if(type=="json")data=eval("("+data+")");return data;},param:function(a){var s=[];if(a.constructor==Array||a.jquery)jQuery.each(a,function(){s.push(encodeURIComponent(this.name)+"="+encodeURIComponent(this.value));});else
|
30
|
-
for(var j in a)if(a[j]&&a[j].constructor==Array)jQuery.each(a[j],function(){s.push(encodeURIComponent(j)+"="+encodeURIComponent(this));});else
|
31
|
-
s.push(encodeURIComponent(j)+"="+encodeURIComponent(a[j]));return s.join("&").replace(/%20/g,"+");}});jQuery.fn.extend({show:function(speed,callback){return speed?this.animate({height:"show",width:"show",opacity:"show"},speed,callback):this.filter(":hidden").each(function(){this.style.display=this.oldblock||"";if(jQuery.css(this,"display")=="none"){var elem=jQuery("<"+this.tagName+" />").appendTo("body");this.style.display=elem.css("display");if(this.style.display=="none")this.style.display="block";elem.remove();}}).end();},hide:function(speed,callback){return speed?this.animate({height:"hide",width:"hide",opacity:"hide"},speed,callback):this.filter(":visible").each(function(){this.oldblock=this.oldblock||jQuery.css(this,"display");this.style.display="none";}).end();},_toggle:jQuery.fn.toggle,toggle:function(fn,fn2){return jQuery.isFunction(fn)&&jQuery.isFunction(fn2)?this._toggle(fn,fn2):fn?this.animate({height:"toggle",width:"toggle",opacity:"toggle"},fn,fn2):this.each(function(){jQuery(this)[jQuery(this).is(":hidden")?"show":"hide"]();});},slideDown:function(speed,callback){return this.animate({height:"show"},speed,callback);},slideUp:function(speed,callback){return this.animate({height:"hide"},speed,callback);},slideToggle:function(speed,callback){return this.animate({height:"toggle"},speed,callback);},fadeIn:function(speed,callback){return this.animate({opacity:"show"},speed,callback);},fadeOut:function(speed,callback){return this.animate({opacity:"hide"},speed,callback);},fadeTo:function(speed,to,callback){return this.animate({opacity:to},speed,callback);},animate:function(prop,speed,easing,callback){var optall=jQuery.speed(speed,easing,callback);return this[optall.queue===false?"each":"queue"](function(){if(this.nodeType!=1)return false;var opt=jQuery.extend({},optall);var hidden=jQuery(this).is(":hidden"),self=this;for(var p in prop){if(prop[p]=="hide"&&hidden||prop[p]=="show"&&!hidden)return jQuery.isFunction(opt.complete)&&opt.complete.apply(this);if(p=="height"||p=="width"){opt.display=jQuery.css(this,"display");opt.overflow=this.style.overflow;}}if(opt.overflow!=null)this.style.overflow="hidden";opt.curAnim=jQuery.extend({},prop);jQuery.each(prop,function(name,val){var e=new jQuery.fx(self,opt,name);if(/toggle|show|hide/.test(val))e[val=="toggle"?hidden?"show":"hide":val](prop);else{var parts=val.toString().match(/^([+-]=)?([\d+-.]+)(.*)$/),start=e.cur(true)||0;if(parts){var end=parseFloat(parts[2]),unit=parts[3]||"px";if(unit!="px"){self.style[name]=(end||1)+unit;start=((end||1)/e.cur(true))*start;self.style[name]=start+unit;}if(parts[1])end=((parts[1]=="-="?-1:1)*end)+start;e.custom(start,end,unit);}else
|
32
|
-
e.custom(start,val,"");}});return true;});},queue:function(type,fn){if(jQuery.isFunction(type)||(type&&type.constructor==Array)){fn=type;type="fx";}if(!type||(typeof type=="string"&&!fn))return queue(this[0],type);return this.each(function(){if(fn.constructor==Array)queue(this,type,fn);else{queue(this,type).push(fn);if(queue(this,type).length==1)fn.apply(this);}});},stop:function(clearQueue,gotoEnd){var timers=jQuery.timers;if(clearQueue)this.queue([]);this.each(function(){for(var i=timers.length-1;i>=0;i--)if(timers[i].elem==this){if(gotoEnd)timers[i](true);timers.splice(i,1);}});if(!gotoEnd)this.dequeue();return this;}});var queue=function(elem,type,array){if(!elem)return undefined;type=type||"fx";var q=jQuery.data(elem,type+"queue");if(!q||array)q=jQuery.data(elem,type+"queue",array?jQuery.makeArray(array):[]);return q;};jQuery.fn.dequeue=function(type){type=type||"fx";return this.each(function(){var q=queue(this,type);q.shift();if(q.length)q[0].apply(this);});};jQuery.extend({speed:function(speed,easing,fn){var opt=speed&&speed.constructor==Object?speed:{complete:fn||!fn&&easing||jQuery.isFunction(speed)&&speed,duration:speed,easing:fn&&easing||easing&&easing.constructor!=Function&&easing};opt.duration=(opt.duration&&opt.duration.constructor==Number?opt.duration:{slow:600,fast:200}[opt.duration])||400;opt.old=opt.complete;opt.complete=function(){if(opt.queue!==false)jQuery(this).dequeue();if(jQuery.isFunction(opt.old))opt.old.apply(this);};return opt;},easing:{linear:function(p,n,firstNum,diff){return firstNum+diff*p;},swing:function(p,n,firstNum,diff){return((-Math.cos(p*Math.PI)/2)+0.5)*diff+firstNum;}},timers:[],timerId:null,fx:function(elem,options,prop){this.options=options;this.elem=elem;this.prop=prop;if(!options.orig)options.orig={};}});jQuery.fx.prototype={update:function(){if(this.options.step)this.options.step.apply(this.elem,[this.now,this]);(jQuery.fx.step[this.prop]||jQuery.fx.step._default)(this);if(this.prop=="height"||this.prop=="width")this.elem.style.display="block";},cur:function(force){if(this.elem[this.prop]!=null&&this.elem.style[this.prop]==null)return this.elem[this.prop];var r=parseFloat(jQuery.css(this.elem,this.prop,force));return r&&r>-10000?r:parseFloat(jQuery.curCSS(this.elem,this.prop))||0;},custom:function(from,to,unit){this.startTime=(new Date()).getTime();this.start=from;this.end=to;this.unit=unit||this.unit||"px";this.now=this.start;this.pos=this.state=0;this.update();var self=this;function t(gotoEnd){return self.step(gotoEnd);}t.elem=this.elem;jQuery.timers.push(t);if(jQuery.timerId==null){jQuery.timerId=setInterval(function(){var timers=jQuery.timers;for(var i=0;i<timers.length;i++)if(!timers[i]())timers.splice(i--,1);if(!timers.length){clearInterval(jQuery.timerId);jQuery.timerId=null;}},13);}},show:function(){this.options.orig[this.prop]=jQuery.attr(this.elem.style,this.prop);this.options.show=true;this.custom(0,this.cur());if(this.prop=="width"||this.prop=="height")this.elem.style[this.prop]="1px";jQuery(this.elem).show();},hide:function(){this.options.orig[this.prop]=jQuery.attr(this.elem.style,this.prop);this.options.hide=true;this.custom(this.cur(),0);},step:function(gotoEnd){var t=(new Date()).getTime();if(gotoEnd||t>this.options.duration+this.startTime){this.now=this.end;this.pos=this.state=1;this.update();this.options.curAnim[this.prop]=true;var done=true;for(var i in this.options.curAnim)if(this.options.curAnim[i]!==true)done=false;if(done){if(this.options.display!=null){this.elem.style.overflow=this.options.overflow;this.elem.style.display=this.options.display;if(jQuery.css(this.elem,"display")=="none")this.elem.style.display="block";}if(this.options.hide)this.elem.style.display="none";if(this.options.hide||this.options.show)for(var p in this.options.curAnim)jQuery.attr(this.elem.style,p,this.options.orig[p]);}if(done&&jQuery.isFunction(this.options.complete))this.options.complete.apply(this.elem);return false;}else{var n=t-this.startTime;this.state=n/this.options.duration;this.pos=jQuery.easing[this.options.easing||(jQuery.easing.swing?"swing":"linear")](this.state,n,0,1,this.options.duration);this.now=this.start+((this.end-this.start)*this.pos);this.update();}return true;}};jQuery.fx.step={scrollLeft:function(fx){fx.elem.scrollLeft=fx.now;},scrollTop:function(fx){fx.elem.scrollTop=fx.now;},opacity:function(fx){jQuery.attr(fx.elem.style,"opacity",fx.now);},_default:function(fx){fx.elem.style[fx.prop]=fx.now+fx.unit;}};jQuery.fn.offset=function(){var left=0,top=0,elem=this[0],results;if(elem)with(jQuery.browser){var parent=elem.parentNode,offsetChild=elem,offsetParent=elem.offsetParent,doc=elem.ownerDocument,safari2=safari&&parseInt(version)<522&&!/adobeair/i.test(userAgent),fixed=jQuery.css(elem,"position")=="fixed";if(elem.getBoundingClientRect){var box=elem.getBoundingClientRect();add(box.left+Math.max(doc.documentElement.scrollLeft,doc.body.scrollLeft),box.top+Math.max(doc.documentElement.scrollTop,doc.body.scrollTop));add(-doc.documentElement.clientLeft,-doc.documentElement.clientTop);}else{add(elem.offsetLeft,elem.offsetTop);while(offsetParent){add(offsetParent.offsetLeft,offsetParent.offsetTop);if(mozilla&&!/^t(able|d|h)$/i.test(offsetParent.tagName)||safari&&!safari2)border(offsetParent);if(!fixed&&jQuery.css(offsetParent,"position")=="fixed")fixed=true;offsetChild=/^body$/i.test(offsetParent.tagName)?offsetChild:offsetParent;offsetParent=offsetParent.offsetParent;}while(parent&&parent.tagName&&!/^body|html$/i.test(parent.tagName)){if(!/^inline|table.*$/i.test(jQuery.css(parent,"display")))add(-parent.scrollLeft,-parent.scrollTop);if(mozilla&&jQuery.css(parent,"overflow")!="visible")border(parent);parent=parent.parentNode;}if((safari2&&(fixed||jQuery.css(offsetChild,"position")=="absolute"))||(mozilla&&jQuery.css(offsetChild,"position")!="absolute"))add(-doc.body.offsetLeft,-doc.body.offsetTop);if(fixed)add(Math.max(doc.documentElement.scrollLeft,doc.body.scrollLeft),Math.max(doc.documentElement.scrollTop,doc.body.scrollTop));}results={top:top,left:left};}function border(elem){add(jQuery.curCSS(elem,"borderLeftWidth",true),jQuery.curCSS(elem,"borderTopWidth",true));}function add(l,t){left+=parseInt(l)||0;top+=parseInt(t)||0;}return results;};})();
|
data/lib/subtrac/trac-plugins/estimationtoolsplugin/estimationtools/htdocs/jquery.jeditable.js
DELETED
@@ -1,409 +0,0 @@
|
|
1
|
-
/*
|
2
|
-
* Jeditable - jQuery in place edit plugin
|
3
|
-
*
|
4
|
-
* Copyright (c) 2006-2008 Mika Tuupola, Dylan Verheul
|
5
|
-
*
|
6
|
-
* Licensed under the MIT license:
|
7
|
-
* http://www.opensource.org/licenses/mit-license.php
|
8
|
-
*
|
9
|
-
* Project home:
|
10
|
-
* http://www.appelsiini.net/projects/jeditable
|
11
|
-
*
|
12
|
-
* Revision: $Id: jquery.jeditable.js 324 2008-02-28 09:50:25Z tuupola $
|
13
|
-
*
|
14
|
-
*/
|
15
|
-
|
16
|
-
/**
|
17
|
-
* Based on editable by Dylan Verheul <dylan@dyve.net>
|
18
|
-
* http://www.dyve.net/jquery/?editable
|
19
|
-
*
|
20
|
-
* Version 1.5.6
|
21
|
-
*
|
22
|
-
* @name Jeditable
|
23
|
-
* @type jQuery
|
24
|
-
* @param String target POST URL or function name to send edited content
|
25
|
-
* @param Hash options additional options
|
26
|
-
* @param String options[name] POST parameter name of edited content
|
27
|
-
* @param String options[id] POST parameter name of edited div id
|
28
|
-
* @param Hash options[submitdata] Extra parameters to send when submitting edited content.
|
29
|
-
* @param String options[type] text, textarea or select
|
30
|
-
* @param Integer options[rows] number of rows if using textarea
|
31
|
-
* @param Integer options[cols] number of columns if using textarea
|
32
|
-
* @param Mixed options[height] 'auto', 'none' or height in pixels
|
33
|
-
* @param Mixed options[width] 'auto', 'none' or width in pixels
|
34
|
-
* @param String options[loadurl] URL to fetch external content before editing
|
35
|
-
* @param String options[loadtype] Request type for load url. Should be GET or POST.
|
36
|
-
* @param String options[loadtext] Text to display while loading external content.
|
37
|
-
* @param Hash options[loaddata] Extra parameters to pass when fetching content before editing.
|
38
|
-
* @param String options[data] Or content given as paramameter.
|
39
|
-
* @param String options[indicator] indicator html to show when saving
|
40
|
-
* @param String options[tooltip] optional tooltip text via title attribute
|
41
|
-
* @param String options[event] jQuery event such as 'click' of 'dblclick'
|
42
|
-
* @param String options[onblur] 'cancel', 'submit' or 'ignore'
|
43
|
-
* @param String options[submit] submit button value, empty means no button
|
44
|
-
* @param String options[cancel] cancel button value, empty means no button
|
45
|
-
* @param String options[cssclass] CSS class to apply to input form. 'inherit' to copy from parent.
|
46
|
-
* @param String options[style] Style to apply to input form 'inherit' to copy from parent.
|
47
|
-
* @param String options[select] true or false, when true text is highlighted
|
48
|
-
* @param String options[placeholder] Placeholder text or html to insert when element is empty.
|
49
|
-
*
|
50
|
-
*/
|
51
|
-
|
52
|
-
(function($) {
|
53
|
-
|
54
|
-
$.fn.editable = function(target, options) {
|
55
|
-
|
56
|
-
var settings = {
|
57
|
-
target : target,
|
58
|
-
name : 'value',
|
59
|
-
id : 'id',
|
60
|
-
type : 'text',
|
61
|
-
width : 'auto',
|
62
|
-
height : 'auto',
|
63
|
-
event : 'click',
|
64
|
-
onblur : 'cancel',
|
65
|
-
loadtype : 'GET',
|
66
|
-
loadtext : 'Loading...',
|
67
|
-
placeholder: 'Click to edit',
|
68
|
-
loaddata : {},
|
69
|
-
submitdata : {}
|
70
|
-
};
|
71
|
-
|
72
|
-
if(options) {
|
73
|
-
$.extend(settings, options);
|
74
|
-
}
|
75
|
-
|
76
|
-
/* setup some functions */
|
77
|
-
var plugin = $.editable.types[settings.type].plugin || function() { };
|
78
|
-
var submit = $.editable.types[settings.type].submit || function() { };
|
79
|
-
var buttons = $.editable.types[settings.type].buttons
|
80
|
-
|| $.editable.types['defaults'].buttons;
|
81
|
-
var content = $.editable.types[settings.type].content
|
82
|
-
|| $.editable.types['defaults'].content;
|
83
|
-
var element = $.editable.types[settings.type].element
|
84
|
-
|| $.editable.types['defaults'].element;
|
85
|
-
var callback = settings.callback || function() { };
|
86
|
-
|
87
|
-
/* add custom event if it does not exist */
|
88
|
-
if (!$.isFunction($(this)[settings.event])) {
|
89
|
-
$.fn[settings.event] = function(fn){
|
90
|
-
return fn ? this.bind(settings.event, fn) : this.trigger(settings.event);
|
91
|
-
}
|
92
|
-
}
|
93
|
-
|
94
|
-
$(this).attr('title', settings.tooltip);
|
95
|
-
|
96
|
-
settings.autowidth = 'auto' == settings.width;
|
97
|
-
settings.autoheight = 'auto' == settings.height;
|
98
|
-
|
99
|
-
return this.each(function() {
|
100
|
-
|
101
|
-
/* if element is empty add something clickable (if requested) */
|
102
|
-
if (!$.trim($(this).html())) {
|
103
|
-
$(this).html(settings.placeholder);
|
104
|
-
}
|
105
|
-
|
106
|
-
$(this)[settings.event](function(e) {
|
107
|
-
|
108
|
-
/* save this to self because this changes when scope changes */
|
109
|
-
var self = this;
|
110
|
-
|
111
|
-
/* prevent throwing an exeption if edit field is clicked again */
|
112
|
-
if (self.editing) {
|
113
|
-
return;
|
114
|
-
}
|
115
|
-
|
116
|
-
/* figure out how wide and tall we are, visibility trick */
|
117
|
-
/* is workaround for http://dev.jquery.com/ticket/2190 */
|
118
|
-
$(self).css("visibility", "hidden");
|
119
|
-
if (settings.width != 'none') {
|
120
|
-
settings.width =
|
121
|
-
settings.autowidth ? $(self).width() : settings.width;
|
122
|
-
}
|
123
|
-
if (settings.height != 'none') {
|
124
|
-
settings.height =
|
125
|
-
settings.autoheight ? $(self).height() : settings.height;
|
126
|
-
}
|
127
|
-
$(this).css("visibility", "");
|
128
|
-
|
129
|
-
|
130
|
-
/* remove placeholder text, replace is here because of IE */
|
131
|
-
if ($(this).html().toLowerCase().replace(/;/, '') ==
|
132
|
-
settings.placeholder.toLowerCase().replace(/;/, '')) {
|
133
|
-
$(this).html('');
|
134
|
-
}
|
135
|
-
|
136
|
-
self.editing = true;
|
137
|
-
self.revert = $(self).html();
|
138
|
-
$(self).html('');
|
139
|
-
|
140
|
-
/* create the form object */
|
141
|
-
var form = $('<form/>');
|
142
|
-
|
143
|
-
/* apply css or style or both */
|
144
|
-
if (settings.cssclass) {
|
145
|
-
if ('inherit' == settings.cssclass) {
|
146
|
-
form.attr('class', $(self).attr('class'));
|
147
|
-
} else {
|
148
|
-
form.attr('class', settings.cssclass);
|
149
|
-
}
|
150
|
-
}
|
151
|
-
|
152
|
-
if (settings.style) {
|
153
|
-
if ('inherit' == settings.style) {
|
154
|
-
form.attr('style', $(self).attr('style'));
|
155
|
-
/* IE needs the second line or display wont be inherited */
|
156
|
-
form.css('display', $(self).css('display'));
|
157
|
-
} else {
|
158
|
-
form.attr('style', settings.style);
|
159
|
-
}
|
160
|
-
}
|
161
|
-
|
162
|
-
/* add main input element to form and store it in input */
|
163
|
-
var input = element.apply(form, [settings, self]);
|
164
|
-
|
165
|
-
/* set input content via POST, GET, given data or existing value */
|
166
|
-
var input_content;
|
167
|
-
|
168
|
-
if (settings.loadurl) {
|
169
|
-
var t = setTimeout(function() {
|
170
|
-
input.disabled = true;
|
171
|
-
content.apply(form, [settings.loadtext, settings, self]);
|
172
|
-
}, 100);
|
173
|
-
|
174
|
-
var loaddata = {};
|
175
|
-
loaddata[settings.id] = self.id;
|
176
|
-
if ($.isFunction(settings.loaddata)) {
|
177
|
-
$.extend(loaddata, settings.loaddata.apply(self, [self.revert, settings]));
|
178
|
-
} else {
|
179
|
-
$.extend(loaddata, settings.loaddata);
|
180
|
-
}
|
181
|
-
$.ajax({
|
182
|
-
type : settings.loadtype,
|
183
|
-
url : settings.loadurl,
|
184
|
-
data : loaddata,
|
185
|
-
async : false,
|
186
|
-
success: function(result) {
|
187
|
-
window.clearTimeout(t);
|
188
|
-
input_content = result;
|
189
|
-
input.disabled = false;
|
190
|
-
}
|
191
|
-
});
|
192
|
-
} else if (settings.data) {
|
193
|
-
input_content = settings.data;
|
194
|
-
if ($.isFunction(settings.data)) {
|
195
|
-
input_content = settings.data.apply(self, [self.revert, settings]);
|
196
|
-
}
|
197
|
-
} else {
|
198
|
-
input_content = self.revert;
|
199
|
-
}
|
200
|
-
content.apply(form, [input_content, settings, self]);
|
201
|
-
|
202
|
-
input.attr('name', settings.name);
|
203
|
-
|
204
|
-
/* add buttons to the form */
|
205
|
-
buttons.apply(form, [settings, self]);
|
206
|
-
|
207
|
-
/* attach 3rd party plugin if requested */
|
208
|
-
plugin.apply(form, [settings, self]);
|
209
|
-
|
210
|
-
/* add created form to self */
|
211
|
-
$(self).append(form);
|
212
|
-
|
213
|
-
/* focus to first visible form element */
|
214
|
-
$(':input:visible:enabled:first', form).focus();
|
215
|
-
|
216
|
-
/* highlight input contents when requested */
|
217
|
-
if (settings.select) {
|
218
|
-
input.select();
|
219
|
-
}
|
220
|
-
|
221
|
-
/* discard changes if pressing esc */
|
222
|
-
input.keydown(function(e) {
|
223
|
-
if (e.keyCode == 27) {
|
224
|
-
e.preventDefault();
|
225
|
-
reset();
|
226
|
-
}
|
227
|
-
});
|
228
|
-
|
229
|
-
/* discard, submit or nothing with changes when clicking outside */
|
230
|
-
/* do nothing is usable when navigating with tab */
|
231
|
-
var t;
|
232
|
-
if ('cancel' == settings.onblur) {
|
233
|
-
input.blur(function(e) {
|
234
|
-
t = setTimeout(reset, 500);
|
235
|
-
});
|
236
|
-
} else if ('submit' == settings.onblur) {
|
237
|
-
input.blur(function(e) {
|
238
|
-
form.submit();
|
239
|
-
});
|
240
|
-
} else if ($.isFunction(settings.onblur)) {
|
241
|
-
input.blur(function(e) {
|
242
|
-
settings.onblur.apply(self, [input.val(), settings]);
|
243
|
-
});
|
244
|
-
} else {
|
245
|
-
input.blur(function(e) {
|
246
|
-
/* TODO: maybe something here */
|
247
|
-
});
|
248
|
-
}
|
249
|
-
|
250
|
-
form.submit(function(e) {
|
251
|
-
|
252
|
-
if (t) {
|
253
|
-
clearTimeout(t);
|
254
|
-
}
|
255
|
-
|
256
|
-
/* do no submit */
|
257
|
-
e.preventDefault();
|
258
|
-
|
259
|
-
/* if this input type has a call before submit hook, call it */
|
260
|
-
submit.apply(form, [settings, self]);
|
261
|
-
|
262
|
-
/* check if given target is function */
|
263
|
-
if ($.isFunction(settings.target)) {
|
264
|
-
var str = settings.target.apply(self, [input.val(), settings]);
|
265
|
-
$(self).html(str);
|
266
|
-
self.editing = false;
|
267
|
-
callback.apply(self, [self.innerHTML, settings]);
|
268
|
-
/* TODO: this is not dry */
|
269
|
-
if (!$.trim($(self).html())) {
|
270
|
-
$(self).html(settings.placeholder);
|
271
|
-
}
|
272
|
-
} else {
|
273
|
-
/* add edited content and id of edited element to POST */
|
274
|
-
var submitdata = {};
|
275
|
-
submitdata[settings.name] = input.val();
|
276
|
-
submitdata[settings.id] = self.id;
|
277
|
-
/* add extra data to be POST:ed */
|
278
|
-
if ($.isFunction(settings.submitdata)) {
|
279
|
-
$.extend(submitdata, settings.submitdata.apply(self, [self.revert, settings]));
|
280
|
-
} else {
|
281
|
-
$.extend(submitdata, settings.submitdata);
|
282
|
-
}
|
283
|
-
|
284
|
-
/* show the saving indicator */
|
285
|
-
$(self).html(settings.indicator);
|
286
|
-
$.post(settings.target, submitdata, function(str) {
|
287
|
-
$(self).html(str);
|
288
|
-
self.editing = false;
|
289
|
-
callback.apply(self, [self.innerHTML, settings]);
|
290
|
-
/* TODO: this is not dry */
|
291
|
-
if (!$.trim($(self).html())) {
|
292
|
-
$(self).html(settings.placeholder);
|
293
|
-
}
|
294
|
-
});
|
295
|
-
}
|
296
|
-
|
297
|
-
return false;
|
298
|
-
});
|
299
|
-
|
300
|
-
function reset() {
|
301
|
-
$(self).html(self.revert);
|
302
|
-
self.editing = false;
|
303
|
-
if (!$.trim($(self).html())) {
|
304
|
-
$(self).html(settings.placeholder);
|
305
|
-
}
|
306
|
-
}
|
307
|
-
|
308
|
-
});
|
309
|
-
});
|
310
|
-
|
311
|
-
};
|
312
|
-
|
313
|
-
|
314
|
-
$.editable = {
|
315
|
-
types: {
|
316
|
-
defaults: {
|
317
|
-
element : function(settings, original) {
|
318
|
-
var input = $('<input type="hidden">');
|
319
|
-
$(this).append(input);
|
320
|
-
return(input);
|
321
|
-
},
|
322
|
-
content : function(string, settings, original) {
|
323
|
-
$(':input:first', this).val(string);
|
324
|
-
},
|
325
|
-
buttons : function(settings, original) {
|
326
|
-
if (settings.submit) {
|
327
|
-
var submit = $('<input type="submit">');
|
328
|
-
submit.val(settings.submit);
|
329
|
-
$(this).append(submit);
|
330
|
-
}
|
331
|
-
if (settings.cancel) {
|
332
|
-
var cancel = $('<input type="button">');
|
333
|
-
cancel.val(settings.cancel);
|
334
|
-
$(this).append(cancel);
|
335
|
-
|
336
|
-
$(cancel).click(function() {
|
337
|
-
$(original).html(original.revert);
|
338
|
-
original.editing = false;
|
339
|
-
});
|
340
|
-
}
|
341
|
-
}
|
342
|
-
},
|
343
|
-
text: {
|
344
|
-
element : function(settings, original) {
|
345
|
-
var input = $('<input>');
|
346
|
-
if (settings.width != 'none') { input.width(settings.width); }
|
347
|
-
if (settings.height != 'none') { input.height(settings.height); }
|
348
|
-
/* https://bugzilla.mozilla.org/show_bug.cgi?id=236791 */
|
349
|
-
//input[0].setAttribute('autocomplete','off');
|
350
|
-
input.attr('autocomplete','off');
|
351
|
-
$(this).append(input);
|
352
|
-
return(input);
|
353
|
-
}
|
354
|
-
},
|
355
|
-
textarea: {
|
356
|
-
element : function(settings, original) {
|
357
|
-
var textarea = $('<textarea>');
|
358
|
-
if (settings.rows) {
|
359
|
-
textarea.attr('rows', settings.rows);
|
360
|
-
} else {
|
361
|
-
textarea.height(settings.height);
|
362
|
-
}
|
363
|
-
if (settings.cols) {
|
364
|
-
textarea.attr('cols', settings.cols);
|
365
|
-
} else {
|
366
|
-
textarea.width(settings.width);
|
367
|
-
}
|
368
|
-
$(this).append(textarea);
|
369
|
-
return(textarea);
|
370
|
-
}
|
371
|
-
},
|
372
|
-
select: {
|
373
|
-
element : function(settings, original) {
|
374
|
-
var select = $('<select>');
|
375
|
-
$(this).append(select);
|
376
|
-
return(select);
|
377
|
-
},
|
378
|
-
content : function(string, settings, original) {
|
379
|
-
if (String == string.constructor) {
|
380
|
-
eval ('var json = ' + string);
|
381
|
-
for (var key in json) {
|
382
|
-
if (!json.hasOwnProperty(key)) {
|
383
|
-
continue;
|
384
|
-
}
|
385
|
-
if ('selected' == key) {
|
386
|
-
continue;
|
387
|
-
}
|
388
|
-
var option = $('<option>').val(key).append(json[key]);
|
389
|
-
$('select', this).append(option);
|
390
|
-
}
|
391
|
-
}
|
392
|
-
/* Loop option again to set selected. IE needed this... */
|
393
|
-
$('select', this).children().each(function() {
|
394
|
-
if ($(this).val() == json['selected'] ||
|
395
|
-
$(this).text() == original.revert) {
|
396
|
-
$(this).attr('selected', 'selected');
|
397
|
-
};
|
398
|
-
});
|
399
|
-
}
|
400
|
-
}
|
401
|
-
},
|
402
|
-
|
403
|
-
/* Add new input type */
|
404
|
-
addInputType: function(name, input) {
|
405
|
-
$.editable.types[name] = input;
|
406
|
-
}
|
407
|
-
};
|
408
|
-
|
409
|
-
})(jQuery);
|