
hudson.model.BuildTimelineWidget Maven / Gradle / Ivy
Show all versions of hudson-core Show documentation
/*
* The MIT License
*
* Copyright (c) 2010, InfraDNA, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package hudson.model;
import hudson.Functions;
import hudson.util.RunList;
import java.util.TimeZone;
import net.sf.json.JSONArray;
import net.sf.json.JSONObject;
import net.sf.json.JsonConfig;
import net.sf.json.processors.JsonValueProcessor;
import org.kohsuke.stapler.HttpResponse;
import org.kohsuke.stapler.StaplerRequest;
import org.kohsuke.stapler.StaplerResponse;
import javax.servlet.ServletException;
import java.io.IOException;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import org.kohsuke.stapler.QueryParameter;
/**
* UI widget for showing the SMILE timeline control.
*
*
* Return this from your "getTimeline" method.
*
* @author Kohsuke Kawaguchi, Winston Prakash
* @since 1.372
*/
public class BuildTimelineWidget {
protected final RunList> builds;
public BuildTimelineWidget(RunList> builds) {
this.builds = builds;
}
public Run, ?> getFirstBuild() {
return builds.getFirstBuild();
}
public Run, ?> getLastBuild() {
return builds.getLastBuild();
}
/**
* Get timezone offset with Daylight time saving support
*
* @return int value
*/
public int getTimeZoneOffset() {
return TimeZone.getDefault().getOffset(System.currentTimeMillis()) / 3600000;
}
public TimelineEventList doData(StaplerRequest req, @QueryParameter long min, @QueryParameter long max) throws IOException {
TimelineEventList result = new TimelineEventList();
for (Run r : builds.byTimestamp(min, max)) {
Event e = new Event();
e.start = r.getTime();
e.end = new Date(r.timestamp + r.getDuration());
e.title = r.getFullDisplayName();
// what to put in the description?
// e.description = "Longish description of event "+r.getFullDisplayName();
// e.durationEvent = true;
e.link = req.getContextPath() + '/' + r.getUrl();
BallColor c = r.getIconColor();
e.color = String.format("#%06X", c.getBaseColor().darker().getRGB() & 0xFFFFFF);
e.classname = "event-" + c.noAnime().toString() + " " + (c.isAnimated() ? "animated" : "");
result.add(e);
}
return result;
}
/**
* List of {@link Event} that the timeline component will display.
*/
private static class TimelineEventList extends ArrayList implements HttpResponse {
/**
* Renders HTTP response.
*/
public void generateResponse(StaplerRequest req, StaplerResponse rsp, Object node) throws IOException, ServletException {
// Date needs to be converted into iso-8601 date format.
JsonConfig config = new JsonConfig();
config.registerJsonValueProcessor(Date.class, new JsonValueProcessor() {
public synchronized Object processArrayValue(Object value, JsonConfig jsonConfig) {
if (value != null){
DateFormat dateFormat = new SimpleDateFormat("MMM dd yyyy HH:mm:ss 'GMT'Z", Functions.getClientLocale());
return dateFormat.format(value);
}
return null;
}
public Object processObjectValue(String key, Object value, JsonConfig jsonConfig) {
return processArrayValue(value, jsonConfig);
}
});
JSONObject o = new JSONObject();
o.put("events", JSONArray.fromObject(this, config));
rsp.setContentType("application/javascript;charset=UTF-8");
o.write(rsp.getWriter());
}
}
/**
* Event data to be rendered on timeline.
* See http://code.google.com/p/simile-widgets/wiki/Timeline_EventSources
*
* This is bound to JSON and sent to the client-side JavaScript.
*/
public static class Event {
//TODO: review and check whether we can do it private
public Date start;
public Date end;
public String title, description;
/**
* If true, the event occurs over a time duration. No icon. The event will be
* drawn as a dark blue tape. The tape color is set with the color attribute.
* Default color is #58A0DC
*
* If false (default), the event is focused on a specific "instant" (shown with the icon).
* The event will be drawn as a blue dot icon (default) with a pale blue tape.
* The tape is the default color (or color attribute color), with opacity
* set to 20. To change the opacity, change the theme's instant: {impreciseOpacity: 20}
* value. Maximum 100.
*/
public Boolean durationEvent;
/**
* Url. The bubble's title text be a hyper-link to this address.
*/
public String link;
/**
* Color of the text and tape (duration events) to display in the timeline.
* If the event has durationEvent = false, then the bar's opacity will
* be applied (default 20%). See durationEvent, above.
*/
public String color;
/**
* CSS class name.
*/
public String classname;
public Date getStart() {
return start;
}
public Date getEnd() {
return end;
}
public String getTitle() {
return title;
}
public String getDescription() {
return description;
}
public Boolean getDurationEvent() {
return durationEvent;
}
public String getLink() {
return link;
}
public String getColor() {
return color;
}
public String getClassname() {
return classname;
}
}
}