[Notice] Announcing the End of Demo Server [Read me]
File name
Commit message
Commit date
File name
Commit message
Commit date
File name
Commit message
Commit date
File name
Commit message
Commit date
2017-01-05
2017-01-05
@**
* Yona, 21st Century Project Hosting SW
*
* Copyright Yona & Yobi Authors & NAVER Corp. & NAVER LABS Corp.
* https://yona.io
**@
@(title:String, issue:Issue, issueForm: play.data.Form[Issue], commentForm: play.data.Form[Comment],project:Project)
@import org.apache.commons.lang.StringUtils
@import models.enumeration.ResourceType
@import models.enumeration.Operation
@import models.Milestone
@import utils.JodaDateUtil
@import utils.TemplateHelper._
@import utils.AccessControl._
@import play.libs.Json.toJson
@import utils.Markdown
@import models.enumeration.State
@import controllers.api.IssueApi
@getTitle(issue:Issue) = @{ issue.title }
@isFirstState(state:State) = {@if(issue.state.state == Issue.availableStates.get(0).state()){dirty}}
@urlToIssues = @{
requestHeader.headers.get("Referer") match {
case Some(u) => urlToList(u, routes.IssueApp.issues(project.owner, project.name, issue.state.state()).toString())
case _ => routes.IssueApp.issues(project.owner, project.name, "open").toString()
}
}
@urlToVote = @{
if(issue.isVotedBy(UserApp.currentUser)){
routes.VoteApp.unvote(project.owner, project.name, issue.getNumber).toString
} else {
routes.VoteApp.vote(project.owner, project.name, issue.getNumber).toString
}
}
@getVoteButtonTitle = {
@if(issue.isVotedBy(UserApp.currentUser)){
@Messages("issue.unvote.description")
} else {
@Messages("issue.vote.description")
}
}
@titleForOGTag = @{getTitle(issue) + " |:| " + issue.body.substring(0, Math.min(issue.body.length, 200))}
@parentIssueId = @{
if(issue.parent != null){
issue.parent.id
} else {
issue.id
}
}
@VOTER_AVATAR_SHOW_LIMIT = @{ 5 }
@showChildIssues(parentIssueId: Long) = {
@defining(Issue.findByParentIssueIdAndState(parentIssueId, State.OPEN)) { openChildIssues =>
@defining(Issue.findByParentIssueIdAndState(parentIssueId, State.CLOSED)) { closedChildIssues =>
@if(!openChildIssues.isEmpty || !closedChildIssues.isEmpty) {
<div class="child-issues">
@defining(Issue.finder.byId(parentIssueId)) { parentIssue =>
<div class="issue-item parent-issue"><a href="@routes.IssueApp.issue(parentIssue.project.owner, parentIssue.project.name, parentIssue.getNumber)" class="@if(parentIssue.id == issue.id){bold}">#@parentIssue.getNumber @parentIssue.title @if(parentIssue.assignee != null) {- @parentIssue.assignee.user.name}</a>
@defining(getPercent(closedChildIssues.size.toDouble, (openChildIssues.size + closedChildIssues.size).toDouble)) { percentage =>
<div class="upload-progress">
<div class="bar @if(percentage == 100){done} else {grey}" style="width: @percentage%;" title="Subtask"></div>
</div>
<span class="@if(percentage == 100){txt-green}">@if(percentage != 100){@closedChildIssues.size/}@(openChildIssues.size + closedChildIssues.size)</span>
<span class="parent-issue-state @parentIssue.state.state">@Messages("issue.state." + parentIssue.state.state)</span>
</div>
}
}
<hr class="parent-issue-delimeter"/>
@for(childIssue <- openChildIssues) {
<div class="issue-item @if(childIssue.id == issue.id){bold}">
<span class="state-label open"></span>
<a href="@routes.IssueApp.issue(childIssue.project.owner, childIssue.project.name, childIssue.getNumber)">
<span class="item-name">@childIssue.title @if(childIssue.assignee != null) {- @childIssue.assignee.user.name}</span>
</a>
</div>
}
@for(childIssue <- closedChildIssues) {
<div class="issue-item @if(childIssue.id == issue.id){bold}">
<span class="state-label closed"><i class=" yobicon-checkmark"></i></span>
<a href="@routes.IssueApp.issue(childIssue.project.owner, childIssue.project.name, childIssue.getNumber)">
<span class="item-name">@childIssue.title @if(childIssue.assignee != null) {- @childIssue.assignee.user.name}</span>
</a>
</div>
}
</div>
}
}
}
}
@conatinsCurrentUserInWatchers = @{issue.getWatchers(false).contains(UserApp.currentUser())}
@projectLayout(titleForOGTag, project, utils.MenuType.ISSUE){
@projectMenu(project, utils.MenuType.ISSUE, "main-menu-only")
<div class="page-wrap-outer">
<div class="project-page-wrap board-view">
@** Post Info **@
<div class="board-header issue">
<div class="pull-right mr10 mt10 hide-in-mobile">
<div class="date" title="@JodaDateUtil.getDateString(issue.createdDate)">
@agoOrDateString(issue.createdDate)
</div>
<span class="badge badge-issue-@issue.state.state.toLowerCase">@Messages("issue.state." + issue.state.state)</span>
</div>
<div class="title">
<strong class="board-id">#@issue.getNumber</strong> @issue.title
<div class="pull-right hide show-in-mobile" style="font-size: 0.7em">
<span class="date" title="@JodaDateUtil.getDateString(issue.createdDate)">
@agoOrDateString(issue.createdDate)
</span>
<span class="badge badge-small badge-issue-@issue.state.state.toLowerCase">@Messages("issue.state." + issue.state.state)</span>
</div>
</div>
</div>
@** Content body **@
<!--board-body-->
<div class="board-body row-fluid">
<div class="span9 span-left-pane">
<div class="author-info">
<a href="@userInfo(issue.authorLoginId)" class="usf-group">
<span class="avatar-wrap smaller">
<img src="@User.findByLoginId(issue.authorLoginId).avatarUrl(32)" width="20" height="20">
</span>
@if(issue.authorLoginId != null){
<strong class="name">@issue.getAuthor.name</strong>
<span class="loginid"> <strong>@{"@"}</strong>@issue.authorLoginId</span>
} else {
<strong class="name">@Messages("issue.noAuthor")</strong>
}
</a>
</div>
@if(StringUtils.isNotEmpty(issue.history)){
@if(UserApp.currentUser().isAnonymous){
<div class="posting-history">
<a href="@routes.UserApp.loginForm()?redirectUrl=@routes.IssueApp.issue(project.owner, project.name, issue.getNumber)" data-toggle="modal">@Messages("change.history")</a>
</div>
} else {
<div class="posting-history">
<a href="#-yona-posting-history" data-toggle="modal">@Messages("change.history")</a>
@common.partial_history(issue)
</div>
}
}
@if(StringUtils.isEmpty(issue.body)){
<div class="content empty-content"></div>
} else {
<div class="content markdown-wrap">@Html(Markdown.render(issue.body, issue.project))</div>
}
<div class="attachments" id="attachments" data-attachments="@toJson(AttachmentApp.getFileList(ResourceType.ISSUE_POST.toString(), issue.id.toString()))"></div>
<div class="board-actrow right-txt">
<div class="pull-left">
<div>
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.WATCH)) {
<button id="watch-button" type="button" class="ybtn @if(conatinsCurrentUserInWatchers) {ybtn-watching}"
data-toggle="button" data-watching="@conatinsCurrentUserInWatchers">
@if(conatinsCurrentUserInWatchers) {
@Messages("issue.unwatch")
} else {
@Messages("issue.watch")
}
</button>
}
<div id="vote" class="vote-wrap">
@if(isResourceCreatable(UserApp.currentUser, issue.asResource(), ResourceType.ISSUE_COMMENT)) {
<a href="@urlToVote" class="ybtn @if(issue.isVotedBy(UserApp.currentUser)){ybtn-watching}" title="@getVoteButtonTitle"
data-request-method="post" data-toggle="tooltip">
<span class="heart"><i class="yobicon-hearts"></i></span>
@if(issue.voters.size > 0) {
<strong class="count">@issue.voters.size</strong>
}
</a>
} else {
<span class="ybtn ybtn-disabled" style="color:#777;" data-toggle="tooltip" title="@Messages("user.login.alert")" data-login="required">
<span class="heart"><i class="yobicon-hearts"></i></span>
@if(issue.voters.size > 0) {
<strong class="count">@issue.voters.size</strong>
}
</span>
}
@if(issue.voters.size > 0){
@partial_voters(issue, 3)
}
</div>
</div>
</div>
@if(StringUtils.isNotBlank(IssueApi.TRANSLATION_API)){
<button type="button" id="translate" class="icon btn-transparent-with-fontsize-lineheight ml10" data-toggle="tooltip" title="@Messages("button.translation")"><i class="yobicon-lang"></i></button>
}
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
<button type="button" class="icon btn-transparent-with-fontsize-lineheight ml10 pt5px" data-toggle="tooltip" title="@Messages("button.edit")" onclick="window.location='@routes.IssueApp.editIssueForm(project.owner, project.name, issue.getNumber)'"><i class="yobicon-edit-2"></i></button>
} else {
<a href="@routes.IssueApp.editIssueForm(project.owner, project.name, issue.getNumber)"><button type="button" class="icon btn-transparent-with-fontsize-lineheight ml10 pt5px" data-toggle="tooltip" title="@Messages("button.show.original")"><i class="yobicon-edit-2"></i></button></a>
}
@if(issue.canBeDeleted) {
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.DELETE)) {
<a href="#deleteConfirm" data-toggle="modal">
<button type="button" class="icon btn-transparent-with-fontsize-lineheight ml6" data-toggle='tooltip' title="@Messages("button.delete")"><i class="yobicon-trash"></i></button></a>
}
} else {
<button type="button" class="icon disabled btn-transparent-with-fontsize-lineheight ml6" data-toggle='popover' data-trigger="hover" data-placement="top" data-content="@Messages("issue.can.not.be.deleted")"><i class="yobicon-trash"></i></button>
}
</div>
<div class="watcher-list"></div>
<div class="subtasks">
@if(issue.parent == null) {
@showChildIssues(issue.id)
} else {
@showChildIssues(issue.parent.id)
}
</div>
@** Comment **@
<div id="comments" class="board-comment-wrap">
<div id="timeline">
<div class="timeline-list">
@partial_comments(project, issue)
</div>
</div>
@common.commentForm(issue.asResource(), ResourceType.ISSUE_COMMENT, routes.IssueApp.newComment(project.owner, project.name, issue.getNumber).toString())
</div>
@** // Comment **@
</div>
<div class="span3 span-right-pane mb20">
<div class="issue-info">
<form id="issueUpdateForm" action="@routes.IssueApp.massUpdate(project.owner, project.name)" method="post">
<input type="hidden" name="issues[0].id" value="@issue.id" />
@**<!-- assignee -->**@
<dl>
@if(project.menuSetting.issue) {
<dd class="project-btn-item">
<a href="@routes.IssueApp.newIssueForm(project.owner, project.name)?parentIssueId=@parentIssueId" class="ybtn ybtn-success">@Messages("button.newIssue")</a>
</dd>
}
<dt>@Messages("issue.assignee")</dt>
<dd>
@defining(issue.assigneeName != null) { isAssigned =>
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
@partial_assignee(project, issue)
} else {
@if(isAssigned){
<a href="@userInfo(issue.assignee.user.loginId)" class="usf-group">
<span class="avatar-wrap smaller">
<img src="@User.findByLoginId(issue.assignee.user.loginId).avatarUrl" width="20" height="20">
</span>
<strong class="name">@issue.assigneeName</strong>
<span class="loginid"> <strong>@{"@"}</strong>@issue.assignee.user.loginId</span>
</a>
} else {
<div>
@Messages("issue.noAssignee")
</div>
}
}
}
</dd>
</dl>
@**<!-- // -->**@
@**<!-- milestones -->**@
@if(project.menuSetting.milestone) {
<dl>
<dt>@Messages("milestone")</dt>
<dd>
@if(Milestone.findByProjectId(project.id).isEmpty()){
<a href="@routes.MilestoneApp.newMilestoneForm(project.owner, project.name)"
class="ybtn ybtn-small ybtn-fullsize" target="_blank">
@Messages("milestone.menu.new")
</a>
} else {
@defining(issue.milestone != null) { hasMilestone =>
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
<select id="milestone" name="milestone.id"
data-toggle="select2" data-format="milestone" data-container-css-class="fullsize">
<option value="@Milestone.NULL_MILESTONE_ID" @if(!hasMilestone){ selected }>
@Messages("issue.noMilestone")
</option>
<optgroup label="@Messages("milestone.state.open")">
@for(milestone <- Milestone.findOpenMilestones(project.id)){
<option value="@milestone.id" data-state="@milestone.state"
@if(hasMilestone && issue.milestone.id == milestone.id){
selected
}>
@milestone.title
</option>
}
</optgroup>
<optgroup label="@Messages("milestone.state.closed")">
@for(milestone <- Milestone.findClosedMilestones(project.id)){
<option value="@milestone.id" data-state="@milestone.state"
@if(hasMilestone && issue.milestone.id == milestone.id){
selected
}>
@milestone.title
</option>
}
</optgroup>
</select>
} else {
@if(hasMilestone){
<a href="@routes.MilestoneApp.milestone(project.owner, project.name, issue.milestone.id)">
@issue.milestone.title
</a>
} else {
@Messages("issue.noMilestone")
}
}
}
}
</dd>
</dl>
}
@**<!-- // -->**@
<dl>
<dt>
@Messages("issue.dueDate")
<span class="duedate-status @if(issue.isOverDueDate) {overdue}">
@if(issue.dueDate != null) {
@if(issue.isOverDueDate) {
(@Messages("issue.dueDate.overdue"))
} else {
(@issue.until)
}
}
</span>
</dt>
<dd>
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
<div class="search search-bar">
<input type="text" name="dueDate" value="@issue.getDueDateString" class="textbox full" autocomplete="off" data-toggle="calendar">
<button type="button" class="search-btn btn-calendar"><i class="yobicon-calendar2"></i></button>
</div>
} else {
@if(issue.dueDate != null) {
@issue.getDueDateString
} else {
@Messages("issue.noDuedate")
}
}
</dd>
</dl>
@**<!-- labels -->**@
@if(!IssueLabel.findByProject(project).isEmpty){
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)){
@partial_select_label(IssueLabel.findByProject(project), issue.getLabelIds, "", "", project)
} else {
@partial_show_selected_label(issue.labels.toList, routes.IssueApp.issues(project.owner, project.name, issue.state.state(), "html", 1).toString)
}
}
@**<!-- // -->**@
</form>
</div>
</div>
</div>
<div class="board-footer">
@help.keymap("issueDetail", project)
</div>
</div>
<script type="text/x-jquery-tmpl" id="tplAttachedFile"><!--
--><li class="attached-file" data-name="${fileName}" data-href="${fileHref}" data-mime="${mimeType}" data-size="${fileSize}">
<strong>${fileName}(${fileSizeReadable})${notice}</strong><!--
--><a class="attached-delete"><i class="ico btn-delete"></i></a></li>
</script>
@** Confirm to delete post **@
<div id="deleteConfirm" class="modal hide fade">
<div class="modal-header">
<button type="button" class="close" data-dismiss="modal">×</button>
<h3>@Messages("issue.delete")</h3>
</div>
<div class="modal-body">
<p>@Messages("post.delete.confirm")</p>
</div>
<div class="modal-footer">
<button type="button" class="ybtn ybtn-danger" data-request-method="delete" data-request-uri="@routes.IssueApp.deleteIssue(project.owner, project.name, issue.getNumber)">@Messages("button.yes")</button>
<button type="button" class="ybtn" data-dismiss="modal">@Messages("button.no")</button>
</div>
</div>
</div>
@common.markdown(project)
@common.commentDeleteModal()
@common.select2()
@common.calendar()
<link rel="stylesheet" type="text/css" media="screen" href="@routes.IssueLabelApp.labelStyles(project.owner, project.name)">
<link rel="stylesheet" type="text/css" media="screen" href="@routes.Assets.at("javascripts/lib/atjs/jquery.atwho.css")">
<script type="text/javascript" src="@routes.Assets.at("javascripts/lib/atjs/jquery.caret.min.js")"></script>
<script type="text/javascript" src="@routes.Assets.at("javascripts/lib/atjs/jquery.atwho.js")"></script>
<script type="text/javascript" src="@routes.Assets.at("javascripts/service/yona.issue.Assginee.js")"></script>
<script type="text/javascript">
$(function(){
// yobi.issue.View
$yobi.loadModule("issue.View", {
"issueId" : "@issue.id",
"nextState": "@issue.nextState().toString.toLowerCase",
"urls" : {
"watch" : "@routes.WatchApp.watch(issue.asResource.asParameter)",
"unwatch" : "@routes.WatchApp.unwatch(issue.asResource.asParameter)",
"timeline" : "@routes.IssueApp.timeline(project.owner, project.name, issue.getNumber)",
"nextState" : "@routes.IssueApp.nextState(project.owner, project.name, issue.getNumber)",
"massUpdate": "@routes.IssueApp.massUpdate(project.owner, project.name)"
}
});
// yobi.ShortcutKey
yobi.ShortcutKey.setKeymapLink({
"L": "@urlToIssues"
@if(project.menuSetting.issue) {
,"N": "@routes.IssueApp.newIssueForm(project.owner, project.name)"
}
@if(isAllowed(UserApp.currentUser(), issue.asResource(), Operation.UPDATE)) {
,"E": "@routes.IssueApp.editIssueForm(project.owner, project.name, issue.getNumber)"
}
});
// yobi.Mention
yobi.Mention({
"target": 'textarea[id^=editor-]',
"url" : "@Html(routes.ProjectApp.mentionList(project.owner, project.name, issue.getNumber, issue.asResource().getType.resource()).toString())"
});
// detect comment which contains mention at me
$(".comment-body:contains('@UserApp.currentUser().loginId')").closest(".comment").addClass("mentioned");
});
</script>
<script>
$(function () {
yonaAssgineeModule(
"@api.routes.IssueApi.findAssignableUsers(project.owner, project.name, issue.getNumber)",
"@api.routes.IssueApi.updateAssginees(project.owner, project.name, issue.getNumber)",
"@Messages("issue.assignee")"
);
$('#translate').one('click', function (e) {
var data = {
owner: "@project.owner",
projectName: "@project.name",
type: "issue",
number: "@issue.getNumber"
};
$.ajax({
url: "/-_-api/v1/translation",
data: JSON.stringify(data),
type: "POST",
dataType: "json",
contentType: "application/json"
}).done(function (data) {
$(".markdown-wrap").first().html(data.translated);
$(this).attr("disabled", true);
});
});
$('.comment-translate').one('click', function (e) {
var payload = {
owner: "@project.owner",
projectName: "@project.name",
type: "issue-comment",
number: $(this).data("commentId")
};
$.ajax({
url: "/-_-api/v1/translation",
data: JSON.stringify(payload),
type: "POST",
dataType: "json",
contentType: "application/json"
}).done(function (data) {
$("#comment-body-" + payload.number).find(".markdown-wrap").html(data.translated)
$(this).attr("disabled", true);
});
});
});
</script>
}